diff --git a/DEPS b/DEPS
index e9984c8..bb2068c 100644
--- a/DEPS
+++ b/DEPS
@@ -105,11 +105,11 @@
   # 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': '903fb041a2f86e6273afa501e8e15335cd4e9972',
+  'skia_revision': '734b2a859784af0b16dbbe07570e6d2f5da34e76',
   # 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': '22a88d89e037d3e95dc320f09544b16e9959a407',
+  'v8_revision': '696e3951665ffc632c6a154a1d2b14c3b3e92212',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling swarming_client
   # and whatever else without interference from each other.
@@ -117,7 +117,7 @@
   # 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': '522095f7d7fe04bf1479a20a9707372d988a9da2',
+  'angle_revision': '06235df9ff91acdf3cf50f2a6febb38b5e26cc6b',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling build tools
   # and whatever else without interference from each other.
@@ -125,11 +125,11 @@
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling SwiftShader
   # and whatever else without interference from each other.
-  'swiftshader_revision': '8fb6f6a129f5ed809db7ba3a8bc862dd5e7bc75d',
+  'swiftshader_revision': 'be5fe7797f1e1dbf96f3f337604a4711e871a30a',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling PDFium
   # and whatever else without interference from each other.
-  'pdfium_revision': 'c62aa207e9acb919c33df5f3694fe159619dda86',
+  'pdfium_revision': 'aa50728980036f07fda232cea974fd80c89b7cb7',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling openmax_dl
   # and whatever else without interference from each other.
@@ -165,7 +165,7 @@
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling catapult
   # and whatever else without interference from each other.
-  'catapult_revision': '2a64e6b6d554193f50f82d9557442615457365d9',
+  'catapult_revision': 'ca2b32c844637750526112f1af7fbb28b17a8f83',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling libFuzzer
   # and whatever else without interference from each other.
@@ -539,7 +539,7 @@
 
   # Build tools for Chrome OS. Note: This depends on third_party/pyelftools.
   'src/third_party/chromite': {
-      'url': Var('chromium_git') + '/chromiumos/chromite.git' + '@' + 'b96253cab7d9acd6c40a22e1f8f192e61fbc027a',
+      'url': Var('chromium_git') + '/chromiumos/chromite.git' + '@' + 'dad47a9531e45fd79083ff2c0919e39cbafef28c',
       'condition': 'checkout_linux',
   },
 
@@ -912,7 +912,7 @@
   },
 
   'src/third_party/perfetto':
-    Var('android_git') + '/platform/external/perfetto.git' + '@' +  'ee5a26d974beef5a5bdab7d6cf0e97785a7e1c48',
+    Var('android_git') + '/platform/external/perfetto.git' + '@' +  '43ee74f7bd72edebbe5c7642d4df956dbf52bd03',
 
   'src/third_party/perl': {
       'url': Var('chromium_git') + '/chromium/deps/perl.git' + '@' + 'ac0d98b5cee6c024b0cffeb4f8f45b6fc5ccdb78',
@@ -1034,7 +1034,7 @@
     Var('chromium_git') + '/external/khronosgroup/webgl.git' + '@' + '21dbf06b5aa6c7dc8cf56314d4a3f96f57956c53',
 
   'src/third_party/webrtc':
-    Var('webrtc_git') + '/src.git' + '@' + '25cc8ad198731ff8eee35859d82919949afbdff7',
+    Var('webrtc_git') + '/src.git' + '@' + 'e09e6ef947d7bb6c29fa36c331bd855baacfafbc',
 
   'src/third_party/xdg-utils': {
       'url': Var('chromium_git') + '/chromium/deps/xdg-utils.git' + '@' + 'd80274d5869b17b8c9067a1022e4416ee7ed5e0d',
diff --git a/android_webview/system_webview_apk_tmpl.gni b/android_webview/system_webview_apk_tmpl.gni
index 2c0f997..09fd7aa 100644
--- a/android_webview/system_webview_apk_tmpl.gni
+++ b/android_webview/system_webview_apk_tmpl.gni
@@ -2,6 +2,7 @@
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
 
+import("//base/android/proguard/proguard.gni")
 import("//build/config/android/config.gni")
 import("//build/config/android/rules.gni")
 import("//build/config/locales.gni")
@@ -53,8 +54,14 @@
         "//android_webview/apk/java/proguard.flags",
         "//base/android/proguard/chromium_apk.flags",
         "//base/android/proguard/chromium_code.flags",
-        "//base/android/proguard/disable_all_obfuscation.flags",
       ]
+      if (enable_proguard_obfuscation) {
+        proguard_configs +=
+            [ "//base/android/proguard/enable_obfuscation.flags" ]
+      } else {
+        proguard_configs +=
+            [ "//base/android/proguard/disable_all_obfuscation.flags" ]
+      }
       png_to_webp = true
     }
     command_line_flags_file = "webview-command-line"
diff --git a/base/allocator/partition_allocator/address_space_randomization.cc b/base/allocator/partition_allocator/address_space_randomization.cc
index a7e17c7c..8701f56 100644
--- a/base/allocator/partition_allocator/address_space_randomization.cc
+++ b/base/allocator/partition_allocator/address_space_randomization.cc
@@ -22,7 +22,7 @@
 
 // This is the same PRNG as used by tcmalloc for mapping address randomness;
 // see http://burtleburtle.net/bob/rand/smallprng.html
-struct ranctx {
+struct RandomContext {
   subtle::SpinLock lock;
   bool initialized;
   uint32_t a;
@@ -31,11 +31,12 @@
   uint32_t d;
 };
 
-static LazyInstance<ranctx>::Leaky s_ranctx = LAZY_INSTANCE_INITIALIZER;
+static LazyInstance<RandomContext>::Leaky s_RandomContext =
+    LAZY_INSTANCE_INITIALIZER;
 
 #define rot(x, k) (((x) << (k)) | ((x) >> (32 - (k))))
 
-uint32_t ranvalInternal(ranctx* x) {
+uint32_t RandomValueInternal(RandomContext* x) {
   uint32_t e = x->a - rot(x->b, 27);
   x->a = x->b ^ rot(x->c, 17);
   x->b = x->c + x->d;
@@ -46,7 +47,7 @@
 
 #undef rot
 
-uint32_t ranval(ranctx* x) {
+uint32_t RandomValue(RandomContext* x) {
   subtle::SpinLock::Guard guard(x->lock);
   if (UNLIKELY(!x->initialized)) {
     const uint64_t r1 = RandUint64();
@@ -60,13 +61,13 @@
     x->initialized = true;
   }
 
-  return ranvalInternal(x);
+  return RandomValueInternal(x);
 }
 
 }  // namespace
 
 void SetRandomPageBaseSeed(int64_t seed) {
-  ranctx* x = s_ranctx.Pointer();
+  RandomContext* x = s_RandomContext.Pointer();
   subtle::SpinLock::Guard guard(x->lock);
   // Set RNG to initial state.
   x->initialized = true;
@@ -75,11 +76,12 @@
 }
 
 void* GetRandomPageBase() {
-  uintptr_t random = static_cast<uintptr_t>(ranval(s_ranctx.Pointer()));
+  uintptr_t random =
+      static_cast<uintptr_t>(RandomValue(s_RandomContext.Pointer()));
 
 #if defined(ARCH_CPU_64_BITS)
   random <<= 32ULL;
-  random |= static_cast<uintptr_t>(ranval(s_ranctx.Pointer()));
+  random |= static_cast<uintptr_t>(RandomValue(s_RandomContext.Pointer()));
 
 // The kASLRMask and kASLROffset constants will be suitable for the
 // OS and build configuration.
diff --git a/base/allocator/partition_allocator/partition_alloc.cc b/base/allocator/partition_allocator/partition_alloc.cc
index 8554673..e730aac 100644
--- a/base/allocator/partition_allocator/partition_alloc.cc
+++ b/base/allocator/partition_allocator/partition_alloc.cc
@@ -146,7 +146,7 @@
   // code simpler and the structures more generic.
   size_t i, j;
   size_t current_size = kGenericSmallestBucket;
-  size_t currentIncrement =
+  size_t current_increment =
       kGenericSmallestBucket >> kGenericNumBucketsPerOrderBits;
   internal::PartitionBucket* bucket = &this->buckets[0];
   for (i = 0; i < kGenericNumBucketedOrders; ++i) {
@@ -155,40 +155,40 @@
       // Disable psuedo buckets so that touching them faults.
       if (current_size % kGenericSmallestBucket)
         bucket->active_pages_head = nullptr;
-      current_size += currentIncrement;
+      current_size += current_increment;
       ++bucket;
     }
-    currentIncrement <<= 1;
+    current_increment <<= 1;
   }
   DCHECK(current_size == 1 << kGenericMaxBucketedOrder);
   DCHECK(bucket == &this->buckets[0] + kGenericNumBuckets);
 
   // Then set up the fast size -> bucket lookup table.
   bucket = &this->buckets[0];
-  internal::PartitionBucket** bucketPtr = &this->bucket_lookups[0];
+  internal::PartitionBucket** bucket_ptr = &this->bucket_lookups[0];
   for (order = 0; order <= kBitsPerSizeT; ++order) {
     for (j = 0; j < kGenericNumBucketsPerOrder; ++j) {
       if (order < kGenericMinBucketedOrder) {
         // Use the bucket of the finest granularity for malloc(0) etc.
-        *bucketPtr++ = &this->buckets[0];
+        *bucket_ptr++ = &this->buckets[0];
       } else if (order > kGenericMaxBucketedOrder) {
-        *bucketPtr++ = internal::PartitionBucket::get_sentinel_bucket();
+        *bucket_ptr++ = internal::PartitionBucket::get_sentinel_bucket();
       } else {
-        internal::PartitionBucket* validBucket = bucket;
+        internal::PartitionBucket* valid_bucket = bucket;
         // Skip over invalid buckets.
-        while (validBucket->slot_size % kGenericSmallestBucket)
-          validBucket++;
-        *bucketPtr++ = validBucket;
+        while (valid_bucket->slot_size % kGenericSmallestBucket)
+          valid_bucket++;
+        *bucket_ptr++ = valid_bucket;
         bucket++;
       }
     }
   }
   DCHECK(bucket == &this->buckets[0] + kGenericNumBuckets);
-  DCHECK(bucketPtr == &this->bucket_lookups[0] +
-                          ((kBitsPerSizeT + 1) * kGenericNumBucketsPerOrder));
+  DCHECK(bucket_ptr == &this->bucket_lookups[0] +
+                           ((kBitsPerSizeT + 1) * kGenericNumBucketsPerOrder));
   // And there's one last bucket lookup that will be hit for e.g. malloc(-1),
   // which tries to overflow to a non-existant order.
-  *bucketPtr = internal::PartitionBucket::get_sentinel_bucket();
+  *bucket_ptr = internal::PartitionBucket::get_sentinel_bucket();
 }
 
 bool PartitionReallocDirectMappedInPlace(PartitionRootGeneric* root,
@@ -221,9 +221,9 @@
       return false;
 
     // Shrink by decommitting unneeded pages and making them inaccessible.
-    size_t decommitSize = current_size - new_size;
-    root->DecommitSystemPages(char_ptr + new_size, decommitSize);
-    CHECK(SetSystemPagesAccess(char_ptr + new_size, decommitSize,
+    size_t decommit_size = current_size - new_size;
+    root->DecommitSystemPages(char_ptr + new_size, decommit_size);
+    CHECK(SetSystemPagesAccess(char_ptr + new_size, decommit_size,
                                PageInaccessible));
   } else if (new_size <=
              internal::PartitionDirectMapExtent::FromPage(page)->map_size) {
@@ -303,7 +303,7 @@
   // new size is a significant percentage smaller. We could do the same if we
   // determine it is a win.
   if (actual_new_size == actual_old_size) {
-    // Trying to allocate a block of size new_size would give us a block of
+    // Trying to allocate a block of size |new_size| would give us a block of
     // the same size as the one we've already got, so re-use the allocation
     // after updating statistics (and cookies, if present).
     page->set_raw_size(internal::PartitionCookieSizeAdjustAdd(new_size));
@@ -341,6 +341,13 @@
   return PartitionReallocGenericFlags(this, 0, ptr, new_size, type_name);
 }
 
+void* PartitionRootGeneric::TryRealloc(void* ptr,
+                                       size_t new_size,
+                                       const char* type_name) {
+  return PartitionReallocGenericFlags(this, PartitionAllocReturnNull, ptr,
+                                      new_size, type_name);
+}
+
 static size_t PartitionPurgePage(internal::PartitionPage* page, bool discard) {
   const internal::PartitionBucket* bucket = page->bucket;
   size_t slot_size = bucket->slot_size;
@@ -352,12 +359,12 @@
 
   size_t raw_size = page->get_raw_size();
   if (raw_size) {
-    uint32_t usedBytes = static_cast<uint32_t>(RoundUpToSystemPage(raw_size));
-    discardable_bytes = bucket->slot_size - usedBytes;
+    uint32_t used_bytes = static_cast<uint32_t>(RoundUpToSystemPage(raw_size));
+    discardable_bytes = bucket->slot_size - used_bytes;
     if (discardable_bytes && discard) {
       char* ptr =
           reinterpret_cast<char*>(internal::PartitionPage::ToPointer(page));
-      ptr += usedBytes;
+      ptr += used_bytes;
       DiscardSystemPages(ptr, discardable_bytes);
     }
     return discardable_bytes;
@@ -380,23 +387,23 @@
   // are not in use.
   for (internal::PartitionFreelistEntry* entry = page->freelist_head; entry;
        /**/) {
-    size_t slotIndex = (reinterpret_cast<char*>(entry) - ptr) / slot_size;
-    DCHECK(slotIndex < num_slots);
-    slot_usage[slotIndex] = 0;
+    size_t slot_index = (reinterpret_cast<char*>(entry) - ptr) / slot_size;
+    DCHECK(slot_index < num_slots);
+    slot_usage[slot_index] = 0;
     entry = internal::PartitionFreelistEntry::Transform(entry->next);
 #if !defined(OS_WIN)
-    // If we have a slot where the masked freelist entry is 0, we can
-    // actually discard that freelist entry because touching a discarded
-    // page is guaranteed to return original content or 0.
-    // (Note that this optimization won't fire on big endian machines
-    // because the masking function is negation.)
+    // If we have a slot where the masked freelist entry is 0, we can actually
+    // discard that freelist entry because touching a discarded page is
+    // guaranteed to return original content or 0. (Note that this optimization
+    // won't fire on big-endian machines because the masking function is
+    // negation.)
     if (!internal::PartitionFreelistEntry::Transform(entry))
-      last_slot = slotIndex;
+      last_slot = slot_index;
 #endif
   }
 
-  // If the slot(s) at the end of the slot span are not in used, we can
-  // truncate them entirely and rewrite the freelist.
+  // If the slot(s) at the end of the slot span are not in used, we can truncate
+  // them entirely and rewrite the freelist.
   size_t truncated_slots = 0;
   while (!slot_usage[num_slots - 1]) {
     truncated_slots++;
@@ -411,8 +418,8 @@
     char* end_ptr = begin_ptr + (slot_size * truncated_slots);
     begin_ptr = reinterpret_cast<char*>(
         RoundUpToSystemPage(reinterpret_cast<size_t>(begin_ptr)));
-    // We round the end pointer here up and not down because we're at the
-    // end of a slot span, so we "own" all the way up the page boundary.
+    // We round the end pointer here up and not down because we're at the end of
+    // a slot span, so we "own" all the way up the page boundary.
     end_ptr = reinterpret_cast<char*>(
         RoundUpToSystemPage(reinterpret_cast<size_t>(end_ptr)));
     DCHECK(end_ptr <= ptr + bucket->get_bytes_per_span());
@@ -426,16 +433,16 @@
       page->num_unprovisioned_slots += static_cast<uint16_t>(truncated_slots);
       // Rewrite the freelist.
       internal::PartitionFreelistEntry** entry_ptr = &page->freelist_head;
-      for (size_t slotIndex = 0; slotIndex < num_slots; ++slotIndex) {
-        if (slot_usage[slotIndex])
+      for (size_t slot_index = 0; slot_index < num_slots; ++slot_index) {
+        if (slot_usage[slot_index])
           continue;
         auto* entry = reinterpret_cast<internal::PartitionFreelistEntry*>(
-            ptr + (slot_size * slotIndex));
+            ptr + (slot_size * slot_index));
         *entry_ptr = internal::PartitionFreelistEntry::Transform(entry);
         entry_ptr = reinterpret_cast<internal::PartitionFreelistEntry**>(entry);
         num_new_entries++;
 #if !defined(OS_WIN)
-        last_slot = slotIndex;
+        last_slot = slot_index;
 #endif
       }
       // Terminate the freelist chain.
@@ -449,16 +456,15 @@
     }
   }
 
-  // Next, walk the slots and for any not in use, consider where the system
-  // page boundaries occur. We can release any system pages back to the
-  // system as long as we don't interfere with a freelist pointer or an
-  // adjacent slot.
+  // Next, walk the slots and for any not in use, consider where the system page
+  // boundaries occur. We can release any system pages back to the system as
+  // long as we don't interfere with a freelist pointer or an adjacent slot.
   for (size_t i = 0; i < num_slots; ++i) {
     if (slot_usage[i])
       continue;
     // The first address we can safely discard is just after the freelist
-    // pointer. There's one quirk: if the freelist pointer is actually a
-    // null, we can discard that pointer value too.
+    // pointer. There's one quirk: if the freelist pointer is actually NULL, we
+    // can discard that pointer value too.
     char* begin_ptr = ptr + (i * slot_size);
     char* end_ptr = begin_ptr + slot_size;
 #if !defined(OS_WIN)
@@ -496,9 +502,9 @@
   if (flags & PartitionPurgeDecommitEmptyPages)
     DecommitEmptyPages();
   // We don't currently do anything for PartitionPurgeDiscardUnusedSystemPages
-  // here because that flag is only useful for allocations >= system page
-  // size. We only have allocations that large inside generic partitions
-  // at the moment.
+  // here because that flag is only useful for allocations >= system page size.
+  // We only have allocations that large inside generic partitions at the
+  // moment.
 }
 
 void PartitionRootGeneric::PurgeMemory(int flags) {
@@ -553,9 +559,8 @@
   DCHECK(!bucket->is_direct_mapped());
   stats_out->is_valid = false;
   // If the active page list is empty (==
-  // internal::PartitionPage::get_sentinel_page()),
-  // the bucket might still need to be reported if it has a list of empty,
-  // decommitted or full pages.
+  // internal::PartitionPage::get_sentinel_page()), the bucket might still need
+  // to be reported if it has a list of empty, decommitted or full pages.
   if (bucket->active_pages_head ==
           internal::PartitionPage::get_sentinel_page() &&
       !bucket->empty_pages_head && !bucket->decommitted_pages_head &&
@@ -693,10 +698,10 @@
     memory_stats = std::unique_ptr<PartitionBucketMemoryStats[]>(
         new PartitionBucketMemoryStats[kMaxReportableBuckets]);
 
-  const size_t partitionNumBuckets = this->num_buckets;
-  DCHECK(partitionNumBuckets <= kMaxReportableBuckets);
+  const size_t partition_num_buckets = this->num_buckets;
+  DCHECK(partition_num_buckets <= kMaxReportableBuckets);
 
-  for (size_t i = 0; i < partitionNumBuckets; ++i) {
+  for (size_t i = 0; i < partition_num_buckets; ++i) {
     PartitionBucketMemoryStats bucket_stats = {0};
     PartitionDumpBucketStats(&bucket_stats, &this->buckets()[i]);
     if (bucket_stats.is_valid) {
@@ -716,7 +721,7 @@
     // PartitionsDumpBucketStats is called after collecting stats because it
     // can use PartitionRoot::Alloc() to allocate and this can affect the
     // statistics.
-    for (size_t i = 0; i < partitionNumBuckets; ++i) {
+    for (size_t i = 0; i < partition_num_buckets; ++i) {
       if (memory_stats[i].is_valid)
         dumper->PartitionsDumpBucketStats(partition_name, &memory_stats[i]);
     }
diff --git a/base/allocator/partition_allocator/partition_alloc.h b/base/allocator/partition_allocator/partition_alloc.h
index 79d09053..0294ef92 100644
--- a/base/allocator/partition_allocator/partition_alloc.h
+++ b/base/allocator/partition_allocator/partition_alloc.h
@@ -148,6 +148,9 @@
   ALWAYS_INLINE void Free(void* ptr);
 
   NOINLINE void* Realloc(void* ptr, size_t new_size, const char* type_name);
+  // Overload that may return nullptr if reallocation isn't possible. In this
+  // case, |ptr| remains valid.
+  NOINLINE void* TryRealloc(void* ptr, size_t new_size, const char* type_name);
 
   ALWAYS_INLINE size_t ActualSize(size_t size);
 
diff --git a/base/allocator/partition_allocator/partition_alloc_unittest.cc b/base/allocator/partition_allocator/partition_alloc_unittest.cc
index afff0906b..ad91965 100644
--- a/base/allocator/partition_allocator/partition_alloc_unittest.cc
+++ b/base/allocator/partition_allocator/partition_alloc_unittest.cc
@@ -182,7 +182,13 @@
     }
   }
 
-  void DoReturnNullTest(size_t allocSize, bool use_realloc) {
+  enum ReturnNullTestMode {
+    kPartitionAllocGenericFlags,
+    kPartitionReallocGenericFlags,
+    kPartitionRootGenericTryRealloc,
+  };
+
+  void DoReturnNullTest(size_t alloc_size, ReturnNullTestMode mode) {
     // TODO(crbug.com/678782): Where necessary and possible, disable the
     // platform's OOM-killing behavior. OOM-killing makes this test flaky on
     // low-memory devices.
@@ -195,30 +201,42 @@
     ASSERT_TRUE(SetAddressSpaceLimit());
 
     // Work out the number of allocations for 6 GB of memory.
-    const int numAllocations = (6 * 1024 * 1024) / (allocSize / 1024);
+    const int num_allocations = (6 * 1024 * 1024) / (alloc_size / 1024);
 
     void** ptrs = reinterpret_cast<void**>(generic_allocator.root()->Alloc(
-        numAllocations * sizeof(void*), type_name));
+        num_allocations * sizeof(void*), type_name));
     int i;
 
-    for (i = 0; i < numAllocations; ++i) {
-      if (use_realloc) {
-        ptrs[i] = PartitionAllocGenericFlags(
-            generic_allocator.root(), PartitionAllocReturnNull, 1, type_name);
-        ptrs[i] = PartitionReallocGenericFlags(generic_allocator.root(),
+    for (i = 0; i < num_allocations; ++i) {
+      switch (mode) {
+        case kPartitionAllocGenericFlags: {
+          ptrs[i] = PartitionAllocGenericFlags(generic_allocator.root(),
                                                PartitionAllocReturnNull,
-                                               ptrs[i], allocSize, type_name);
-      } else {
-        ptrs[i] = PartitionAllocGenericFlags(generic_allocator.root(),
-                                             PartitionAllocReturnNull,
-                                             allocSize, type_name);
+                                               alloc_size, type_name);
+          break;
+        }
+        case kPartitionReallocGenericFlags: {
+          ptrs[i] = PartitionAllocGenericFlags(
+              generic_allocator.root(), PartitionAllocReturnNull, 1, type_name);
+          ptrs[i] = PartitionReallocGenericFlags(
+              generic_allocator.root(), PartitionAllocReturnNull, ptrs[i],
+              alloc_size, type_name);
+          break;
+        }
+        case kPartitionRootGenericTryRealloc: {
+          ptrs[i] = PartitionAllocGenericFlags(
+              generic_allocator.root(), PartitionAllocReturnNull, 1, type_name);
+          ptrs[i] = generic_allocator.root()->TryRealloc(ptrs[i], alloc_size,
+                                                         type_name);
+        }
       }
+
       if (!i)
         EXPECT_TRUE(ptrs[0]);
       if (!ptrs[i]) {
         ptrs[i] = PartitionAllocGenericFlags(generic_allocator.root(),
                                              PartitionAllocReturnNull,
-                                             allocSize, type_name);
+                                             alloc_size, type_name);
         EXPECT_FALSE(ptrs[i]);
         break;
       }
@@ -226,14 +244,14 @@
 
     // We shouldn't succeed in allocating all 6 GB of memory. If we do, then
     // we're not actually testing anything here.
-    EXPECT_LT(i, numAllocations);
+    EXPECT_LT(i, num_allocations);
 
     // Free, reallocate and free again each block we allocated. We do this to
     // check that freeing memory also works correctly after a failed allocation.
     for (--i; i >= 0; --i) {
       generic_allocator.root()->Free(ptrs[i]);
       ptrs[i] = PartitionAllocGenericFlags(generic_allocator.root(),
-                                           PartitionAllocReturnNull, allocSize,
+                                           PartitionAllocReturnNull, alloc_size,
                                            type_name);
       EXPECT_TRUE(ptrs[i]);
       generic_allocator.root()->Free(ptrs[i]);
@@ -335,11 +353,11 @@
 // Check that the most basic of allocate / free pairs work.
 TEST_F(PartitionAllocTest, Basic) {
   PartitionBucket* bucket = &allocator.root()->buckets()[kTestBucketIndex];
-  PartitionPage* seedPage = PartitionPage::get_sentinel_page();
+  PartitionPage* seed_page = PartitionPage::get_sentinel_page();
 
   EXPECT_FALSE(bucket->empty_pages_head);
   EXPECT_FALSE(bucket->decommitted_pages_head);
-  EXPECT_EQ(seedPage, bucket->active_pages_head);
+  EXPECT_EQ(seed_page, bucket->active_pages_head);
   EXPECT_EQ(nullptr, bucket->active_pages_head->next_page);
 
   void* ptr = allocator.root()->Alloc(kTestAllocSize, type_name);
@@ -467,9 +485,9 @@
   PartitionFree(ptr);
   // Trying to allocate at this time should cause us to cycle around to page2
   // and find the recently freed slot.
-  char* newPtr = reinterpret_cast<char*>(
+  char* new_ptr = reinterpret_cast<char*>(
       allocator.root()->Alloc(kTestAllocSize, type_name));
-  EXPECT_EQ(ptr, newPtr);
+  EXPECT_EQ(ptr, new_ptr);
   EXPECT_EQ(page2, bucket->active_pages_head);
   EXPECT_EQ(page3, page2->next_page);
 
@@ -479,9 +497,9 @@
       reinterpret_cast<char*>(PartitionPage::ToPointer(page1)) + kPointerOffset;
   PartitionFree(ptr);
   // This allocation should be satisfied by page1.
-  newPtr = reinterpret_cast<char*>(
+  new_ptr = reinterpret_cast<char*>(
       allocator.root()->Alloc(kTestAllocSize, type_name));
-  EXPECT_EQ(ptr, newPtr);
+  EXPECT_EQ(ptr, new_ptr);
   EXPECT_EQ(page1, bucket->active_pages_head);
   EXPECT_EQ(page2, page1->next_page);
 
@@ -500,19 +518,19 @@
 TEST_F(PartitionAllocTest, FreePageListPageTransitions) {
   PartitionBucket* bucket = &allocator.root()->buckets()[kTestBucketIndex];
 
-  size_t numToFillFreeListPage =
+  size_t num_to_fill_free_list_page =
       kPartitionPageSize / (sizeof(PartitionPage) + kExtraAllocSize);
   // The +1 is because we need to account for the fact that the current page
   // never gets thrown on the freelist.
-  ++numToFillFreeListPage;
-  auto pages = std::make_unique<PartitionPage* []>(numToFillFreeListPage);
+  ++num_to_fill_free_list_page;
+  auto pages = std::make_unique<PartitionPage* []>(num_to_fill_free_list_page);
 
   size_t i;
-  for (i = 0; i < numToFillFreeListPage; ++i) {
+  for (i = 0; i < num_to_fill_free_list_page; ++i) {
     pages[i] = GetFullPage(kTestAllocSize);
   }
-  EXPECT_EQ(pages[numToFillFreeListPage - 1], bucket->active_pages_head);
-  for (i = 0; i < numToFillFreeListPage; ++i)
+  EXPECT_EQ(pages[num_to_fill_free_list_page - 1], bucket->active_pages_head);
+  for (i = 0; i < num_to_fill_free_list_page; ++i)
     FreeFullPage(pages[i]);
   EXPECT_EQ(PartitionPage::get_sentinel_page(), bucket->active_pages_head);
   EXPECT_TRUE(bucket->empty_pages_head);
@@ -525,12 +543,12 @@
   FreeFullPage(page1);
   FreeFullPage(page2);
 
-  for (i = 0; i < numToFillFreeListPage; ++i) {
+  for (i = 0; i < num_to_fill_free_list_page; ++i) {
     pages[i] = GetFullPage(kTestAllocSize);
   }
-  EXPECT_EQ(pages[numToFillFreeListPage - 1], bucket->active_pages_head);
+  EXPECT_EQ(pages[num_to_fill_free_list_page - 1], bucket->active_pages_head);
 
-  for (i = 0; i < numToFillFreeListPage; ++i)
+  for (i = 0; i < num_to_fill_free_list_page; ++i)
     FreeFullPage(pages[i]);
   EXPECT_EQ(PartitionPage::get_sentinel_page(), bucket->active_pages_head);
   EXPECT_TRUE(bucket->empty_pages_head);
@@ -541,32 +559,32 @@
 TEST_F(PartitionAllocTest, MultiPageAllocs) {
   // This is guaranteed to cross a super page boundary because the first
   // partition page "slot" will be taken up by a guard page.
-  size_t numPagesNeeded = kNumPartitionPagesPerSuperPage;
+  size_t num_pages_needed = kNumPartitionPagesPerSuperPage;
   // The super page should begin and end in a guard so we one less page in
   // order to allocate a single page in the new super page.
-  --numPagesNeeded;
+  --num_pages_needed;
 
-  EXPECT_GT(numPagesNeeded, 1u);
-  auto pages = std::make_unique<PartitionPage* []>(numPagesNeeded);
-  uintptr_t firstSuperPageBase = 0;
+  EXPECT_GT(num_pages_needed, 1u);
+  auto pages = std::make_unique<PartitionPage* []>(num_pages_needed);
+  uintptr_t first_super_page_base = 0;
   size_t i;
-  for (i = 0; i < numPagesNeeded; ++i) {
+  for (i = 0; i < num_pages_needed; ++i) {
     pages[i] = GetFullPage(kTestAllocSize);
-    void* storagePtr = PartitionPage::ToPointer(pages[i]);
+    void* storage_ptr = PartitionPage::ToPointer(pages[i]);
     if (!i)
-      firstSuperPageBase =
-          reinterpret_cast<uintptr_t>(storagePtr) & kSuperPageBaseMask;
-    if (i == numPagesNeeded - 1) {
-      uintptr_t secondSuperPageBase =
-          reinterpret_cast<uintptr_t>(storagePtr) & kSuperPageBaseMask;
-      uintptr_t secondSuperPageOffset =
-          reinterpret_cast<uintptr_t>(storagePtr) & kSuperPageOffsetMask;
-      EXPECT_FALSE(secondSuperPageBase == firstSuperPageBase);
+      first_super_page_base =
+          reinterpret_cast<uintptr_t>(storage_ptr) & kSuperPageBaseMask;
+    if (i == num_pages_needed - 1) {
+      uintptr_t second_super_page_base =
+          reinterpret_cast<uintptr_t>(storage_ptr) & kSuperPageBaseMask;
+      uintptr_t second_super_page_offset =
+          reinterpret_cast<uintptr_t>(storage_ptr) & kSuperPageOffsetMask;
+      EXPECT_FALSE(second_super_page_base == first_super_page_base);
       // Check that we allocated a guard page for the second page.
-      EXPECT_EQ(kPartitionPageSize, secondSuperPageOffset);
+      EXPECT_EQ(kPartitionPageSize, second_super_page_offset);
     }
   }
-  for (i = 0; i < numPagesNeeded; ++i)
+  for (i = 0; i < num_pages_needed; ++i)
     FreeFullPage(pages[i]);
 }
 
@@ -582,81 +600,81 @@
 
   ptr = generic_allocator.root()->Alloc(1, type_name);
   EXPECT_TRUE(ptr);
-  void* origPtr = ptr;
-  char* charPtr = static_cast<char*>(ptr);
-  *charPtr = 'A';
+  void* orig_ptr = ptr;
+  char* char_ptr = static_cast<char*>(ptr);
+  *char_ptr = 'A';
 
   // Change the size of the realloc, remaining inside the same bucket.
-  void* newPtr = generic_allocator.root()->Realloc(ptr, 2, type_name);
-  EXPECT_EQ(ptr, newPtr);
-  newPtr = generic_allocator.root()->Realloc(ptr, 1, type_name);
-  EXPECT_EQ(ptr, newPtr);
-  newPtr =
+  void* new_ptr = generic_allocator.root()->Realloc(ptr, 2, type_name);
+  EXPECT_EQ(ptr, new_ptr);
+  new_ptr = generic_allocator.root()->Realloc(ptr, 1, type_name);
+  EXPECT_EQ(ptr, new_ptr);
+  new_ptr =
       generic_allocator.root()->Realloc(ptr, kGenericSmallestBucket, type_name);
-  EXPECT_EQ(ptr, newPtr);
+  EXPECT_EQ(ptr, new_ptr);
 
   // Change the size of the realloc, switching buckets.
-  newPtr = generic_allocator.root()->Realloc(ptr, kGenericSmallestBucket + 1,
-                                             type_name);
-  EXPECT_NE(newPtr, ptr);
+  new_ptr = generic_allocator.root()->Realloc(ptr, kGenericSmallestBucket + 1,
+                                              type_name);
+  EXPECT_NE(new_ptr, ptr);
   // Check that the realloc copied correctly.
-  char* newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'A');
+  char* new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'A');
 #if DCHECK_IS_ON()
   // Subtle: this checks for an old bug where we copied too much from the
   // source of the realloc. The condition can be detected by a trashing of
   // the uninitialized value in the space of the upsized allocation.
-  EXPECT_EQ(kUninitializedByte,
-            static_cast<unsigned char>(*(newCharPtr + kGenericSmallestBucket)));
+  EXPECT_EQ(kUninitializedByte, static_cast<unsigned char>(
+                                    *(new_char_ptr + kGenericSmallestBucket)));
 #endif
-  *newCharPtr = 'B';
+  *new_char_ptr = 'B';
   // The realloc moved. To check that the old allocation was freed, we can
   // do an alloc of the old allocation size and check that the old allocation
   // address is at the head of the freelist and reused.
-  void* reusedPtr = generic_allocator.root()->Alloc(1, type_name);
-  EXPECT_EQ(reusedPtr, origPtr);
-  generic_allocator.root()->Free(reusedPtr);
+  void* reused_ptr = generic_allocator.root()->Alloc(1, type_name);
+  EXPECT_EQ(reused_ptr, orig_ptr);
+  generic_allocator.root()->Free(reused_ptr);
 
   // Downsize the realloc.
-  ptr = newPtr;
-  newPtr = generic_allocator.root()->Realloc(ptr, 1, type_name);
-  EXPECT_EQ(newPtr, origPtr);
-  newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'B');
-  *newCharPtr = 'C';
+  ptr = new_ptr;
+  new_ptr = generic_allocator.root()->Realloc(ptr, 1, type_name);
+  EXPECT_EQ(new_ptr, orig_ptr);
+  new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'B');
+  *new_char_ptr = 'C';
 
   // Upsize the realloc to outside the partition.
-  ptr = newPtr;
-  newPtr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed + 1,
-                                             type_name);
-  EXPECT_NE(newPtr, ptr);
-  newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'C');
-  *newCharPtr = 'D';
+  ptr = new_ptr;
+  new_ptr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed + 1,
+                                              type_name);
+  EXPECT_NE(new_ptr, ptr);
+  new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'C');
+  *new_char_ptr = 'D';
 
   // Upsize and downsize the realloc, remaining outside the partition.
-  ptr = newPtr;
-  newPtr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed * 10,
-                                             type_name);
-  newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'D');
-  *newCharPtr = 'E';
-  ptr = newPtr;
-  newPtr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed * 2,
-                                             type_name);
-  newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'E');
-  *newCharPtr = 'F';
+  ptr = new_ptr;
+  new_ptr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed * 10,
+                                              type_name);
+  new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'D');
+  *new_char_ptr = 'E';
+  ptr = new_ptr;
+  new_ptr = generic_allocator.root()->Realloc(ptr, kGenericMaxBucketed * 2,
+                                              type_name);
+  new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'E');
+  *new_char_ptr = 'F';
 
   // Downsize the realloc to inside the partition.
-  ptr = newPtr;
-  newPtr = generic_allocator.root()->Realloc(ptr, 1, type_name);
-  EXPECT_NE(newPtr, ptr);
-  EXPECT_EQ(newPtr, origPtr);
-  newCharPtr = static_cast<char*>(newPtr);
-  EXPECT_EQ(*newCharPtr, 'F');
+  ptr = new_ptr;
+  new_ptr = generic_allocator.root()->Realloc(ptr, 1, type_name);
+  EXPECT_NE(new_ptr, ptr);
+  EXPECT_EQ(new_ptr, orig_ptr);
+  new_char_ptr = static_cast<char*>(new_ptr);
+  EXPECT_EQ(*new_char_ptr, 'F');
 
-  generic_allocator.root()->Free(newPtr);
+  generic_allocator.root()->Free(new_ptr);
 }
 
 // Test the generic allocation functions can handle some specific sizes of
@@ -706,10 +724,10 @@
   EXPECT_NE(-1, page->empty_cache_index);
   EXPECT_EQ(0, page->num_allocated_slots);
   EXPECT_EQ(0, page->num_unprovisioned_slots);
-  void* newPtr = generic_allocator.root()->Alloc(size, type_name);
-  EXPECT_EQ(ptr3, newPtr);
-  newPtr = generic_allocator.root()->Alloc(size, type_name);
-  EXPECT_EQ(ptr2, newPtr);
+  void* new_ptr = generic_allocator.root()->Alloc(size, type_name);
+  EXPECT_EQ(ptr3, new_ptr);
+  new_ptr = generic_allocator.root()->Alloc(size, type_name);
+  EXPECT_EQ(ptr2, new_ptr);
 #if defined(OS_LINUX) && !DCHECK_IS_ON()
   // On Linux, we have a guarantee that freelisting a page should cause its
   // contents to be nulled out. We check for null here to detect an bug we
@@ -718,9 +736,9 @@
   // We only run the check when asserts are disabled because when they are
   // enabled, the allocated area is overwritten with an "uninitialized"
   // byte pattern.
-  EXPECT_EQ(0, *(reinterpret_cast<char*>(newPtr) + (size - 1)));
+  EXPECT_EQ(0, *(reinterpret_cast<char*>(new_ptr) + (size - 1)));
 #endif
-  generic_allocator.root()->Free(newPtr);
+  generic_allocator.root()->Free(new_ptr);
   generic_allocator.root()->Free(ptr3);
   generic_allocator.root()->Free(ptr4);
 
@@ -739,8 +757,8 @@
   size -= kSystemPageSize;
   size -= 1;
   ptr = generic_allocator.root()->Alloc(size, type_name);
-  char* charPtr = reinterpret_cast<char*>(ptr);
-  *(charPtr + (size - 1)) = 'A';
+  char* char_ptr = reinterpret_cast<char*>(ptr);
+  *(char_ptr + (size - 1)) = 'A';
   generic_allocator.root()->Free(ptr);
 
   // Can we free null?
@@ -794,8 +812,8 @@
   EXPECT_EQ(predicted_size, actual_size);
   EXPECT_EQ(requested_size + kSystemPageSize, actual_size);
   // Check that we can write at the end of the reported size too.
-  char* charPtr = reinterpret_cast<char*>(ptr);
-  *(charPtr + (actual_size - 1)) = 'A';
+  char* char_ptr = reinterpret_cast<char*>(ptr);
+  *(char_ptr + (actual_size - 1)) = 'A';
   generic_allocator.root()->Free(ptr);
 
   // Allocate something very large, and uneven.
@@ -837,22 +855,22 @@
   memset(ptr, 'A', size);
   ptr2 = generic_allocator.root()->Realloc(ptr, size + 1, type_name);
   EXPECT_NE(ptr, ptr2);
-  char* charPtr2 = static_cast<char*>(ptr2);
-  EXPECT_EQ('A', charPtr2[0]);
-  EXPECT_EQ('A', charPtr2[size - 1]);
+  char* char_ptr2 = static_cast<char*>(ptr2);
+  EXPECT_EQ('A', char_ptr2[0]);
+  EXPECT_EQ('A', char_ptr2[size - 1]);
 #if DCHECK_IS_ON()
-  EXPECT_EQ(kUninitializedByte, static_cast<unsigned char>(charPtr2[size]));
+  EXPECT_EQ(kUninitializedByte, static_cast<unsigned char>(char_ptr2[size]));
 #endif
 
   // Test that shrinking an allocation with realloc() also copies everything
   // from the old allocation.
   ptr = generic_allocator.root()->Realloc(ptr2, size - 1, type_name);
   EXPECT_NE(ptr2, ptr);
-  char* charPtr = static_cast<char*>(ptr);
-  EXPECT_EQ('A', charPtr[0]);
-  EXPECT_EQ('A', charPtr[size - 2]);
+  char* char_ptr = static_cast<char*>(ptr);
+  EXPECT_EQ('A', char_ptr[0]);
+  EXPECT_EQ('A', char_ptr[size - 2]);
 #if DCHECK_IS_ON()
-  EXPECT_EQ(kUninitializedByte, static_cast<unsigned char>(charPtr[size - 1]));
+  EXPECT_EQ(kUninitializedByte, static_cast<unsigned char>(char_ptr[size - 1]));
 #endif
 
   generic_allocator.root()->Free(ptr);
@@ -895,10 +913,10 @@
 
   PartitionPage* page =
       PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
-  size_t totalSlots =
+  size_t total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
       (big_size + kExtraAllocSize);
-  EXPECT_EQ(4u, totalSlots);
+  EXPECT_EQ(4u, total_slots);
   // The freelist should have one entry, because we were able to exactly fit
   // one object slot and one freelist pointer (the null that the head points
   // to) into a system page.
@@ -958,12 +976,12 @@
   EXPECT_TRUE(ptr);
   page = PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
   EXPECT_EQ(1, page->num_allocated_slots);
-  totalSlots =
+  total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
       (mediumSize + kExtraAllocSize);
-  size_t firstPageSlots = kSystemPageSize / (mediumSize + kExtraAllocSize);
-  EXPECT_EQ(2u, firstPageSlots);
-  EXPECT_EQ(totalSlots - firstPageSlots, page->num_unprovisioned_slots);
+  size_t first_page_slots = kSystemPageSize / (mediumSize + kExtraAllocSize);
+  EXPECT_EQ(2u, first_page_slots);
+  EXPECT_EQ(total_slots - first_page_slots, page->num_unprovisioned_slots);
 
   PartitionFree(ptr);
 
@@ -976,11 +994,11 @@
   EXPECT_TRUE(ptr);
   page = PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
   EXPECT_EQ(1, page->num_allocated_slots);
-  totalSlots =
+  total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
       (smallSize + kExtraAllocSize);
-  firstPageSlots = kSystemPageSize / (smallSize + kExtraAllocSize);
-  EXPECT_EQ(totalSlots - firstPageSlots, page->num_unprovisioned_slots);
+  first_page_slots = kSystemPageSize / (smallSize + kExtraAllocSize);
+  EXPECT_EQ(total_slots - first_page_slots, page->num_unprovisioned_slots);
 
   PartitionFree(ptr);
   EXPECT_TRUE(page->freelist_head);
@@ -995,11 +1013,11 @@
   EXPECT_TRUE(ptr);
   page = PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
   EXPECT_EQ(1, page->num_allocated_slots);
-  totalSlots =
+  total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
       (verySmallSize + kExtraAllocSize);
-  firstPageSlots = kSystemPageSize / (verySmallSize + kExtraAllocSize);
-  EXPECT_EQ(totalSlots - firstPageSlots, page->num_unprovisioned_slots);
+  first_page_slots = kSystemPageSize / (verySmallSize + kExtraAllocSize);
+  EXPECT_EQ(total_slots - first_page_slots, page->num_unprovisioned_slots);
 
   PartitionFree(ptr);
   EXPECT_TRUE(page->freelist_head);
@@ -1007,17 +1025,17 @@
 
   // And try an allocation size (against the generic allocator) that is
   // larger than a system page.
-  size_t pageAndAHalfSize =
+  size_t page_and_a_half_size =
       (kSystemPageSize + (kSystemPageSize / 2)) - kExtraAllocSize;
-  ptr = generic_allocator.root()->Alloc(pageAndAHalfSize, type_name);
+  ptr = generic_allocator.root()->Alloc(page_and_a_half_size, type_name);
   EXPECT_TRUE(ptr);
   page = PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
   EXPECT_EQ(1, page->num_allocated_slots);
   EXPECT_TRUE(page->freelist_head);
-  totalSlots =
+  total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
-      (pageAndAHalfSize + kExtraAllocSize);
-  EXPECT_EQ(totalSlots - 2, page->num_unprovisioned_slots);
+      (page_and_a_half_size + kExtraAllocSize);
+  EXPECT_EQ(total_slots - 2, page->num_unprovisioned_slots);
   generic_allocator.root()->Free(ptr);
 
   // And then make sure than exactly the page size only faults one page.
@@ -1027,10 +1045,10 @@
   page = PartitionPage::FromPointer(PartitionCookieFreePointerAdjust(ptr));
   EXPECT_EQ(1, page->num_allocated_slots);
   EXPECT_FALSE(page->freelist_head);
-  totalSlots =
+  total_slots =
       (page->bucket->num_system_pages_per_slot_span * kSystemPageSize) /
       (pageSize + kExtraAllocSize);
-  EXPECT_EQ(totalSlots - 1, page->num_unprovisioned_slots);
+  EXPECT_EQ(total_slots - 1, page->num_unprovisioned_slots);
   generic_allocator.root()->Free(ptr);
 }
 
@@ -1092,49 +1110,50 @@
 TEST_F(PartitionAllocTest, MappingCollision) {
   // The -2 is because the first and last partition pages in a super page are
   // guard pages.
-  size_t numPartitionPagesNeeded = kNumPartitionPagesPerSuperPage - 2;
-  auto firstSuperPagePages =
-      std::make_unique<PartitionPage* []>(numPartitionPagesNeeded);
-  auto secondSuperPagePages =
-      std::make_unique<PartitionPage* []>(numPartitionPagesNeeded);
+  size_t num_partition_pages_needed = kNumPartitionPagesPerSuperPage - 2;
+  auto first_super_page_pages =
+      std::make_unique<PartitionPage* []>(num_partition_pages_needed);
+  auto second_super_page_pages =
+      std::make_unique<PartitionPage* []>(num_partition_pages_needed);
 
   size_t i;
-  for (i = 0; i < numPartitionPagesNeeded; ++i)
-    firstSuperPagePages[i] = GetFullPage(kTestAllocSize);
+  for (i = 0; i < num_partition_pages_needed; ++i)
+    first_super_page_pages[i] = GetFullPage(kTestAllocSize);
 
-  char* pageBase =
-      reinterpret_cast<char*>(PartitionPage::ToPointer(firstSuperPagePages[0]));
+  char* page_base = reinterpret_cast<char*>(
+      PartitionPage::ToPointer(first_super_page_pages[0]));
   EXPECT_EQ(kPartitionPageSize,
-            reinterpret_cast<uintptr_t>(pageBase) & kSuperPageOffsetMask);
-  pageBase -= kPartitionPageSize;
+            reinterpret_cast<uintptr_t>(page_base) & kSuperPageOffsetMask);
+  page_base -= kPartitionPageSize;
   // Map a single system page either side of the mapping for our allocations,
   // with the goal of tripping up alignment of the next mapping.
-  void* map1 = AllocPages(pageBase - kPageAllocationGranularity,
+  void* map1 = AllocPages(page_base - kPageAllocationGranularity,
                           kPageAllocationGranularity,
                           kPageAllocationGranularity, PageInaccessible);
   EXPECT_TRUE(map1);
-  void* map2 = AllocPages(pageBase + kSuperPageSize, kPageAllocationGranularity,
-                          kPageAllocationGranularity, PageInaccessible);
+  void* map2 =
+      AllocPages(page_base + kSuperPageSize, kPageAllocationGranularity,
+                 kPageAllocationGranularity, PageInaccessible);
   EXPECT_TRUE(map2);
 
-  for (i = 0; i < numPartitionPagesNeeded; ++i)
-    secondSuperPagePages[i] = GetFullPage(kTestAllocSize);
+  for (i = 0; i < num_partition_pages_needed; ++i)
+    second_super_page_pages[i] = GetFullPage(kTestAllocSize);
 
   FreePages(map1, kPageAllocationGranularity);
   FreePages(map2, kPageAllocationGranularity);
 
-  pageBase = reinterpret_cast<char*>(
-      PartitionPage::ToPointer(secondSuperPagePages[0]));
+  page_base = reinterpret_cast<char*>(
+      PartitionPage::ToPointer(second_super_page_pages[0]));
   EXPECT_EQ(kPartitionPageSize,
-            reinterpret_cast<uintptr_t>(pageBase) & kSuperPageOffsetMask);
-  pageBase -= kPartitionPageSize;
+            reinterpret_cast<uintptr_t>(page_base) & kSuperPageOffsetMask);
+  page_base -= kPartitionPageSize;
   // Map a single system page either side of the mapping for our allocations,
   // with the goal of tripping up alignment of the next mapping.
-  map1 = AllocPages(pageBase - kPageAllocationGranularity,
+  map1 = AllocPages(page_base - kPageAllocationGranularity,
                     kPageAllocationGranularity, kPageAllocationGranularity,
                     PageReadWrite);
   EXPECT_TRUE(map1);
-  map2 = AllocPages(pageBase + kSuperPageSize, kPageAllocationGranularity,
+  map2 = AllocPages(page_base + kSuperPageSize, kPageAllocationGranularity,
                     kPageAllocationGranularity, PageReadWrite);
   EXPECT_TRUE(map2);
   EXPECT_TRUE(
@@ -1142,32 +1161,32 @@
   EXPECT_TRUE(
       SetSystemPagesAccess(map2, kPageAllocationGranularity, PageInaccessible));
 
-  PartitionPage* pageInThirdSuperPage = GetFullPage(kTestAllocSize);
+  PartitionPage* page_in_third_super_page = GetFullPage(kTestAllocSize);
   FreePages(map1, kPageAllocationGranularity);
   FreePages(map2, kPageAllocationGranularity);
 
   EXPECT_EQ(0u, reinterpret_cast<uintptr_t>(
-                    PartitionPage::ToPointer(pageInThirdSuperPage)) &
+                    PartitionPage::ToPointer(page_in_third_super_page)) &
                     kPartitionPageOffsetMask);
 
   // And make sure we really did get a page in a new superpage.
   EXPECT_NE(reinterpret_cast<uintptr_t>(
-                PartitionPage::ToPointer(firstSuperPagePages[0])) &
+                PartitionPage::ToPointer(first_super_page_pages[0])) &
                 kSuperPageBaseMask,
             reinterpret_cast<uintptr_t>(
-                PartitionPage::ToPointer(pageInThirdSuperPage)) &
+                PartitionPage::ToPointer(page_in_third_super_page)) &
                 kSuperPageBaseMask);
   EXPECT_NE(reinterpret_cast<uintptr_t>(
-                PartitionPage::ToPointer(secondSuperPagePages[0])) &
+                PartitionPage::ToPointer(second_super_page_pages[0])) &
                 kSuperPageBaseMask,
             reinterpret_cast<uintptr_t>(
-                PartitionPage::ToPointer(pageInThirdSuperPage)) &
+                PartitionPage::ToPointer(page_in_third_super_page)) &
                 kSuperPageBaseMask);
 
-  FreeFullPage(pageInThirdSuperPage);
-  for (i = 0; i < numPartitionPagesNeeded; ++i) {
-    FreeFullPage(firstSuperPagePages[i]);
-    FreeFullPage(secondSuperPagePages[i]);
+  FreeFullPage(page_in_third_super_page);
+  for (i = 0; i < num_partition_pages_needed; ++i) {
+    FreeFullPage(first_super_page_pages[i]);
+    FreeFullPage(second_super_page_pages[i]);
   }
 }
 
@@ -1315,23 +1334,41 @@
 // process, so they won't pollute other tests.
 TEST_F(PartitionAllocDeathTest, RepeatedAllocReturnNullDirect) {
   // A direct-mapped allocation size.
-  EXPECT_DEATH(DoReturnNullTest(32 * 1024 * 1024, false), "DoReturnNullTest");
+  EXPECT_DEATH(DoReturnNullTest(32 * 1024 * 1024, kPartitionAllocGenericFlags),
+               "DoReturnNullTest");
 }
 
 // Repeating above test with Realloc
 TEST_F(PartitionAllocDeathTest, RepeatedReallocReturnNullDirect) {
-  EXPECT_DEATH(DoReturnNullTest(32 * 1024 * 1024, true), "DoReturnNullTest");
+  EXPECT_DEATH(
+      DoReturnNullTest(32 * 1024 * 1024, kPartitionReallocGenericFlags),
+      "DoReturnNullTest");
+}
+
+// Repeating above test with TryRealloc
+TEST_F(PartitionAllocDeathTest, RepeatedTryReallocReturnNullDirect) {
+  EXPECT_DEATH(
+      DoReturnNullTest(32 * 1024 * 1024, kPartitionRootGenericTryRealloc),
+      "DoReturnNullTest");
 }
 
 // Test "return null" with a 512 kB block size.
 TEST_F(PartitionAllocDeathTest, RepeatedAllocReturnNull) {
   // A single-slot but non-direct-mapped allocation size.
-  EXPECT_DEATH(DoReturnNullTest(512 * 1024, false), "DoReturnNullTest");
+  EXPECT_DEATH(DoReturnNullTest(512 * 1024, kPartitionAllocGenericFlags),
+               "DoReturnNullTest");
 }
 
 // Repeating above test with Realloc.
 TEST_F(PartitionAllocDeathTest, RepeatedReallocReturnNull) {
-  EXPECT_DEATH(DoReturnNullTest(512 * 1024, true), "DoReturnNullTest");
+  EXPECT_DEATH(DoReturnNullTest(512 * 1024, kPartitionReallocGenericFlags),
+               "DoReturnNullTest");
+}
+
+// Repeating above test with TryRealloc.
+TEST_F(PartitionAllocDeathTest, RepeatedTryReallocReturnNull) {
+  EXPECT_DEATH(DoReturnNullTest(512 * 1024, kPartitionRootGenericTryRealloc),
+               "DoReturnNullTest");
 }
 
 #endif  // !defined(ARCH_CPU_64_BITS) || (defined(OS_POSIX) &&
@@ -1398,10 +1435,10 @@
   void* ptr = generic_allocator.root()->Alloc(size, type_name);
 
   EXPECT_TRUE(ptr);
-  char* charPtr = reinterpret_cast<char*>(ptr) - kPointerOffset;
+  char* char_ptr = reinterpret_cast<char*>(ptr) - kPointerOffset;
 
-  EXPECT_DEATH(*(charPtr - 1) = 'A', "");
-  EXPECT_DEATH(*(charPtr + size + kExtraAllocSize) = 'A', "");
+  EXPECT_DEATH(*(char_ptr - 1) = 'A', "");
+  EXPECT_DEATH(*(char_ptr + size + kExtraAllocSize) = 'A', "");
 
   generic_allocator.root()->Free(ptr);
 }
@@ -1428,10 +1465,10 @@
 TEST_F(PartitionAllocTest, DumpMemoryStats) {
   {
     void* ptr = allocator.root()->Alloc(kTestAllocSize, type_name);
-    MockPartitionStatsDumper mockStatsDumper;
+    MockPartitionStatsDumper mock_stats_dumper;
     allocator.root()->DumpStats("mock_allocator", false /* detailed dump */,
-                                &mockStatsDumper);
-    EXPECT_TRUE(mockStatsDumper.IsMemoryAllocationRecorded());
+                                &mock_stats_dumper);
+    EXPECT_TRUE(mock_stats_dumper.IsMemoryAllocationRecorded());
     PartitionFree(ptr);
   }
 
@@ -1720,13 +1757,13 @@
   // state of the free cache ring.
   generic_allocator.root()->PurgeMemory(PartitionPurgeDecommitEmptyPages);
 
-  char* bigPtr = reinterpret_cast<char*>(
+  char* big_ptr = reinterpret_cast<char*>(
       generic_allocator.root()->Alloc(256 * 1024, type_name));
-  generic_allocator.root()->Free(bigPtr);
+  generic_allocator.root()->Free(big_ptr);
   generic_allocator.root()->PurgeMemory(PartitionPurgeDecommitEmptyPages);
 
   CHECK_PAGE_IN_CORE(ptr - kPointerOffset, false);
-  CHECK_PAGE_IN_CORE(bigPtr - kPointerOffset, false);
+  CHECK_PAGE_IN_CORE(big_ptr - kPointerOffset, false);
 }
 
 // Tests that we prefer to allocate into a non-empty partition page over an
diff --git a/base/allocator/partition_allocator/partition_bucket.cc b/base/allocator/partition_allocator/partition_bucket.cc
index fcea5232..01d5b42f 100644
--- a/base/allocator/partition_allocator/partition_bucket.cc
+++ b/base/allocator/partition_allocator/partition_bucket.cc
@@ -38,7 +38,7 @@
   map_size &= kPageAllocationGranularityBaseMask;
 
   // TODO: these pages will be zero-filled. Consider internalizing an
-  // allocZeroed() API so we can avoid a memset() entirely in this case.
+  // AllocZeroed() API so we can avoid a memset() entirely in this case.
   char* ptr = reinterpret_cast<char*>(
       AllocPages(nullptr, map_size, kSuperPageSize, PageReadWrite));
   if (UNLIKELY(!ptr))
@@ -217,9 +217,9 @@
   // address region as much as possible. This is important for not causing
   // page table bloat and not fragmenting address spaces in 32 bit
   // architectures.
-  char* requestedAddress = root->next_super_page;
+  char* requested_address = root->next_super_page;
   char* super_page = reinterpret_cast<char*>(AllocPages(
-      requestedAddress, kSuperPageSize, kSuperPageSize, PageReadWrite));
+      requested_address, kSuperPageSize, kSuperPageSize, PageReadWrite));
   if (UNLIKELY(!super_page))
     return nullptr;
 
@@ -262,7 +262,7 @@
   // distributions will allocate the mapping directly before the last
   // successful mapping, which is far from random. So we just get fresh
   // randomness for the next mapping attempt.
-  if (requestedAddress && requestedAddress != super_page)
+  if (requested_address && requested_address != super_page)
     root->next_super_page = nullptr;
 
   // We allocated a new super page so update super page metadata.
@@ -281,8 +281,8 @@
   latest_extent->next = nullptr;
 
   PartitionSuperPageExtentEntry* current_extent = root->current_extent;
-  bool isNewExtent = (super_page != requestedAddress);
-  if (UNLIKELY(isNewExtent)) {
+  bool is_new_extent = (super_page != requested_address);
+  if (UNLIKELY(is_new_extent)) {
     if (UNLIKELY(!current_extent)) {
       DCHECK(!root->first_extent);
       root->first_extent = latest_extent;
@@ -512,9 +512,9 @@
   } else {
     // Third. If we get here, we need a brand new page.
     uint16_t num_partition_pages = this->get_pages_per_slot_span();
-    void* rawPages = AllocNewSlotSpan(root, flags, num_partition_pages);
-    if (LIKELY(rawPages != nullptr)) {
-      new_page = PartitionPage::FromPointerNoAlignmentCheck(rawPages);
+    void* raw_pages = AllocNewSlotSpan(root, flags, num_partition_pages);
+    if (LIKELY(raw_pages != nullptr)) {
+      new_page = PartitionPage::FromPointerNoAlignmentCheck(raw_pages);
       InitializeSlotSpan(new_page);
     }
   }
diff --git a/base/android/proguard/proguard.gni b/base/android/proguard/proguard.gni
new file mode 100644
index 0000000..1e78bc0
--- /dev/null
+++ b/base/android/proguard/proguard.gni
@@ -0,0 +1,9 @@
+# Copyright 2018 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+declare_args() {
+  # Controls whether proguard obfuscation is enabled for targets
+  # configured to use it.
+  enable_proguard_obfuscation = true
+}
diff --git a/base/logging.cc b/base/logging.cc
index 8eabda0e..7752cce 100644
--- a/base/logging.cc
+++ b/base/logging.cc
@@ -852,7 +852,11 @@
       }
 #endif
       // Crash the process to generate a dump.
+#if defined(OFFICIAL_BUILD) && defined(NDEBUG)
+      IMMEDIATE_CRASH();
+#else
       base::debug::BreakDebugger();
+#endif
     }
   }
 }
diff --git a/base/message_loop/incoming_task_queue.cc b/base/message_loop/incoming_task_queue.cc
index 6821730f..c05c38f 100644
--- a/base/message_loop/incoming_task_queue.cc
+++ b/base/message_loop/incoming_task_queue.cc
@@ -72,9 +72,6 @@
   }
 #endif
 
-  if (!delay.is_zero())
-    UMA_HISTOGRAM_LONG_TIMES("MessageLoop.DelayedTaskQueue.PostedDelay", delay);
-
   return PostPendingTask(&pending_task);
 }
 
@@ -85,7 +82,7 @@
 
 void IncomingTaskQueue::ReportMetricsOnIdle() const {
   UMA_HISTOGRAM_COUNTS_1M(
-      "MessageLoop.DelayedTaskQueue.PendingTasksCountOnIdle",
+      "MessageLoop.DelayedTaskQueueForUI.PendingTasksCountOnIdle",
       delayed_tasks_.Size());
 }
 
diff --git a/base/message_loop/message_loop.cc b/base/message_loop/message_loop.cc
index 1b76cbb3..3723960a 100644
--- a/base/message_loop/message_loop.cc
+++ b/base/message_loop/message_loop.cc
@@ -15,7 +15,6 @@
 #include "base/message_loop/message_pump_default.h"
 #include "base/message_loop/message_pump_for_io.h"
 #include "base/message_loop/message_pump_for_ui.h"
-#include "base/metrics/histogram_macros.h"
 #include "base/run_loop.h"
 #include "base/third_party/dynamic_annotations/dynamic_annotations.h"
 #include "base/threading/thread_id_name_manager.h"
@@ -36,37 +35,6 @@
   return pump;
 }
 
-enum class ScheduledWakeupResult {
-  // The MessageLoop went to sleep with a timeout and woke up because of that
-  // timeout.
-  kCompleted,
-  // The MessageLoop went to sleep with a timeout but was woken up before it
-  // fired.
-  kInterrupted,
-};
-
-// Reports a ScheduledWakeup's result when waking up from a non-infinite sleep.
-// Reports are using a 14 day spread (maximum examined delay for
-// https://crbug.com/850450#c3), with 50 buckets that still yields 7 buckets
-// under 16ms and hence plenty of resolution.
-void ReportScheduledWakeupResult(ScheduledWakeupResult result,
-                                 TimeDelta intended_sleep) {
-  switch (result) {
-    case ScheduledWakeupResult::kCompleted:
-      UMA_HISTOGRAM_CUSTOM_TIMES("MessageLoop.ScheduledSleep.Completed",
-                                 intended_sleep,
-                                 base::TimeDelta::FromMilliseconds(1),
-                                 base::TimeDelta::FromDays(14), 50);
-      break;
-    case ScheduledWakeupResult::kInterrupted:
-      UMA_HISTOGRAM_CUSTOM_TIMES("MessageLoop.ScheduledSleep.Interrupted",
-                                 intended_sleep,
-                                 base::TimeDelta::FromMilliseconds(1),
-                                 base::TimeDelta::FromDays(14), 50);
-      break;
-  }
-}
-
 }  // namespace
 
 class MessageLoop::Controller : public internal::IncomingTaskQueue::Observer {
@@ -496,20 +464,6 @@
 
   // Execute oldest task.
   while (incoming_task_queue_->triage_tasks().HasTasks()) {
-    if (!scheduled_wakeup_.next_run_time.is_null()) {
-      // While the frontmost task may racily be ripe. The MessageLoop was awaken
-      // without needing the timeout anyways. Since this metric is about
-      // determining whether sleeping for long periods ever succeeds: it's
-      // easier to just consider any untriaged task as an interrupt (this also
-      // makes the logic simpler for untriaged delayed tasks which may alter the
-      // top of the task queue prior to DoDelayedWork() but did cause a wakeup
-      // regardless -- per currently requiring this immediate triage step even
-      // for long delays).
-      ReportScheduledWakeupResult(ScheduledWakeupResult::kInterrupted,
-                                  scheduled_wakeup_.intended_sleep);
-      scheduled_wakeup_ = ScheduledWakeup();
-    }
-
     PendingTask pending_task = incoming_task_queue_->triage_tasks().Pop();
     if (pending_task.task.IsCancelled())
       continue;
@@ -533,27 +487,9 @@
 }
 
 bool MessageLoop::DoDelayedWork(TimeTicks* next_delayed_work_time) {
-  if (!task_execution_allowed_) {
+  if (!task_execution_allowed_ ||
+      !incoming_task_queue_->delayed_tasks().HasTasks()) {
     *next_delayed_work_time = TimeTicks();
-    // |scheduled_wakeup_| isn't used in nested loops that don't process
-    // application tasks.
-    DCHECK(scheduled_wakeup_.next_run_time.is_null());
-    return false;
-  }
-
-  if (!incoming_task_queue_->delayed_tasks().HasTasks()) {
-    *next_delayed_work_time = TimeTicks();
-
-    // It's possible to be woken up by a system event and have it cancel the
-    // upcoming delayed task from under us before DoDelayedWork() -- see comment
-    // under |next_run_time > recent_time_|. This condition covers the special
-    // case where such a system event cancelled *all* pending delayed tasks.
-    if (!scheduled_wakeup_.next_run_time.is_null()) {
-      ReportScheduledWakeupResult(ScheduledWakeupResult::kInterrupted,
-                                  scheduled_wakeup_.intended_sleep);
-      scheduled_wakeup_ = ScheduledWakeup();
-    }
-
     return false;
   }
 
@@ -571,36 +507,10 @@
     recent_time_ = TimeTicks::Now();  // Get a better view of Now();
     if (next_run_time > recent_time_) {
       *next_delayed_work_time = CapAtOneDay(next_run_time);
-
-      // If the loop was woken up early by an untriaged task:
-      // |scheduled_wakeup_| will have been handled already in DoWork(). If it
-      // wasn't, it means the early wake up was caused by a system event (e.g.
-      // MessageLoopForUI or IO).
-      if (!scheduled_wakeup_.next_run_time.is_null()) {
-        // Handling the system event may have resulted in cancelling the
-        // upcoming delayed task (and then it being pruned by
-        // DelayedTaskQueue::HasTasks()); hence, we cannot check for strict
-        // equality here. We can however check that the pending task is either
-        // still there or that a later delay replaced it in front of the queue.
-        // There shouldn't have been new tasks added in |delayed_tasks()| per
-        // DoWork() not having triaged new tasks since the last DoIdleWork().
-        DCHECK_GE(next_run_time, scheduled_wakeup_.next_run_time);
-
-        ReportScheduledWakeupResult(ScheduledWakeupResult::kInterrupted,
-                                    scheduled_wakeup_.intended_sleep);
-        scheduled_wakeup_ = ScheduledWakeup();
-      }
-
       return false;
     }
   }
 
-  if (next_run_time == scheduled_wakeup_.next_run_time) {
-    ReportScheduledWakeupResult(ScheduledWakeupResult::kCompleted,
-                                scheduled_wakeup_.intended_sleep);
-    scheduled_wakeup_ = ScheduledWakeup();
-  }
-
   PendingTask pending_task = incoming_task_queue_->delayed_tasks().Pop();
 
   if (incoming_task_queue_->delayed_tasks().HasTasks()) {
@@ -619,43 +529,19 @@
   bool need_high_res_timers = false;
 #endif
 
-  // Do not report idle metrics nor do any logic related to delayed tasks if
-  // about to quit the loop and/or in a nested loop where
-  // |!task_execution_allowed_|. In the former case, the loop isn't going to
-  // sleep and in the latter case DoDelayedWork() will not actually do the work
-  // this is prepping for.
+  // Do not report idle metrics if about to quit the loop and/or in a nested
+  // loop where |!task_execution_allowed_|. In the former case, the loop isn't
+  // going to sleep and in the latter case DoDelayedWork() will not actually do
+  // the work this is prepping for.
   if (ShouldQuitWhenIdle()) {
     pump_->Quit();
   } else if (task_execution_allowed_) {
-    incoming_task_queue_->ReportMetricsOnIdle();
-
-    if (incoming_task_queue_->delayed_tasks().HasTasks()) {
-      TimeTicks scheduled_wakeup_time =
-          incoming_task_queue_->delayed_tasks().Peek().delayed_run_time;
-
-      if (!scheduled_wakeup_.next_run_time.is_null()) {
-        // It's possible for DoIdleWork() to be invoked twice in a row (e.g. if
-        // the MessagePump processed system work and became idle twice in a row
-        // without application tasks in between -- some pumps with a native
-        // message loop do not invoke DoWork() / DoDelayedWork() when awaken for
-        // system work only). As in DoDelayedWork(), we cannot check for strict
-        // equality below as the system work may have cancelled the frontmost
-        // task.
-        DCHECK_GE(scheduled_wakeup_time, scheduled_wakeup_.next_run_time);
-
-        ReportScheduledWakeupResult(ScheduledWakeupResult::kInterrupted,
-                                    scheduled_wakeup_.intended_sleep);
-        scheduled_wakeup_ = ScheduledWakeup();
-      }
-
-      // Store the remaining delay as well as the programmed wakeup time in
-      // order to know next time this MessageLoop wakes up whether it woke up
-      // because of this pending task (is it still the frontmost task in the
-      // queue?) and be able to report the slept delta (which is lost if not
-      // saved here).
-      scheduled_wakeup_ = ScheduledWakeup{
-          scheduled_wakeup_time, scheduled_wakeup_time - TimeTicks::Now()};
-    }
+    // Only track idle metrics in MessageLoopForUI to avoid too much contention
+    // logging the histogram (https://crbug.com/860801) -- there's typically
+    // only one UI thread per process and, for practical purposes, restricting
+    // the MessageLoop diagnostic metrics to it yields similar information.
+    if (type_ == TYPE_UI)
+      incoming_task_queue_->ReportMetricsOnIdle();
 
 #if defined(OS_WIN)
     // On Windows we activate the high resolution timer so that the wait
diff --git a/base/message_loop/message_loop.h b/base/message_loop/message_loop.h
index ab49d87..7c31a12 100644
--- a/base/message_loop/message_loop.h
+++ b/base/message_loop/message_loop.h
@@ -287,15 +287,6 @@
   // A recent snapshot of Time::Now(), used to check delayed_work_queue_.
   TimeTicks recent_time_;
 
-  // Non-null when the last thing this MessageLoop did is become idle with
-  // pending delayed tasks. Used to report metrics on the following wake up.
-  struct ScheduledWakeup {
-    // The scheduled time of the next delayed task when this loop became idle.
-    TimeTicks next_run_time;
-    // The delta until |next_run_time| when this loop became idle.
-    TimeDelta intended_sleep;
-  } scheduled_wakeup_;
-
   ObserverList<DestructionObserver> destruction_observers_;
 
   // A boolean which prevents unintentional reentrant task execution (e.g. from
diff --git a/base/message_loop/message_loop_unittest.cc b/base/message_loop/message_loop_unittest.cc
index 1a21fc5..f0b07d4 100644
--- a/base/message_loop/message_loop_unittest.cc
+++ b/base/message_loop/message_loop_unittest.cc
@@ -23,7 +23,9 @@
 #include "base/single_thread_task_runner.h"
 #include "base/synchronization/waitable_event.h"
 #include "base/task_scheduler/task_scheduler.h"
+#include "base/test/bind_test_util.h"
 #include "base/test/gtest_util.h"
+#include "base/test/metrics/histogram_tester.h"
 #include "base/test/test_simple_task_runner.h"
 #include "base/test/test_timeouts.h"
 #include "base/threading/platform_thread.h"
@@ -1758,6 +1760,63 @@
   run_loop.Run();
 }
 
+#if defined(OS_MACOSX)
+// This metric is a bit broken on Mac OS because CFRunLoop doesn't
+// deterministically invoke MessageLoop::DoIdleWork(). This being a temporary
+// diagnosis metric, we let this fly and simply not test it on Mac.
+#define MAYBE_MetricsOnlyFromUILoops DISABLED_MetricsOnlyFromUILoops
+#else
+#define MAYBE_MetricsOnlyFromUILoops MetricsOnlyFromUILoops
+#endif
+
+TEST_P(MessageLoopTypedTest, MAYBE_MetricsOnlyFromUILoops) {
+  MessageLoop loop(GetMessageLoopType());
+
+  const bool histograms_expected = GetMessageLoopType() == MessageLoop::TYPE_UI;
+
+  HistogramTester histogram_tester;
+
+  // A delay which is expected to give enough time for the MessageLoop to go
+  // idle after triaging it.
+  TimeDelta delay_that_leads_to_idle = TimeDelta::FromMilliseconds(1);
+
+  // On some platforms testing under emulation, 1 ms is not enough. See how long
+  // it takes to resolve a 1ms delayed task and use 10X that for the real test.
+  {
+    TimeTicks begin_run_loop = TimeTicks::Now();
+
+    RunLoop run_loop;
+    loop.task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(),
+                                        delay_that_leads_to_idle);
+    run_loop.Run();
+
+    delay_that_leads_to_idle = 10 * (TimeTicks::Now() - begin_run_loop);
+  }
+
+  SCOPED_TRACE(delay_that_leads_to_idle);
+
+  // Loop that goes idle with one pending task.
+  RunLoop run_loop;
+  loop.task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(),
+                                      delay_that_leads_to_idle);
+  run_loop.Run();
+
+  const std::vector<Bucket> buckets = histogram_tester.GetAllSamples(
+      "MessageLoop.DelayedTaskQueueForUI.PendingTasksCountOnIdle");
+  if (histograms_expected) {
+    // DoIdleWork() should have triggered at least once in the second RunLoop.
+    // It may also have triggered in the first one if the test environment is
+    // fast enough. It can sometimes also trigger additional times when a system
+    // message (e.g. system ping) interrupts the sleep. All cases should
+    // nonetheless report in the "1" bucket.
+    EXPECT_EQ(buckets.size(), 1U);
+    EXPECT_EQ(buckets[0].min, 1);
+    EXPECT_GE(buckets[0].count, 1);
+  } else {
+    EXPECT_TRUE(buckets.empty());
+  }
+}
+
 INSTANTIATE_TEST_CASE_P(
     ,
     MessageLoopTypedTest,
diff --git a/base/process/process_linux.cc b/base/process/process_linux.cc
index faf39af..5f07012 100644
--- a/base/process/process_linux.cc
+++ b/base/process/process_linux.cc
@@ -5,6 +5,7 @@
 #include "base/process/process.h"
 
 #include <errno.h>
+#include <limits.h>
 #include <sys/resource.h>
 
 #include "base/files/file_util.h"
@@ -74,12 +75,25 @@
 const int kBackgroundPriority = 5;
 #endif  // defined(OS_CHROMEOS)
 
+// NZERO should be defined in <limits.h> per POSIX, and should be at least
+// 20. (NZERO-1) is the highest possible niceness value (i.e. lowest process
+// priority). Most platforms use NZERO=20.
+//
+// RLIMIT_NICE tells us how much we can reduce niceness (increase priority) if
+// we start at NZERO.
+//
+// e.g. if NZERO is 20 and the rlimit is 30, we can lower niceness anywhere
+// within the [-10, 19] range (20 - 30 = -10).
+//
+// So, to re-raise priority to kForegroundPriority, we need at least this much:
+constexpr int kMinNiceRLimitForReraising = NZERO - kForegroundPriority;
+
 bool CanReraisePriority() {
   // We won't be able to raise the priority if we don't have the right rlimit.
   // The limit may be adjusted in /etc/security/limits.conf for PAM systems.
   struct rlimit rlim;
   return (getrlimit(RLIMIT_NICE, &rlim) == 0) &&
-         (20 - kForegroundPriority) <= static_cast<int>(rlim.rlim_cur);
+         static_cast<int>(rlim.rlim_cur) >= kMinNiceRLimitForReraising;
 }
 
 }  // namespace
diff --git a/base/task_scheduler/service_thread.cc b/base/task_scheduler/service_thread.cc
index 12cd3282..7e7731f 100644
--- a/base/task_scheduler/service_thread.cc
+++ b/base/task_scheduler/service_thread.cc
@@ -13,7 +13,6 @@
 #include "base/task_scheduler/task_scheduler.h"
 #include "base/task_scheduler/task_tracker.h"
 #include "base/task_scheduler/task_traits.h"
-#include "build/build_config.h"
 
 namespace base {
 namespace internal {
@@ -37,9 +36,6 @@
   // environment, do not perform the heartbeat report in that case since it
   // relies on such an environment.
   if (task_tracker_ && TaskScheduler::GetInstance()) {
-// Seemingly causing power regression on Android, disable to see if truly at
-// fault : https://crbug.com/848255
-#if !defined(OS_ANDROID)
     // Compute the histogram every hour (with a slight offset to drift if that
     // hour tick happens to line up with specific events). Once per hour per
     // user was deemed sufficient to gather a reliable metric.
@@ -51,7 +47,6 @@
                                           : g_heartbeat_for_testing,
         BindRepeating(&ServiceThread::PerformHeartbeatLatencyReport,
                       Unretained(this)));
-#endif
   }
 }
 
diff --git a/base/task_scheduler/service_thread_unittest.cc b/base/task_scheduler/service_thread_unittest.cc
index 3c42762..0214a72 100644
--- a/base/task_scheduler/service_thread_unittest.cc
+++ b/base/task_scheduler/service_thread_unittest.cc
@@ -52,18 +52,10 @@
   service_thread.FlushForTesting();
 }
 
-#if defined(OS_ANDROID)
-// The heartbeat latency report has been temporarily disabled on Android per
-// https://crbug.com/848255.
-#define MAYBE_HeartbeatLatencyReport DISABLED_HeartbeatLatencyReport
-#else
-#define MAYBE_HeartbeatLatencyReport HeartbeatLatencyReport
-#endif
-
 // Integration test verifying that a service thread running in a fully
 // integrated TaskScheduler environment results in reporting
 // HeartbeatLatencyMicroseconds metrics.
-TEST(TaskSchedulerServiceThreadIntegrationTest, MAYBE_HeartbeatLatencyReport) {
+TEST(TaskSchedulerServiceThreadIntegrationTest, HeartbeatLatencyReport) {
   ServiceThread::SetHeartbeatIntervalForTesting(TimeDelta::FromMilliseconds(1));
 
   TaskScheduler::SetInstance(
diff --git a/build/android/PRESUBMIT.py b/build/android/PRESUBMIT.py
index 65d37d1..9617b51 100644
--- a/build/android/PRESUBMIT.py
+++ b/build/android/PRESUBMIT.py
@@ -74,6 +74,7 @@
           J('pylib', 'results', 'json_results_test.py'),
           J('pylib', 'symbols', 'apk_native_libs_unittest.py'),
           J('pylib', 'symbols', 'elf_symbolizer_unittest.py'),
+          J('pylib', 'symbols', 'symbol_utils_unittest.py'),
           J('pylib', 'utils', 'decorators_test.py'),
           J('pylib', 'utils', 'device_dependencies_test.py'),
           J('pylib', 'utils', 'dexdump_test.py'),
diff --git a/build/android/asan_symbolize.py b/build/android/asan_symbolize.py
index 19d71ca..9f2e88a 100755
--- a/build/android/asan_symbolize.py
+++ b/build/android/asan_symbolize.py
@@ -14,6 +14,7 @@
 from pylib import constants
 from pylib.constants import host_paths
 
+# pylint: disable=wrong-import-order
 # Uses symbol.py from third_party/android_platform, not python's.
 with host_paths.SysPath(
     host_paths.ANDROID_PLATFORM_DEVELOPMENT_SCRIPTS_PATH,
diff --git a/build/android/devil_chromium.py b/build/android/devil_chromium.py
index af2141a..2ae64367 100644
--- a/build/android/devil_chromium.py
+++ b/build/android/devil_chromium.py
@@ -167,4 +167,3 @@
 
   devil_env.config.Initialize(
       configs=[devil_dynamic_config], config_files=[_DEVIL_CONFIG])
-
diff --git a/build/android/gyp/generate_resource_rewriter.py b/build/android/gyp/generate_resource_rewriter.py
index 82ddc21..ba635a29 100755
--- a/build/android/gyp/generate_resource_rewriter.py
+++ b/build/android/gyp/generate_resource_rewriter.py
@@ -107,4 +107,3 @@
 
 if __name__ == '__main__':
   sys.exit(main())
-
diff --git a/build/android/gyp/java_google_api_keys.py b/build/android/gyp/java_google_api_keys.py
index f57e0b3..349821a8 100755
--- a/build/android/gyp/java_google_api_keys.py
+++ b/build/android/gyp/java_google_api_keys.py
@@ -121,4 +121,3 @@
 
 if __name__ == '__main__':
   _DoMain(sys.argv[1:])
-
diff --git a/build/android/gyp/main_dex_list.py b/build/android/gyp/main_dex_list.py
index 9ccfea0..2435859 100755
--- a/build/android/gyp/main_dex_list.py
+++ b/build/android/gyp/main_dex_list.py
@@ -172,4 +172,3 @@
 
 if __name__ == '__main__':
   sys.exit(main(sys.argv[1:]))
-
diff --git a/build/android/gyp/util/__init__.py b/build/android/gyp/util/__init__.py
index 727e987e..96196cff 100644
--- a/build/android/gyp/util/__init__.py
+++ b/build/android/gyp/util/__init__.py
@@ -1,4 +1,3 @@
 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/gyp/write_ordered_libraries.py b/build/android/gyp/write_ordered_libraries.py
index e3d87c1..3b9a3374 100755
--- a/build/android/gyp/write_ordered_libraries.py
+++ b/build/android/gyp/write_ordered_libraries.py
@@ -114,5 +114,3 @@
 
 if __name__ == '__main__':
   sys.exit(main())
-
-
diff --git a/build/android/incremental_install/__init__.py b/build/android/incremental_install/__init__.py
index 1aaf0e1..50b23df 100644
--- a/build/android/incremental_install/__init__.py
+++ b/build/android/incremental_install/__init__.py
@@ -1,4 +1,3 @@
 # Copyright 2015 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/lighttpd_server.py b/build/android/lighttpd_server.py
index 89a828a..c77d740 100755
--- a/build/android/lighttpd_server.py
+++ b/build/android/lighttpd_server.py
@@ -104,7 +104,7 @@
         break
       self.process.close()
 
-      if self.fixed_port or not 'in use' in server_error:
+      if self.fixed_port or 'in use' not in server_error:
         print 'Client error:', client_error
         print 'Server error:', server_error
         return False
diff --git a/build/android/method_count.py b/build/android/method_count.py
index 68823e1..490887a 100755
--- a/build/android/method_count.py
+++ b/build/android/method_count.py
@@ -114,4 +114,3 @@
 
 if __name__ == '__main__':
   sys.exit(main())
-
diff --git a/build/android/pylib/android/__init__.py b/build/android/pylib/android/__init__.py
index 7a90b53..a67c3501 100644
--- a/build/android/pylib/android/__init__.py
+++ b/build/android/pylib/android/__init__.py
@@ -1,4 +1,3 @@
 # Copyright (c) 2016 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/base/__init__.py b/build/android/pylib/base/__init__.py
index 727e987e..96196cff 100644
--- a/build/android/pylib/base/__init__.py
+++ b/build/android/pylib/base/__init__.py
@@ -1,4 +1,3 @@
 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/base/base_test_result.py b/build/android/pylib/base/base_test_result.py
index af9f68f8..a553532 100644
--- a/build/android/pylib/base/base_test_result.py
+++ b/build/android/pylib/base/base_test_result.py
@@ -260,4 +260,3 @@
   def DidRunPass(self):
     """Return whether the test run was successful."""
     return not self.GetNotPass() - self.GetSkip()
-
diff --git a/build/android/pylib/base/test_exception.py b/build/android/pylib/base/test_exception.py
index f00f0d0e..c98d2cb 100644
--- a/build/android/pylib/base/test_exception.py
+++ b/build/android/pylib/base/test_exception.py
@@ -6,4 +6,3 @@
 class TestException(Exception):
   """Base class for exceptions thrown by the test runner."""
   pass
-
diff --git a/build/android/pylib/base/test_instance.py b/build/android/pylib/base/test_instance.py
index cdf678f2..f0ff830 100644
--- a/build/android/pylib/base/test_instance.py
+++ b/build/android/pylib/base/test_instance.py
@@ -32,4 +32,3 @@
 
   def __exit__(self, _exc_type, _exc_val, _exc_tb):
     self.TearDown()
-
diff --git a/build/android/pylib/base/test_run.py b/build/android/pylib/base/test_run.py
index 9b16f89..59e595d 100644
--- a/build/android/pylib/base/test_run.py
+++ b/build/android/pylib/base/test_run.py
@@ -41,4 +41,3 @@
 
   def __exit__(self, exc_type, exc_val, exc_tb):
     self.TearDown()
-
diff --git a/build/android/pylib/base/test_run_factory.py b/build/android/pylib/base/test_run_factory.py
index 1a28728..1f63a059 100644
--- a/build/android/pylib/base/test_run_factory.py
+++ b/build/android/pylib/base/test_run_factory.py
@@ -54,4 +54,3 @@
 
   error_func('Unable to create test run for %s tests in %s environment'
              % (str(test_instance), str(env)))
-
diff --git a/build/android/pylib/base/test_server.py b/build/android/pylib/base/test_server.py
index 085a51e9..763e1212 100644
--- a/build/android/pylib/base/test_server.py
+++ b/build/android/pylib/base/test_server.py
@@ -16,4 +16,3 @@
 
   def TearDown(self):
     raise NotImplementedError
-
diff --git a/build/android/pylib/gtest/__init__.py b/build/android/pylib/gtest/__init__.py
index 727e987e..96196cff 100644
--- a/build/android/pylib/gtest/__init__.py
+++ b/build/android/pylib/gtest/__init__.py
@@ -1,4 +1,3 @@
 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/gtest/gtest_test_instance.py b/build/android/pylib/gtest/gtest_test_instance.py
index d93edc6..7e0923a 100644
--- a/build/android/pylib/gtest/gtest_test_instance.py
+++ b/build/android/pylib/gtest/gtest_test_instance.py
@@ -555,4 +555,3 @@
   def TearDown(self):
     """Do nothing."""
     pass
-
diff --git a/build/android/pylib/gtest/gtest_test_instance_test.py b/build/android/pylib/gtest/gtest_test_instance_test.py
index 3466924..a34ab84 100755
--- a/build/android/pylib/gtest/gtest_test_instance_test.py
+++ b/build/android/pylib/gtest/gtest_test_instance_test.py
@@ -260,4 +260,3 @@
 
 if __name__ == '__main__':
   unittest.main(verbosity=2)
-
diff --git a/build/android/pylib/instrumentation/__init__.py b/build/android/pylib/instrumentation/__init__.py
index 727e987e..96196cff 100644
--- a/build/android/pylib/instrumentation/__init__.py
+++ b/build/android/pylib/instrumentation/__init__.py
@@ -1,4 +1,3 @@
 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/junit/__init__.py b/build/android/pylib/junit/__init__.py
index 5cac0261..4d6aabb9 100644
--- a/build/android/pylib/junit/__init__.py
+++ b/build/android/pylib/junit/__init__.py
@@ -1,4 +1,3 @@
 # Copyright 2014 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/linker/__init__.py b/build/android/pylib/linker/__init__.py
index af99437..9228df89 100644
--- a/build/android/pylib/linker/__init__.py
+++ b/build/android/pylib/linker/__init__.py
@@ -1,4 +1,3 @@
 # Copyright 2013 The Chromium Authors. All rights reserved.
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
-
diff --git a/build/android/pylib/linker/linker_test_instance.py b/build/android/pylib/linker/linker_test_instance.py
index 1de3e17..6ace7a36 100644
--- a/build/android/pylib/linker/linker_test_instance.py
+++ b/build/android/pylib/linker/linker_test_instance.py
@@ -48,4 +48,3 @@
 
   def TestType(self):
     return 'linker'
-
diff --git a/build/android/pylib/local/device/local_device_gtest_run.py b/build/android/pylib/local/device/local_device_gtest_run.py
index 21e9fe85..4734883 100644
--- a/build/android/pylib/local/device/local_device_gtest_run.py
+++ b/build/android/pylib/local/device/local_device_gtest_run.py
@@ -152,6 +152,7 @@
       extras[gtest_test_instance.EXTRA_SHARD_NANO_TIMEOUT] = int(
           kwargs['timeout'] * _SECONDS_TO_NANOS)
 
+    # pylint: disable=redefined-variable-type
     command_line_file = _NullContextManager()
     if flags:
       if len(flags) > _MAX_INLINE_FLAGS_LENGTH:
@@ -169,6 +170,7 @@
         extras[_EXTRA_TEST_LIST] = test_list_file.name
       else:
         extras[_EXTRA_TEST] = test[0]
+    # pylint: enable=redefined-variable-type
 
     stdout_file = device_temp_file.DeviceTempFile(
         device.adb, dir=device.GetExternalStoragePath(), suffix='.gtest_out')
@@ -283,11 +285,13 @@
     assert isinstance(test_instance, gtest_test_instance.GtestTestInstance)
     super(LocalDeviceGtestRun, self).__init__(env, test_instance)
 
+    # pylint: disable=redefined-variable-type
     if self._test_instance.apk:
       self._delegate = _ApkDelegate(self._test_instance, env.tool)
     elif self._test_instance.exe_dist_dir:
       self._delegate = _ExeDelegate(self, self._test_instance.exe_dist_dir,
                                     self._env.tool)
+    # pylint: enable=redefined-variable-type
     self._crashes = set()
     self._servers = collections.defaultdict(list)
 
diff --git a/build/android/pylib/local/device/local_device_instrumentation_test_run.py b/build/android/pylib/local/device/local_device_instrumentation_test_run.py
index 40b857d..de7ed93 100644
--- a/build/android/pylib/local/device/local_device_instrumentation_test_run.py
+++ b/build/android/pylib/local/device/local_device_instrumentation_test_run.py
@@ -153,7 +153,11 @@
           self._replace_package_contextmanager = system_app.ReplaceSystemApp(
               dev, self._test_instance.replace_system_package.package,
               self._test_instance.replace_system_package.replacement_apk)
+          # Pylint is not smart enough to realize that this field has
+          # an __enter__ method, and will complain loudly.
+          # pylint: disable=no-member
           self._replace_package_contextmanager.__enter__()
+          # pylint: enable=no-member
 
         steps.append(replace_package)
 
@@ -308,12 +312,15 @@
       valgrind_tools.SetChromeTimeoutScale(dev, None)
 
       if self._replace_package_contextmanager:
+        # See pylint-related commend above with __enter__()
+        # pylint: disable=no-member
         self._replace_package_contextmanager.__exit__(*sys.exc_info())
+        # pylint: enable=no-member
 
     self._env.parallel_devices.pMap(individual_device_tear_down)
 
   def _CreateFlagChangerIfNeeded(self, device):
-    if not str(device) in self._flag_changers:
+    if str(device) not in self._flag_changers:
       self._flag_changers[str(device)] = flag_changer.FlagChanger(
         device, "test-cmdline-file")
 
diff --git a/build/android/pylib/local/device/local_device_test_run.py b/build/android/pylib/local/device/local_device_test_run.py
index 90e6b67..00d715b 100644
--- a/build/android/pylib/local/device/local_device_test_run.py
+++ b/build/android/pylib/local/device/local_device_test_run.py
@@ -220,7 +220,7 @@
         if hash(self._GetUniqueTestName(t)) % total_shards == shard_index]
 
   def GetTool(self, device):
-    if not str(device) in self._tools:
+    if str(device) not in self._tools:
       self._tools[str(device)] = valgrind_tools.CreateTool(
           self._env.tool, device)
     return self._tools[str(device)]
diff --git a/build/android/pylib/local/local_test_server_spawner.py b/build/android/pylib/local/local_test_server_spawner.py
index 8e416fac..6cd282e 100644
--- a/build/android/pylib/local/local_test_server_spawner.py
+++ b/build/android/pylib/local/local_test_server_spawner.py
@@ -98,4 +98,3 @@
     self.Reset()
     self._spawning_server.Stop()
     forwarder.Forwarder.UnmapDevicePort(self.port, self._device)
-
diff --git a/build/android/pylib/results/flakiness_dashboard/json_results_generator.py b/build/android/pylib/results/flakiness_dashboard/json_results_generator.py
index 9693977..5e5f83f 100644
--- a/build/android/pylib/results/flakiness_dashboard/json_results_generator.py
+++ b/build/android/pylib/results/flakiness_dashboard/json_results_generator.py
@@ -380,6 +380,7 @@
                          urllib2.quote(self._test_type),
                          urllib2.quote(self._master_name)))
 
+    # pylint: disable=redefined-variable-type
     try:
       # FIXME: We should talk to the network via a Host object.
       results_file = urllib2.urlopen(results_file_url)
@@ -391,6 +392,7 @@
         error = http_error
     except urllib2.URLError, url_error:
       error = url_error
+    # pylint: enable=redefined-variable-type
 
     if old_results:
       # Strip the prefix and suffix so we can get the actual JSON object.
diff --git a/build/android/pylib/results/json_results.py b/build/android/pylib/results/json_results.py
index 1243691..3f87b46 100644
--- a/build/android/pylib/results/json_results.py
+++ b/build/android/pylib/results/json_results.py
@@ -175,4 +175,3 @@
                                            duration=tr['elapsed_time_ms'])
           for tr in test_runs])
   return results_list
-
diff --git a/build/android/pylib/results/json_results_test.py b/build/android/pylib/results/json_results_test.py
index 27617135..e8b983b5 100755
--- a/build/android/pylib/results/json_results_test.py
+++ b/build/android/pylib/results/json_results_test.py
@@ -178,4 +178,3 @@
 
 if __name__ == '__main__':
   unittest.main(verbosity=2)
-
diff --git a/build/android/pylib/results/presentation/test_results_presentation.py b/build/android/pylib/results/presentation/test_results_presentation.py
index eaae68f85..82d6c88 100755
--- a/build/android/pylib/results/presentation/test_results_presentation.py
+++ b/build/android/pylib/results/presentation/test_results_presentation.py
@@ -268,6 +268,7 @@
 
 
 def feedback_url(result_details_link):
+  # pylint: disable=redefined-variable-type
   url_args = [
       ('labels', 'Pri-2,Type-Bug,Restrict-View-Google'),
       ('summary', 'Result Details Feedback:'),
@@ -276,6 +277,7 @@
   if result_details_link:
     url_args.append(('comment', 'Please check out: %s' % result_details_link))
   url_args = urllib.urlencode(url_args)
+  # pylint: enable=redefined-variable-type
   return 'https://bugs.chromium.org/p/chromium/issues/entry?%s' % url_args
 
 
@@ -370,6 +372,7 @@
     # This will be reported as an error by result_details, no need to duplicate.
     return None
   ui_screenshots = []
+  # pylint: disable=too-many-nested-blocks
   for testsuite_run in json_object['per_iteration_data']:
     for _, test_runs in testsuite_run.iteritems():
       for test_run in test_runs:
@@ -388,6 +391,7 @@
             test_screenshots = json.loads(
                 screenshot_string)
           ui_screenshots.extend(test_screenshots)
+  # pylint: enable=too-many-nested-blocks
 
   if ui_screenshots:
     return json.dumps(ui_screenshots)
diff --git a/build/android/pylib/symbols/symbol_utils.py b/build/android/pylib/symbols/symbol_utils.py
new file mode 100644
index 0000000..e4e3faa
--- /dev/null
+++ b/build/android/pylib/symbols/symbol_utils.py
@@ -0,0 +1,812 @@
+# Copyright 2018 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+import bisect
+import collections
+import logging
+import os
+import re
+
+from pylib.constants import host_paths
+from pylib.symbols import elf_symbolizer
+
+
+def _AndroidAbiToCpuArch(android_abi):
+  """Return the Chromium CPU architecture name for a given Android ABI."""
+  _ARCH_MAP = {
+    'armeabi': 'arm',
+    'armeabi-v7a': 'arm',
+    'arm64-v8a': 'arm64',
+    'x86_64': 'x64',
+  }
+  return _ARCH_MAP.get(android_abi, android_abi)
+
+
+def _HexAddressRegexpFor(android_abi):
+  """Return a regexp matching hexadecimal addresses for a given Android ABI."""
+  if android_abi in ['x86_64', 'arm64-v8a', 'mips64']:
+    width = 16
+  else:
+    width = 8
+  return '[0-9a-f]{%d}' % width
+
+
+class HostLibraryFinder(object):
+  """Translate device library path to matching host unstripped library path.
+
+  Usage is the following:
+    1) Create instance.
+    2) Call AddSearchDir() once or more times to add host directory path to
+       look for unstripped native libraries.
+    3) Call Find(device_libpath) repeatedly to translate a device-specific
+       library path into the corresponding host path to the unstripped
+       version.
+  """
+  def __init__(self):
+    """Initialize instance."""
+    self._search_dirs = []
+    self._lib_map = {}        # Map of library name to host file paths.
+
+  def AddSearchDir(self, lib_dir):
+    """Add a directory to the search path for host native shared libraries.
+
+    Args:
+      lib_dir: host path containing native libraries.
+    """
+    if not os.path.exists(lib_dir):
+      logging.warning('Ignoring missing host library directory: %s', lib_dir)
+      return
+    if not os.path.isdir(lib_dir):
+      logging.warning('Ignoring invalid host library directory: %s', lib_dir)
+      return
+    self._search_dirs.append(lib_dir)
+    self._lib_map = {}  # Reset the map.
+
+  def Find(self, device_libpath):
+    """Find the host file path matching a specific device library path.
+
+    Args:
+      device_libpath: device-specific file path to library or executable.
+    Returns:
+      host file path to the unstripped version of the library, or None.
+    """
+    host_lib_path = None
+    lib_name = os.path.basename(device_libpath)
+    host_lib_path = self._lib_map.get(lib_name)
+    if not host_lib_path:
+      for search_dir in self._search_dirs:
+        lib_path = os.path.join(search_dir, lib_name)
+        if os.path.exists(lib_path):
+          host_lib_path = lib_path
+          break
+
+      if not host_lib_path:
+        logging.debug('Could not find host library for: %s', lib_name)
+      self._lib_map[lib_name] = host_lib_path
+
+    return host_lib_path
+
+
+
+class SymbolResolver(object):
+  """A base class for objets that can symbolize library (path, offset)
+     pairs into symbol information strings. Usage is the following:
+
+     1) Create new instance (by calling the constructor of a derived
+        class, since this is only the base one).
+
+     2) Call SetAndroidAbi() before any call to FindSymbolInfo() in order
+        to set the Android CPU ABI used for symbolization.
+
+     3) Before the first call to FindSymbolInfo(), one can call
+        AddLibraryOffset(), or AddLibraryOffsets() to record a set of offsets
+        that you will want to symbolize later through FindSymbolInfo(). Doing
+        so allows some SymbolResolver derived classes to work faster (e.g. the
+        one that invokes the 'addr2line' program, since the latter works faster
+        if the offsets provided as inputs are sorted in increasing order).
+
+     3) Call FindSymbolInfo(path, offset) to return the corresponding
+        symbol information string, or None if this doesn't correspond
+        to anything the instance can handle.
+
+        Note that whether the path is specific to the device or to the
+        host depends on the derived class implementation.
+  """
+  def __init__(self):
+    self._android_abi = None
+    self._lib_offsets_map = collections.defaultdict(set)
+
+  def SetAndroidAbi(self, android_abi):
+    """Set the Android ABI value for this instance.
+
+    Calling this function before FindSymbolInfo() is required by some
+    derived class implementations.
+
+    Args:
+      android_abi: Native Android CPU ABI name (e.g. 'armeabi-v7a').
+    Raises:
+      Exception if the ABI was already set with a different value.
+    """
+    if self._android_abi and self._android_abi != android_abi:
+      raise Exception('Cannot reset Android ABI to new value %s, already set '
+                      'to %s' % (android_abi, self._android_abi))
+
+    self._android_abi = android_abi
+
+  def AddLibraryOffset(self, lib_path, offset):
+    """Associate a single offset to a given device library.
+
+    This must be called before FindSymbolInfo(), otherwise its input arguments
+    will be ignored.
+
+    Args:
+      lib_path: A library path.
+      offset: An integer offset within the corresponding library that will be
+        symbolized by future calls to FindSymbolInfo.
+    """
+    self._lib_offsets_map[lib_path].add(offset)
+
+  def AddLibraryOffsets(self, lib_path, lib_offsets):
+    """Associate a set of wanted offsets to a given device library.
+
+    This must be called before FindSymbolInfo(), otherwise its input arguments
+    will be ignored.
+
+    Args:
+      lib_path: A library path.
+      lib_offsets: An iterable of integer offsets within the corresponding
+        library that will be symbolized by future calls to FindSymbolInfo.
+    """
+    self._lib_offsets_map[lib_path].update(lib_offsets)
+
+  # pylint: disable=unused-argument,no-self-use
+  def FindSymbolInfo(self, lib_path, lib_offset):
+    """Symbolize a device library path and offset.
+
+    Args:
+      lib_path: Library path (device or host specific, depending on the
+        derived class implementation).
+      lib_offset: Integer offset within the library.
+    Returns:
+      Corresponding symbol information string, or None.
+    """
+    # The base implementation cannot symbolize anything.
+    return None
+  # pylint: enable=unused-argument,no-self-use
+
+
+class ElfSymbolResolver(SymbolResolver):
+  """A SymbolResolver that can symbolize host path + offset values using
+     an elf_symbolizer.ELFSymbolizer instance.
+  """
+  def __init__(self, addr2line_path_for_tests=None):
+    super(ElfSymbolResolver, self).__init__()
+    self._addr2line_path = addr2line_path_for_tests
+
+    # Used to cache one ELFSymbolizer instance per library path.
+    self._elf_symbolizer_cache = {}
+
+    # Used to cache FindSymbolInfo() results. Maps host library paths
+    # to (offset -> symbol info string) dictionaries.
+    self._symbol_info_cache = collections.defaultdict(dict)
+    self._allow_symbolizer = True
+
+  def _CreateSymbolizerFor(self, host_path):
+    """Create the ELFSymbolizer instance associated with a given lib path."""
+    addr2line_path = self._addr2line_path
+    if not addr2line_path:
+      if not self._android_abi:
+        raise Exception(
+            'Android CPU ABI must be set before calling FindSymbolInfo!')
+
+      cpu_arch = _AndroidAbiToCpuArch(self._android_abi)
+      self._addr2line_path = host_paths.ToolPath('addr2line', cpu_arch)
+
+    return elf_symbolizer.ELFSymbolizer(
+        elf_file_path=host_path, addr2line_path=self._addr2line_path,
+        callback=ElfSymbolResolver._Callback, inlines=True)
+
+  def DisallowSymbolizerForTesting(self):
+    """Disallow FindSymbolInfo() from using a symbolizer.
+
+    This is used during unit-testing to ensure that the offsets that were
+    recorded via AddLibraryOffset()/AddLibraryOffsets() are properly
+    symbolized, but not anything else.
+    """
+    self._allow_symbolizer = False
+
+  def FindSymbolInfo(self, host_path, offset):
+    """Override SymbolResolver.FindSymbolInfo.
+
+    Args:
+      host_path: Host-specific path to the native shared library.
+      offset: Integer offset within the native library.
+    Returns:
+      A symbol info string, or None.
+    """
+    offset_map = self._symbol_info_cache[host_path]
+    symbol_info = offset_map.get(offset)
+    if symbol_info:
+      return symbol_info
+
+    # Create symbolizer on demand.
+    symbolizer = self._elf_symbolizer_cache.get(host_path)
+    if not symbolizer:
+      symbolizer = self._CreateSymbolizerFor(host_path)
+      self._elf_symbolizer_cache[host_path] = symbolizer
+
+      # If there are pre-recorded offsets for this path, symbolize them now.
+      offsets = self._lib_offsets_map.get(host_path)
+      if offsets:
+        offset_map = {}
+        for pre_offset in offsets:
+          symbolizer.SymbolizeAsync(
+              pre_offset, callback_arg=(offset_map, pre_offset))
+        symbolizer.WaitForIdle()
+        self._symbol_info_cache[host_path] = offset_map
+
+        symbol_info = offset_map.get(offset)
+        if symbol_info:
+          return symbol_info
+
+    if not self._allow_symbolizer:
+      return None
+
+    # Symbolize single offset. Slower if addresses are not provided in
+    # increasing order to addr2line.
+    symbolizer.SymbolizeAsync(offset,
+                              callback_arg=(offset_map, offset))
+    symbolizer.WaitForIdle()
+    return offset_map.get(offset)
+
+  @staticmethod
+  def _Callback(sym_info, callback_arg):
+    offset_map, offset = callback_arg
+    offset_map[offset] = str(sym_info)
+
+
+class DeviceSymbolResolver(SymbolResolver):
+  """A SymbolResolver instance that accepts device-specific path.
+
+  Usage is the following:
+    1) Create new instance, passing a parent SymbolResolver instance that
+       accepts host-specific paths, and a HostLibraryFinder instance.
+
+    2) Optional: call AddApkOffsets() to add offsets from within an APK
+       that contains uncompressed native shared libraries.
+
+    3) Use it as any SymbolResolver instance.
+  """
+  def __init__(self, host_resolver, host_lib_finder):
+    """Initialize instance.
+
+    Args:
+      host_resolver: A parent SymbolResolver instance that will be used
+        to resolve symbols from host library paths.
+      host_lib_finder: A HostLibraryFinder instance used to locate
+        unstripped libraries on the host.
+    """
+    super(DeviceSymbolResolver, self).__init__()
+    self._host_lib_finder = host_lib_finder
+    self._bad_device_lib_paths = set()
+    self._host_resolver = host_resolver
+
+  def SetAndroidAbi(self, android_abi):
+    super(DeviceSymbolResolver, self).SetAndroidAbi(android_abi)
+    self._host_resolver.SetAndroidAbi(android_abi)
+
+  def AddLibraryOffsets(self, device_lib_path, lib_offsets):
+    """Associate a set of wanted offsets to a given device library.
+
+    This must be called before FindSymbolInfo(), otherwise its input arguments
+    will be ignored.
+
+    Args:
+      device_lib_path: A device-specific library path.
+      lib_offsets: An iterable of integer offsets within the corresponding
+        library that will be symbolized by future calls to FindSymbolInfo.
+        want to symbolize.
+    """
+    if device_lib_path in self._bad_device_lib_paths:
+      return
+
+    host_lib_path = self._host_lib_finder.Find(device_lib_path)
+    if not host_lib_path:
+      # NOTE: self._bad_device_lib_paths is only used to only print this
+      #       warning once per bad library.
+      logging.warning('Could not find host library matching device path: %s',
+                      device_lib_path)
+      self._bad_device_lib_paths.add(device_lib_path)
+      return
+
+    self._host_resolver.AddLibraryOffsets(host_lib_path, lib_offsets)
+
+  def AddApkOffsets(self, device_apk_path, apk_offsets, apk_translator):
+    """Associate a set of wanted offsets to a given device APK path.
+
+    This converts the APK-relative offsets into offsets relative to the
+    uncompressed libraries it contains, then calls AddLibraryOffsets()
+    for each one of the libraries.
+
+    Must be called before FindSymbolInfo() as well, otherwise input arguments
+    will be ignored.
+
+    Args:
+      device_apk_path: Device-specific APK path.
+      apk_offsets: Iterable of offsets within the APK file.
+      apk_translator: An ApkLibraryPathTranslator instance used to extract
+        library paths from the APK.
+    """
+    libraries_map = collections.defaultdict(set)
+    for offset in apk_offsets:
+      lib_path, lib_offset = apk_translator.TranslatePath(device_apk_path,
+                                                          offset)
+      libraries_map[lib_path].add(lib_offset)
+
+    for lib_path, lib_offsets in libraries_map.iteritems():
+      self.AddLibraryOffsets(lib_path, lib_offsets)
+
+  def FindSymbolInfo(self, device_path, offset):
+    """Overrides SymbolResolver.FindSymbolInfo.
+
+    Args:
+      device_path: Device-specific library path (e.g.
+        '/data/app/com.example.app-1/lib/x86/libfoo.so')
+      offset: Offset in device library path.
+    Returns:
+      Corresponding symbol information string, or None.
+    """
+    host_path = self._host_lib_finder.Find(device_path)
+    if not host_path:
+      return None
+
+    return self._host_resolver.FindSymbolInfo(host_path, offset)
+
+
+class MemoryMap(object):
+  """Models the memory map of a given process. Usage is:
+
+    1) Create new instance, passing the Android ABI.
+
+    2) Call TranslateLine() whenever you want to detect and translate any
+       memory map input line.
+
+    3) Otherwise, it is possible to parse the whole memory map input with
+       ParseLines(), then call FindSectionForAddress() repeatedly in order
+       to translate a memory address into the corresponding mapping and
+       file information tuple (e.g. to symbolize stack entries).
+  """
+
+  # A named tuple describing interesting memory map line items.
+  # Fields:
+  #   addr_start: Mapping start address in memory.
+  #   file_offset: Corresponding file offset.
+  #   file_size: Corresponding mapping size in bytes.
+  #   file_path: Input file path.
+  #   match: Corresponding regular expression match object.
+  LineTuple = collections.namedtuple('MemoryMapLineTuple',
+                                     'addr_start,file_offset,file_size,'
+                                     'file_path, match')
+
+  # A name tuple describing a memory map section.
+  # Fields:
+  #   address: Memory address.
+  #   size: Size in bytes in memory
+  #   offset: Starting file offset.
+  #   path: Input file path.
+  SectionTuple = collections.namedtuple('MemoryMapSection',
+                                        'address,size,offset,path')
+
+  def __init__(self, android_abi):
+    """Initializes instance.
+
+    Args:
+      android_abi: Android CPU ABI name (e.g. 'armeabi-v7a')
+    """
+    hex_addr = _HexAddressRegexpFor(android_abi)
+
+    # pylint: disable=line-too-long
+    # A regular expression used to match memory map entries which look like:
+    #    b278c000-b2790fff r--   4fda000      5000  /data/app/com.google.android.apps.chrome-2/base.apk
+    # pylint: enable=line-too-long
+    self._re_map_section = re.compile(
+        r'\s*(?P<addr_start>' + hex_addr + r')-(?P<addr_end>' + hex_addr + ')' +
+        r'\s+' +
+        r'(?P<perm>...)\s+' +
+        r'(?P<file_offset>[0-9a-f]+)\s+' +
+        r'(?P<file_size>[0-9a-f]+)\s*' +
+        r'(?P<file_path>[^ \t]+)?')
+
+    self._addr_map = []  # Sorted list of (address, size, path, offset) tuples.
+    self._sorted_addresses = []  # Sorted list of address fields in _addr_map.
+    self._in_section = False
+
+  def TranslateLine(self, line, apk_path_translator):
+    """Try to translate a memory map input line, if detected.
+
+    This only takes care of converting mapped APK file path and offsets
+    into a corresponding uncompressed native library file path + new offsets,
+    e.g. '..... <offset> <size> /data/.../base.apk' gets
+    translated into '.... <new-offset> <size> /data/.../base.apk!lib/libfoo.so'
+
+    This function should always work, even if ParseLines() was not called
+    previously.
+
+    Args:
+      line: Input memory map / tombstone line.
+      apk_translator: An ApkLibraryPathTranslator instance, used to map
+        APK offsets into uncompressed native libraries + new offsets.
+    Returns:
+      Translated memory map line, if relevant, or unchanged input line
+      otherwise.
+    """
+    t = self._ParseLine(line.rstrip())
+    if not t:
+      return line
+
+    new_path, new_offset = apk_path_translator.TranslatePath(
+        t.file_path, t.file_offset)
+
+    if new_path == t.file_path:
+      return line
+
+    pos = t.match.start('file_path')
+    return '%s%s (offset 0x%x)%s' % (line[0:pos], new_path, new_offset,
+                                     line[t.match.end('file_path'):])
+
+  def ParseLines(self, input_lines, in_section=False):
+    """Parse a list of input lines and extract the APK memory map out of it.
+
+    Args:
+      input_lines: list, or iterable, of input lines.
+      in_section: Optional. If true, considers that the input lines are
+        already part of the memory map. Otherwise, wait until the start of
+        the section appears in the input before trying to record data.
+    Returns:
+      True iff APK-related memory map entries were found. False otherwise.
+    """
+    addr_list = []  # list of (address, size, file_path, file_offset) tuples.
+    self._in_section = in_section
+    for line in input_lines:
+      t = self._ParseLine(line.rstrip())
+      if not t:
+        continue
+
+      addr_list.append(t)
+
+    self._addr_map = sorted(addr_list,
+                            lambda x, y: cmp(x.addr_start, y.addr_start))
+    self._sorted_addresses = [e.addr_start for e in self._addr_map]
+    return bool(self._addr_map)
+
+  def _ParseLine(self, line):
+    """Used internally to recognized memory map input lines.
+
+    Args:
+      line: Input logcat or tomstone line.
+    Returns:
+      A LineTuple instance on success, or None on failure.
+    """
+    if not self._in_section:
+      self._in_section = line.startswith('memory map:')
+      return None
+
+    m = self._re_map_section.match(line)
+    if not m:
+      self._in_section = False  # End of memory map section
+      return None
+
+    # Only accept .apk and .so files that are not from the system partitions.
+    file_path = m.group('file_path')
+    if not file_path:
+      return None
+
+    if file_path.startswith('/system') or file_path.startswith('/vendor'):
+      return None
+
+    if not (file_path.endswith('.apk') or file_path.endswith('.so')):
+      return None
+
+    addr_start = int(m.group('addr_start'), 16)
+    file_offset = int(m.group('file_offset'), 16)
+    file_size = int(m.group('file_size'), 16)
+
+    return self.LineTuple(addr_start, file_offset, file_size, file_path, m)
+
+  def Dump(self):
+    """Print memory map for debugging."""
+    print 'MEMORY MAP ['
+    for t in self._addr_map:
+      print '[%08x-%08x %08x %08x %s]' % (
+          t.addr_start, t.addr_start + t.file_size, t.file_size, t.file_offset,
+          t.file_path)
+    print '] MEMORY MAP'
+
+  def FindSectionForAddress(self, addr):
+    """Find the map section corresponding to a specific memory address.
+
+    Call this method only after using ParseLines() was called to extract
+    relevant information from the memory map.
+
+    Args:
+      addr: Memory address
+    Returns:
+      A SectionTuple instance on success, or None on failure.
+    """
+    pos = bisect.bisect_right(self._sorted_addresses, addr)
+    if pos > 0:
+      # All values in [0,pos) are <= addr, just ensure that the last
+      # one contains the address as well.
+      entry = self._addr_map[pos - 1]
+      if entry.addr_start + entry.file_size > addr:
+        return self.SectionTuple(entry.addr_start, entry.file_size,
+                                 entry.file_offset, entry.file_path)
+    return None
+
+
+class BacktraceTranslator(object):
+  """Translates backtrace-related lines in a tombstone or crash report.
+
+  Usage is the following:
+    1) Create new instance with appropriate arguments.
+    2) If the tombstone / logcat input is available, one can call
+       FindLibraryOffsets() in order to detect which library offsets
+       will need to be symbolized during a future parse. Doing so helps
+       speed up the ELF symbolizer.
+    3) For each tombstone/logcat input line, call TranslateLine() to
+       try to detect and symbolize backtrace lines.
+  """
+
+  # A named tuple for relevant input backtrace lines.
+  # Fields:
+  #   rel_pc: Instruction pointer, relative to offset in library start.
+  #   location: Library or APK file path.
+  #   offset: Load base of executable code in library or apk file path.
+  #   match: The corresponding regular expression match object.
+  # Note:
+  #   The actual instruction pointer always matches the position at
+  #   |offset + rel_pc| in |location|.
+  LineTuple = collections.namedtuple('BacktraceLineTuple',
+                                      'rel_pc,location,offset,match')
+
+  def __init__(self, android_abi, apk_translator):
+    """Initialize instance.
+
+    Args:
+      android_abi: Android CPU ABI name (e.g. 'armeabi-v7a').
+      apk_translator: ApkLibraryPathTranslator instance used to convert
+        mapped APK file offsets into uncompressed library file paths with
+        new offsets.
+    """
+    hex_addr = _HexAddressRegexpFor(android_abi)
+
+    # A regular expression used to match backtrace lines.
+    self._re_backtrace = re.compile(
+        r'.*#(?P<frame>[0-9]{2})\s+' +
+        r'(..)\s+' +
+        r'(?P<rel_pc>' + hex_addr + r')\s+' +
+        r'(?P<location>[^ \t]+)' +
+        r'(\s+\(offset 0x(?P<offset>[0-9a-f]+)\))?')
+
+    # In certain cases, offset will be provided as <location>+0x<offset>
+    # instead of <location> (offset 0x<offset>). This is a regexp to detect
+    # this.
+    self._re_location_offset = re.compile(
+        r'.*\+0x(?P<offset>[0-9a-f]+)$')
+
+    self._apk_translator = apk_translator
+    self._in_section = False
+
+  def _ParseLine(self, line):
+    """Used internally to detect and decompose backtrace input lines.
+
+    Args:
+      line: input tombstone line.
+    Returns:
+      A LineTuple instance on success, None on failure.
+    """
+    if not self._in_section:
+      self._in_section = line.startswith('backtrace:')
+      return None
+
+    line = line.rstrip()
+    m = self._re_backtrace.match(line)
+    if not m:
+      self._in_section = False
+      return None
+
+    location = m.group('location')
+    offset = m.group('offset')
+    if not offset:
+      m2 = self._re_location_offset.match(location)
+      if m2:
+        offset = m2.group('offset')
+        location = location[0:m2.start('offset') - 3]
+
+    if not offset:
+      return None
+
+    offset = int(offset, 16)
+    rel_pc = int(m.group('rel_pc'), 16)
+
+    # Two cases to consider here:
+    #
+    # * If this is a library file directly mapped in memory, then |rel_pc|
+    #   if the direct offset within the library, and doesn't need any kind
+    #   of adjustement.
+    #
+    # * If this is a library mapped directly from an .apk file, then
+    #   |rel_pc| is the offset in the APK, and |offset| happens to be the
+    #   load base of the corresponding library.
+    #
+    if location.endswith('.so'):
+      # For a native library directly mapped from the file system,
+      return self.LineTuple(rel_pc, location, offset, m)
+
+    if location.endswith('.apk'):
+      # For a native library inside an memory-mapped APK file,
+      new_location, new_offset = self._apk_translator.TranslatePath(
+          location, offset)
+
+      return self.LineTuple(rel_pc, new_location, new_offset, m)
+
+    # Ignore anything else (e.g. .oat or .odex files).
+    return None
+
+  def FindLibraryOffsets(self, input_lines, in_section=False):
+    """Parse a tombstone's backtrace section and find all library offsets in it.
+
+    Args:
+      input_lines: List or iterables of intput tombstone lines.
+      in_section: Optional. If True, considers that the stack section has
+        already started.
+    Returns:
+      A dictionary mapping device library paths to sets of offsets within
+      then.
+    """
+    self._in_section = in_section
+    result = collections.defaultdict(set)
+    for line in input_lines:
+      t = self._ParseLine(line)
+      if not t:
+        continue
+
+      result[t.location].add(t.offset + t.rel_pc)
+    return result
+
+  def TranslateLine(self, line, symbol_resolver):
+    """Symbolize backtrace line if recognized.
+
+    Args:
+      line: input backtrace line.
+      symbol_resolver: symbol resolver instance to use. This method will
+        call its FindSymbolInfo(device_lib_path, lib_offset) method to
+        convert offsets into symbol informations strings.
+    Returns:
+      Translated line (unchanged if not recognized as a back trace).
+    """
+    t = self._ParseLine(line)
+    if not t:
+      return line
+
+    symbol_info = symbol_resolver.FindSymbolInfo(t.location,
+                                                 t.offset + t.rel_pc)
+    if not symbol_info:
+      symbol_info = 'offset 0x%x' % t.offset
+
+    pos = t.match.start('location')
+    pos2 = t.match.end('offset') + 1
+    if pos2 <= 0:
+      pos2 = t.match.end('location')
+    return '%s%s (%s)%s' % (line[:pos], t.location, symbol_info, line[pos2:])
+
+
+class StackTranslator(object):
+  """Translates stack-related lines in a tombstone or crash report."""
+
+  # A named tuple describing relevant stack input lines.
+  # Fields:
+  #  address: Address as it appears in the stack.
+  #  lib_path: Library path where |address| is mapped.
+  #  lib_offset: Library load base offset. for |lib_path|.
+  #  match: Corresponding regular expression match object.
+  LineTuple = collections.namedtuple('StackLineTuple',
+                                     'address, lib_path, lib_offset, match')
+
+  def __init__(self, android_abi, memory_map, apk_translator):
+    """Initialize instance."""
+    hex_addr = _HexAddressRegexpFor(android_abi)
+
+    # pylint: disable=line-too-long
+    # A regular expression used to recognize stack entries like:
+    #
+    #    #05  bf89a180  bf89a1e4  [stack]
+    #         bf89a1c8  a0c01c51  /data/app/com.google.android.apps.chrome-2/base.apk
+    #         bf89a080  00000000
+    #         ........  ........
+    # pylint: enable=line-too-long
+    self._re_stack_line = re.compile(
+        r'\s+(?P<frame_number>#[0-9]+)?\s*' +
+        r'(?P<stack_addr>' + hex_addr + r')\s+' +
+        r'(?P<stack_value>' + hex_addr + r')' +
+        r'(\s+(?P<location>[^ \t]+))?')
+
+    self._re_stack_abbrev = re.compile(r'\s+[.]+\s+[.]+')
+
+    self._memory_map = memory_map
+    self._apk_translator = apk_translator
+    self._in_section = False
+
+  def _ParseLine(self, line):
+    """Check a given input line for a relevant _re_stack_line match.
+
+    Args:
+      line: input tombstone line.
+    Returns:
+      A LineTuple instance on success, None on failure.
+    """
+    line = line.rstrip()
+    if not self._in_section:
+      self._in_section = line.startswith('stack:')
+      return None
+
+    m = self._re_stack_line.match(line)
+    if not m:
+      if not self._re_stack_abbrev.match(line):
+        self._in_section = False
+      return None
+
+    location = m.group('location')
+    if not location:
+      return None
+
+    if not location.endswith('.apk') and not location.endswith('.so'):
+      return None
+
+    addr = int(m.group('stack_value'), 16)
+    t = self._memory_map.FindSectionForAddress(addr)
+    if t is None:
+      return None
+
+    lib_path = t.path
+    lib_offset = t.offset + (addr - t.address)
+
+    if lib_path.endswith('.apk'):
+      lib_path, lib_offset = self._apk_translator.TranslatePath(
+          lib_path, lib_offset)
+
+    return self.LineTuple(addr, lib_path, lib_offset, m)
+
+  def FindLibraryOffsets(self, input_lines, in_section=False):
+    """Parse a tombstone's stack section and find all library offsets in it.
+
+    Args:
+      input_lines: List or iterables of intput tombstone lines.
+      in_section: Optional. If True, considers that the stack section has
+        already started.
+    Returns:
+      A dictionary mapping device library paths to sets of offsets within
+      then.
+    """
+    result = collections.defaultdict(set)
+    self._in_section = in_section
+    for line in input_lines:
+      t = self._ParseLine(line)
+      if t:
+        result[t.lib_path].add(t.lib_offset)
+    return result
+
+  def TranslateLine(self, line, symbol_resolver=None):
+    """Try to translate a line of the stack dump."""
+    t = self._ParseLine(line)
+    if not t:
+      return line
+
+    symbol_info = symbol_resolver.FindSymbolInfo(t.lib_path, t.lib_offset)
+    if not symbol_info:
+      return line
+
+    pos = t.match.start('location')
+    pos2 = t.match.end('location')
+    return '%s%s (%s)%s' % (line[:pos], t.lib_path, symbol_info, line[pos2:])
diff --git a/build/android/pylib/symbols/symbol_utils_unittest.py b/build/android/pylib/symbols/symbol_utils_unittest.py
new file mode 100644
index 0000000..d350c17
--- /dev/null
+++ b/build/android/pylib/symbols/symbol_utils_unittest.py
@@ -0,0 +1,944 @@
+# Copyright 2018 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+import collections
+import contextlib
+import logging
+import os
+import re
+import shutil
+import tempfile
+import unittest
+
+from pylib.symbols import apk_native_libs_unittest
+from pylib.symbols import mock_addr2line
+from pylib.symbols import symbol_utils
+
+_MOCK_ELF_DATA = apk_native_libs_unittest.MOCK_ELF_DATA
+
+_MOCK_A2L_PATH = os.path.join(os.path.dirname(mock_addr2line.__file__),
+                              'mock_addr2line')
+
+
+# pylint: disable=line-too-long
+
+# list of (start_offset, end_offset, size, libpath) tuples corresponding
+# to the content of base.apk. This was taken from an x86 ChromeModern.apk
+# component build.
+_TEST_APK_LIBS = [
+  (0x01331000, 0x013696bc, 0x000386bc, 'libaccessibility.cr.so'),
+  (0x0136a000, 0x013779c4, 0x0000d9c4, 'libanimation.cr.so'),
+  (0x01378000, 0x0137f7e8, 0x000077e8, 'libapdu.cr.so'),
+  (0x01380000, 0x0155ccc8, 0x001dccc8, 'libbase.cr.so'),
+  (0x0155d000, 0x015ab98c, 0x0004e98c, 'libbase_i18n.cr.so'),
+  (0x015ac000, 0x015dff4c, 0x00033f4c, 'libbindings.cr.so'),
+  (0x015e0000, 0x015f5a54, 0x00015a54, 'libbindings_base.cr.so'),
+  (0x015f6000, 0x0160d770, 0x00017770, 'libblink_android_mojo_bindings_shared.cr.so'),
+  (0x0160e000, 0x01731960, 0x00123960, 'libblink_common.cr.so'),
+  (0x01732000, 0x0174ce54, 0x0001ae54, 'libblink_controller.cr.so'),
+  (0x0174d000, 0x0318c528, 0x01a3f528, 'libblink_core.cr.so'),
+  (0x0318d000, 0x03191700, 0x00004700, 'libblink_core_mojo_bindings_shared.cr.so'),
+  (0x03192000, 0x03cd7918, 0x00b45918, 'libblink_modules.cr.so'),
+  (0x03cd8000, 0x03d137d0, 0x0003b7d0, 'libblink_mojo_bindings_shared.cr.so'),
+  (0x03d14000, 0x03d2670c, 0x0001270c, 'libblink_offscreen_canvas_mojo_bindings_shared.cr.so'),
+  (0x03d27000, 0x046c7054, 0x009a0054, 'libblink_platform.cr.so'),
+  (0x046c8000, 0x0473fbfc, 0x00077bfc, 'libbluetooth.cr.so'),
+  (0x04740000, 0x04878f40, 0x00138f40, 'libboringssl.cr.so'),
+  (0x04879000, 0x0498466c, 0x0010b66c, 'libc++_shared.so'),
+  (0x04985000, 0x0498d93c, 0x0000893c, 'libcaptive_portal.cr.so'),
+  (0x0498e000, 0x049947cc, 0x000067cc, 'libcapture_base.cr.so'),
+  (0x04995000, 0x04b39f18, 0x001a4f18, 'libcapture_lib.cr.so'),
+  (0x04b3a000, 0x04b488ec, 0x0000e8ec, 'libcbor.cr.so'),
+  (0x04b49000, 0x04e9ea5c, 0x00355a5c, 'libcc.cr.so'),
+  (0x04e9f000, 0x04ed6404, 0x00037404, 'libcc_animation.cr.so'),
+  (0x04ed7000, 0x04ef5ab4, 0x0001eab4, 'libcc_base.cr.so'),
+  (0x04ef6000, 0x04fd9364, 0x000e3364, 'libcc_blink.cr.so'),
+  (0x04fda000, 0x04fe2758, 0x00008758, 'libcc_debug.cr.so'),
+  (0x04fe3000, 0x0500ae0c, 0x00027e0c, 'libcc_ipc.cr.so'),
+  (0x0500b000, 0x05078f38, 0x0006df38, 'libcc_paint.cr.so'),
+  (0x05079000, 0x0507e734, 0x00005734, 'libcdm_manager.cr.so'),
+  (0x0507f000, 0x06f4d744, 0x01ece744, 'libchrome.cr.so'),
+  (0x06f54000, 0x06feb830, 0x00097830, 'libchromium_sqlite3.cr.so'),
+  (0x06fec000, 0x0706f554, 0x00083554, 'libclient.cr.so'),
+  (0x07070000, 0x0708da60, 0x0001da60, 'libcloud_policy_proto_generated_compile.cr.so'),
+  (0x0708e000, 0x07121f28, 0x00093f28, 'libcodec.cr.so'),
+  (0x07122000, 0x07134ab8, 0x00012ab8, 'libcolor_space.cr.so'),
+  (0x07135000, 0x07138614, 0x00003614, 'libcommon.cr.so'),
+  (0x07139000, 0x0717c938, 0x00043938, 'libcompositor.cr.so'),
+  (0x0717d000, 0x0923d78c, 0x020c078c, 'libcontent.cr.so'),
+  (0x0923e000, 0x092ae87c, 0x0007087c, 'libcontent_common_mojo_bindings_shared.cr.so'),
+  (0x092af000, 0x092be718, 0x0000f718, 'libcontent_public_common_mojo_bindings_shared.cr.so'),
+  (0x092bf000, 0x092d9a20, 0x0001aa20, 'libcrash_key.cr.so'),
+  (0x092da000, 0x092eda58, 0x00013a58, 'libcrcrypto.cr.so'),
+  (0x092ee000, 0x092f16e0, 0x000036e0, 'libdevice_base.cr.so'),
+  (0x092f2000, 0x092fe8d8, 0x0000c8d8, 'libdevice_event_log.cr.so'),
+  (0x092ff000, 0x093026a4, 0x000036a4, 'libdevice_features.cr.so'),
+  (0x09303000, 0x093f1220, 0x000ee220, 'libdevice_gamepad.cr.so'),
+  (0x093f2000, 0x09437f54, 0x00045f54, 'libdevice_vr_mojo_bindings.cr.so'),
+  (0x09438000, 0x0954c168, 0x00114168, 'libdevice_vr_mojo_bindings_blink.cr.so'),
+  (0x0954d000, 0x0955d720, 0x00010720, 'libdevice_vr_mojo_bindings_shared.cr.so'),
+  (0x0955e000, 0x0956b9c0, 0x0000d9c0, 'libdevices.cr.so'),
+  (0x0956c000, 0x0957cae8, 0x00010ae8, 'libdiscardable_memory_client.cr.so'),
+  (0x0957d000, 0x09588854, 0x0000b854, 'libdiscardable_memory_common.cr.so'),
+  (0x09589000, 0x0959cbb4, 0x00013bb4, 'libdiscardable_memory_service.cr.so'),
+  (0x0959d000, 0x095b6b90, 0x00019b90, 'libdisplay.cr.so'),
+  (0x095b7000, 0x095be930, 0x00007930, 'libdisplay_types.cr.so'),
+  (0x095bf000, 0x095c46c4, 0x000056c4, 'libdisplay_util.cr.so'),
+  (0x095c5000, 0x095f54a4, 0x000304a4, 'libdomain_reliability.cr.so'),
+  (0x095f6000, 0x0966fe08, 0x00079e08, 'libembedder.cr.so'),
+  (0x09670000, 0x096735f8, 0x000035f8, 'libembedder_switches.cr.so'),
+  (0x09674000, 0x096a3460, 0x0002f460, 'libevents.cr.so'),
+  (0x096a4000, 0x096b6d40, 0x00012d40, 'libevents_base.cr.so'),
+  (0x096b7000, 0x0981a778, 0x00163778, 'libffmpeg.cr.so'),
+  (0x0981b000, 0x09945c94, 0x0012ac94, 'libfido.cr.so'),
+  (0x09946000, 0x09a330dc, 0x000ed0dc, 'libfingerprint.cr.so'),
+  (0x09a34000, 0x09b53170, 0x0011f170, 'libfreetype_harfbuzz.cr.so'),
+  (0x09b54000, 0x09bc5c5c, 0x00071c5c, 'libgcm.cr.so'),
+  (0x09bc6000, 0x09cc8584, 0x00102584, 'libgeolocation.cr.so'),
+  (0x09cc9000, 0x09cdc8d4, 0x000138d4, 'libgeometry.cr.so'),
+  (0x09cdd000, 0x09cec8b4, 0x0000f8b4, 'libgeometry_skia.cr.so'),
+  (0x09ced000, 0x09d10e14, 0x00023e14, 'libgesture_detection.cr.so'),
+  (0x09d11000, 0x09d7595c, 0x0006495c, 'libgfx.cr.so'),
+  (0x09d76000, 0x09d7d7cc, 0x000077cc, 'libgfx_ipc.cr.so'),
+  (0x09d7e000, 0x09d82708, 0x00004708, 'libgfx_ipc_buffer_types.cr.so'),
+  (0x09d83000, 0x09d89748, 0x00006748, 'libgfx_ipc_color.cr.so'),
+  (0x09d8a000, 0x09d8f6f4, 0x000056f4, 'libgfx_ipc_geometry.cr.so'),
+  (0x09d90000, 0x09d94754, 0x00004754, 'libgfx_ipc_skia.cr.so'),
+  (0x09d95000, 0x09d9869c, 0x0000369c, 'libgfx_switches.cr.so'),
+  (0x09d99000, 0x09dba0ac, 0x000210ac, 'libgin.cr.so'),
+  (0x09dbb000, 0x09e0a8cc, 0x0004f8cc, 'libgl_in_process_context.cr.so'),
+  (0x09e0b000, 0x09e17a18, 0x0000ca18, 'libgl_init.cr.so'),
+  (0x09e18000, 0x09ee34e4, 0x000cb4e4, 'libgl_wrapper.cr.so'),
+  (0x09ee4000, 0x0a1a2e00, 0x002bee00, 'libgles2.cr.so'),
+  (0x0a1a3000, 0x0a24556c, 0x000a256c, 'libgles2_implementation.cr.so'),
+  (0x0a246000, 0x0a267038, 0x00021038, 'libgles2_utils.cr.so'),
+  (0x0a268000, 0x0a3288e4, 0x000c08e4, 'libgpu.cr.so'),
+  (0x0a329000, 0x0a3627ec, 0x000397ec, 'libgpu_ipc_service.cr.so'),
+  (0x0a363000, 0x0a388a18, 0x00025a18, 'libgpu_util.cr.so'),
+  (0x0a389000, 0x0a506d8c, 0x0017dd8c, 'libhost.cr.so'),
+  (0x0a507000, 0x0a6f0ec0, 0x001e9ec0, 'libicui18n.cr.so'),
+  (0x0a6f1000, 0x0a83b4c8, 0x0014a4c8, 'libicuuc.cr.so'),
+  (0x0a83c000, 0x0a8416e4, 0x000056e4, 'libinterfaces_shared.cr.so'),
+  (0x0a842000, 0x0a87e2a0, 0x0003c2a0, 'libipc.cr.so'),
+  (0x0a87f000, 0x0a88c98c, 0x0000d98c, 'libipc_mojom.cr.so'),
+  (0x0a88d000, 0x0a8926e4, 0x000056e4, 'libipc_mojom_shared.cr.so'),
+  (0x0a893000, 0x0a8a1e18, 0x0000ee18, 'libkeyed_service_content.cr.so'),
+  (0x0a8a2000, 0x0a8b4a30, 0x00012a30, 'libkeyed_service_core.cr.so'),
+  (0x0a8b5000, 0x0a930a80, 0x0007ba80, 'libleveldatabase.cr.so'),
+  (0x0a931000, 0x0a9b3908, 0x00082908, 'libmanager.cr.so'),
+  (0x0a9b4000, 0x0aea9bb4, 0x004f5bb4, 'libmedia.cr.so'),
+  (0x0aeaa000, 0x0b08cb88, 0x001e2b88, 'libmedia_blink.cr.so'),
+  (0x0b08d000, 0x0b0a4728, 0x00017728, 'libmedia_devices_mojo_bindings_shared.cr.so'),
+  (0x0b0a5000, 0x0b1943ec, 0x000ef3ec, 'libmedia_gpu.cr.so'),
+  (0x0b195000, 0x0b2d07d4, 0x0013b7d4, 'libmedia_mojo_services.cr.so'),
+  (0x0b2d1000, 0x0b2d4760, 0x00003760, 'libmessage_center.cr.so'),
+  (0x0b2d5000, 0x0b2e0938, 0x0000b938, 'libmessage_support.cr.so'),
+  (0x0b2e1000, 0x0b2f3ad0, 0x00012ad0, 'libmetrics_cpp.cr.so'),
+  (0x0b2f4000, 0x0b313bb8, 0x0001fbb8, 'libmidi.cr.so'),
+  (0x0b314000, 0x0b31b848, 0x00007848, 'libmojo_base_lib.cr.so'),
+  (0x0b31c000, 0x0b3329f8, 0x000169f8, 'libmojo_base_mojom.cr.so'),
+  (0x0b333000, 0x0b34b98c, 0x0001898c, 'libmojo_base_mojom_blink.cr.so'),
+  (0x0b34c000, 0x0b354700, 0x00008700, 'libmojo_base_mojom_shared.cr.so'),
+  (0x0b355000, 0x0b3608b0, 0x0000b8b0, 'libmojo_base_shared_typemap_traits.cr.so'),
+  (0x0b361000, 0x0b3ad454, 0x0004c454, 'libmojo_edk.cr.so'),
+  (0x0b3ae000, 0x0b3c4a20, 0x00016a20, 'libmojo_edk_ports.cr.so'),
+  (0x0b3c5000, 0x0b3d38a0, 0x0000e8a0, 'libmojo_mojom_bindings.cr.so'),
+  (0x0b3d4000, 0x0b3da6e8, 0x000066e8, 'libmojo_mojom_bindings_shared.cr.so'),
+  (0x0b3db000, 0x0b3e27f0, 0x000077f0, 'libmojo_public_system.cr.so'),
+  (0x0b3e3000, 0x0b3fa9fc, 0x000179fc, 'libmojo_public_system_cpp.cr.so'),
+  (0x0b3fb000, 0x0b407728, 0x0000c728, 'libmojom_core_shared.cr.so'),
+  (0x0b408000, 0x0b421744, 0x00019744, 'libmojom_platform_shared.cr.so'),
+  (0x0b422000, 0x0b43451c, 0x0001251c, 'libnative_theme.cr.so'),
+  (0x0b435000, 0x0baaa1bc, 0x006751bc, 'libnet.cr.so'),
+  (0x0baab000, 0x0bac3c08, 0x00018c08, 'libnet_with_v8.cr.so'),
+  (0x0bac4000, 0x0bb74670, 0x000b0670, 'libnetwork_cpp.cr.so'),
+  (0x0bb75000, 0x0bbaee8c, 0x00039e8c, 'libnetwork_cpp_base.cr.so'),
+  (0x0bbaf000, 0x0bd21844, 0x00172844, 'libnetwork_service.cr.so'),
+  (0x0bd22000, 0x0bd256e4, 0x000036e4, 'libnetwork_session_configurator.cr.so'),
+  (0x0bd26000, 0x0bd33734, 0x0000d734, 'libonc.cr.so'),
+  (0x0bd34000, 0x0bd9ce18, 0x00068e18, 'libperfetto.cr.so'),
+  (0x0bd9d000, 0x0bda4854, 0x00007854, 'libplatform.cr.so'),
+  (0x0bda5000, 0x0bec5ce4, 0x00120ce4, 'libpolicy_component.cr.so'),
+  (0x0bec6000, 0x0bf5ab58, 0x00094b58, 'libpolicy_proto.cr.so'),
+  (0x0bf5b000, 0x0bf86fbc, 0x0002bfbc, 'libprefs.cr.so'),
+  (0x0bf87000, 0x0bfa5d74, 0x0001ed74, 'libprinting.cr.so'),
+  (0x0bfa6000, 0x0bfe0e80, 0x0003ae80, 'libprotobuf_lite.cr.so'),
+  (0x0bfe1000, 0x0bff0a18, 0x0000fa18, 'libproxy_config.cr.so'),
+  (0x0bff1000, 0x0c0f6654, 0x00105654, 'libpublic.cr.so'),
+  (0x0c0f7000, 0x0c0fa6a4, 0x000036a4, 'librange.cr.so'),
+  (0x0c0fb000, 0x0c118058, 0x0001d058, 'libraster.cr.so'),
+  (0x0c119000, 0x0c133d00, 0x0001ad00, 'libresource_coordinator_cpp.cr.so'),
+  (0x0c134000, 0x0c1396a0, 0x000056a0, 'libresource_coordinator_cpp_base.cr.so'),
+  (0x0c13a000, 0x0c1973b8, 0x0005d3b8, 'libresource_coordinator_public_mojom.cr.so'),
+  (0x0c198000, 0x0c2033e8, 0x0006b3e8, 'libresource_coordinator_public_mojom_blink.cr.so'),
+  (0x0c204000, 0x0c219744, 0x00015744, 'libresource_coordinator_public_mojom_shared.cr.so'),
+  (0x0c21a000, 0x0c21e700, 0x00004700, 'libsandbox.cr.so'),
+  (0x0c21f000, 0x0c22f96c, 0x0001096c, 'libsandbox_services.cr.so'),
+  (0x0c230000, 0x0c249d58, 0x00019d58, 'libseccomp_bpf.cr.so'),
+  (0x0c24a000, 0x0c24e714, 0x00004714, 'libseccomp_starter_android.cr.so'),
+  (0x0c24f000, 0x0c4ae9f0, 0x0025f9f0, 'libservice.cr.so'),
+  (0x0c4af000, 0x0c4c3ae4, 0x00014ae4, 'libservice_manager_cpp.cr.so'),
+  (0x0c4c4000, 0x0c4cb708, 0x00007708, 'libservice_manager_cpp_types.cr.so'),
+  (0x0c4cc000, 0x0c4fbe30, 0x0002fe30, 'libservice_manager_mojom.cr.so'),
+  (0x0c4fc000, 0x0c532e78, 0x00036e78, 'libservice_manager_mojom_blink.cr.so'),
+  (0x0c533000, 0x0c53669c, 0x0000369c, 'libservice_manager_mojom_constants.cr.so'),
+  (0x0c537000, 0x0c53e85c, 0x0000785c, 'libservice_manager_mojom_constants_blink.cr.so'),
+  (0x0c53f000, 0x0c542668, 0x00003668, 'libservice_manager_mojom_constants_shared.cr.so'),
+  (0x0c543000, 0x0c54d700, 0x0000a700, 'libservice_manager_mojom_shared.cr.so'),
+  (0x0c54e000, 0x0c8fc6ec, 0x003ae6ec, 'libsessions.cr.so'),
+  (0x0c8fd000, 0x0c90a924, 0x0000d924, 'libshared_memory_support.cr.so'),
+  (0x0c90b000, 0x0c9148ec, 0x000098ec, 'libshell_dialogs.cr.so'),
+  (0x0c915000, 0x0cf8de70, 0x00678e70, 'libskia.cr.so'),
+  (0x0cf8e000, 0x0cf978bc, 0x000098bc, 'libsnapshot.cr.so'),
+  (0x0cf98000, 0x0cfb7d9c, 0x0001fd9c, 'libsql.cr.so'),
+  (0x0cfb8000, 0x0cfbe744, 0x00006744, 'libstartup_tracing.cr.so'),
+  (0x0cfbf000, 0x0d19b4e4, 0x001dc4e4, 'libstorage_browser.cr.so'),
+  (0x0d19c000, 0x0d2a773c, 0x0010b73c, 'libstorage_common.cr.so'),
+  (0x0d2a8000, 0x0d2ac6fc, 0x000046fc, 'libsurface.cr.so'),
+  (0x0d2ad000, 0x0d2baa98, 0x0000da98, 'libtracing.cr.so'),
+  (0x0d2bb000, 0x0d2f36b0, 0x000386b0, 'libtracing_cpp.cr.so'),
+  (0x0d2f4000, 0x0d326e70, 0x00032e70, 'libtracing_mojom.cr.so'),
+  (0x0d327000, 0x0d33270c, 0x0000b70c, 'libtracing_mojom_shared.cr.so'),
+  (0x0d333000, 0x0d46d804, 0x0013a804, 'libui_android.cr.so'),
+  (0x0d46e000, 0x0d4cb3f8, 0x0005d3f8, 'libui_base.cr.so'),
+  (0x0d4cc000, 0x0d4dbc40, 0x0000fc40, 'libui_base_ime.cr.so'),
+  (0x0d4dc000, 0x0d4e58d4, 0x000098d4, 'libui_data_pack.cr.so'),
+  (0x0d4e6000, 0x0d51d1e0, 0x000371e0, 'libui_devtools.cr.so'),
+  (0x0d51e000, 0x0d52b984, 0x0000d984, 'libui_message_center_cpp.cr.so'),
+  (0x0d52c000, 0x0d539a48, 0x0000da48, 'libui_touch_selection.cr.so'),
+  (0x0d53a000, 0x0d55bc60, 0x00021c60, 'liburl.cr.so'),
+  (0x0d55c000, 0x0d55f6b4, 0x000036b4, 'liburl_ipc.cr.so'),
+  (0x0d560000, 0x0d5af110, 0x0004f110, 'liburl_matcher.cr.so'),
+  (0x0d5b0000, 0x0d5e2fac, 0x00032fac, 'libuser_manager.cr.so'),
+  (0x0d5e3000, 0x0d5e66e4, 0x000036e4, 'libuser_prefs.cr.so'),
+  (0x0d5e7000, 0x0e3e1cc8, 0x00dfacc8, 'libv8.cr.so'),
+  (0x0e3e2000, 0x0e400ae0, 0x0001eae0, 'libv8_libbase.cr.so'),
+  (0x0e401000, 0x0e4d91d4, 0x000d81d4, 'libviz_common.cr.so'),
+  (0x0e4da000, 0x0e4df7e4, 0x000057e4, 'libviz_resource_format.cr.so'),
+  (0x0e4e0000, 0x0e5b7120, 0x000d7120, 'libweb_dialogs.cr.so'),
+  (0x0e5b8000, 0x0e5c7a18, 0x0000fa18, 'libwebdata_common.cr.so'),
+  (0x0e5c8000, 0x0e61bfe4, 0x00053fe4, 'libwtf.cr.so'),
+]
+
+
+# A small memory map fragment extracted from a tombstone for a process that
+# had loaded the APK corresponding to _TEST_APK_LIBS above.
+_TEST_MEMORY_MAP = r'''memory map:
+12c00000-12ccafff rw-         0     cb000  /dev/ashmem/dalvik-main space (deleted)
+12ccb000-130cafff rw-     cb000    400000  /dev/ashmem/dalvik-main space (deleted)
+130cb000-32bfffff ---    4cb000  1fb35000  /dev/ashmem/dalvik-main space (deleted)
+32c00000-32c00fff rw-         0      1000  /dev/ashmem/dalvik-main space 1 (deleted)
+32c01000-52bfffff ---      1000  1ffff000  /dev/ashmem/dalvik-main space 1 (deleted)
+6f3b8000-6fd90fff rw-         0    9d9000  /data/dalvik-cache/x86/system@framework@boot.art
+6fd91000-71c42fff r--         0   1eb2000  /data/dalvik-cache/x86/system@framework@boot.oat
+71c43000-7393efff r-x   1eb2000   1cfc000  /data/dalvik-cache/x86/system@framework@boot.oat (load base 0x71c43000)
+7393f000-7393ffff rw-   3bae000      1000  /data/dalvik-cache/x86/system@framework@boot.oat
+73940000-73a1bfff rw-         0     dc000  /dev/ashmem/dalvik-zygote space (deleted)
+73a1c000-73a1cfff rw-         0      1000  /dev/ashmem/dalvik-non moving space (deleted)
+73a1d000-73a2dfff rw-      1000     11000  /dev/ashmem/dalvik-non moving space (deleted)
+73a2e000-77540fff ---     12000   3b13000  /dev/ashmem/dalvik-non moving space (deleted)
+77541000-7793ffff rw-   3b25000    3ff000  /dev/ashmem/dalvik-non moving space (deleted)
+923aa000-92538fff r--    8a9000    18f000  /data/app/com.example.app-2/base.apk
+92539000-9255bfff r--         0     23000  /data/data/com.example.app/app_data/paks/es.pak@162db1c6689
+9255c000-92593fff r--    213000     38000  /data/app/com.example.app-2/base.apk
+92594000-925c0fff r--    87d000     2d000  /data/app/com.example.app-2/base.apk
+925c1000-927d3fff r--    a37000    213000  /data/app/com.example.app-2/base.apk
+927d4000-92e07fff r--    24a000    634000  /data/app/com.example.app-2/base.apk
+92e08000-92e37fff r--   a931000     30000  /data/app/com.example.app-2/base.apk
+92e38000-92e86fff r-x   a961000     4f000  /data/app/com.example.app-2/base.apk
+92e87000-92e8afff rw-   a9b0000      4000  /data/app/com.example.app-2/base.apk
+92e8b000-92e8bfff rw-         0      1000
+92e8c000-92e9dfff r--   d5b0000     12000  /data/app/com.example.app-2/base.apk
+92e9e000-92ebcfff r-x   d5c2000     1f000  /data/app/com.example.app-2/base.apk
+92ebd000-92ebefff rw-   d5e1000      2000  /data/app/com.example.app-2/base.apk
+92ebf000-92ebffff rw-         0      1000
+'''
+
+# list of (address, size, path, offset)  tuples that must appear in
+# _TEST_MEMORY_MAP. Not all sections need to be listed.
+_TEST_MEMORY_MAP_SECTIONS = [
+  (0x923aa000, 0x18f000, '/data/app/com.example.app-2/base.apk', 0x8a9000),
+  (0x9255c000, 0x038000, '/data/app/com.example.app-2/base.apk', 0x213000),
+  (0x92594000, 0x02d000, '/data/app/com.example.app-2/base.apk', 0x87d000),
+  (0x925c1000, 0x213000, '/data/app/com.example.app-2/base.apk', 0xa37000),
+]
+
+_EXPECTED_TEST_MEMORY_MAP = r'''memory map:
+12c00000-12ccafff rw-         0     cb000  /dev/ashmem/dalvik-main space (deleted)
+12ccb000-130cafff rw-     cb000    400000  /dev/ashmem/dalvik-main space (deleted)
+130cb000-32bfffff ---    4cb000  1fb35000  /dev/ashmem/dalvik-main space (deleted)
+32c00000-32c00fff rw-         0      1000  /dev/ashmem/dalvik-main space 1 (deleted)
+32c01000-52bfffff ---      1000  1ffff000  /dev/ashmem/dalvik-main space 1 (deleted)
+6f3b8000-6fd90fff rw-         0    9d9000  /data/dalvik-cache/x86/system@framework@boot.art
+6fd91000-71c42fff r--         0   1eb2000  /data/dalvik-cache/x86/system@framework@boot.oat
+71c43000-7393efff r-x   1eb2000   1cfc000  /data/dalvik-cache/x86/system@framework@boot.oat (load base 0x71c43000)
+7393f000-7393ffff rw-   3bae000      1000  /data/dalvik-cache/x86/system@framework@boot.oat
+73940000-73a1bfff rw-         0     dc000  /dev/ashmem/dalvik-zygote space (deleted)
+73a1c000-73a1cfff rw-         0      1000  /dev/ashmem/dalvik-non moving space (deleted)
+73a1d000-73a2dfff rw-      1000     11000  /dev/ashmem/dalvik-non moving space (deleted)
+73a2e000-77540fff ---     12000   3b13000  /dev/ashmem/dalvik-non moving space (deleted)
+77541000-7793ffff rw-   3b25000    3ff000  /dev/ashmem/dalvik-non moving space (deleted)
+923aa000-92538fff r--    8a9000    18f000  /data/app/com.example.app-2/base.apk
+92539000-9255bfff r--         0     23000  /data/data/com.example.app/app_data/paks/es.pak@162db1c6689
+9255c000-92593fff r--    213000     38000  /data/app/com.example.app-2/base.apk
+92594000-925c0fff r--    87d000     2d000  /data/app/com.example.app-2/base.apk
+925c1000-927d3fff r--    a37000    213000  /data/app/com.example.app-2/base.apk
+927d4000-92e07fff r--    24a000    634000  /data/app/com.example.app-2/base.apk
+92e08000-92e37fff r--   a931000     30000  /data/app/com.example.app-2/base.apk!lib/libmanager.cr.so (offset 0x0)
+92e38000-92e86fff r-x   a961000     4f000  /data/app/com.example.app-2/base.apk!lib/libmanager.cr.so (offset 0x30000)
+92e87000-92e8afff rw-   a9b0000      4000  /data/app/com.example.app-2/base.apk!lib/libmanager.cr.so (offset 0x7f000)
+92e8b000-92e8bfff rw-         0      1000
+92e8c000-92e9dfff r--   d5b0000     12000  /data/app/com.example.app-2/base.apk!lib/libuser_manager.cr.so (offset 0x0)
+92e9e000-92ebcfff r-x   d5c2000     1f000  /data/app/com.example.app-2/base.apk!lib/libuser_manager.cr.so (offset 0x12000)
+92ebd000-92ebefff rw-   d5e1000      2000  /data/app/com.example.app-2/base.apk!lib/libuser_manager.cr.so (offset 0x31000)
+92ebf000-92ebffff rw-         0      1000
+'''
+
+# Example stack section, taken from the same tombstone that _TEST_MEMORY_MAP
+# was extracted from.
+_TEST_STACK = r'''stack:
+        bf89a070  b7439468  /system/lib/libc.so
+        bf89a074  bf89a1e4  [stack]
+        bf89a078  932d4000  /data/app/com.example.app-2/base.apk
+        bf89a07c  b73bfbc9  /system/lib/libc.so (pthread_mutex_lock+65)
+        bf89a080  00000000
+        bf89a084  4000671c  /dev/ashmem/dalvik-main space 1 (deleted)
+        bf89a088  932d1d86  /data/app/com.example.app-2/base.apk
+        bf89a08c  b743671c  /system/lib/libc.so
+        bf89a090  b77f8c00  /system/bin/linker
+        bf89a094  b743cc90
+        bf89a098  932d1d4a  /data/app/com.example.app-2/base.apk
+        bf89a09c  b73bf271  /system/lib/libc.so (__pthread_internal_find(long)+65)
+        bf89a0a0  b743cc90
+        bf89a0a4  bf89a0b0  [stack]
+        bf89a0a8  bf89a0b8  [stack]
+        bf89a0ac  00000008
+        ........  ........
+  #00  bf89a0b0  00000006
+        bf89a0b4  00000002
+        bf89a0b8  b743671c  /system/lib/libc.so
+        bf89a0bc  b73bf5d9  /system/lib/libc.so (pthread_kill+71)
+  #01  bf89a0c0  00006937
+        bf89a0c4  00006937
+        bf89a0c8  00000006
+        bf89a0cc  b77fd3a9  /system/bin/app_process32 (sigprocmask+141)
+        bf89a0d0  00000002
+        bf89a0d4  bf89a0ec  [stack]
+        bf89a0d8  00000000
+        bf89a0dc  b743671c  /system/lib/libc.so
+        bf89a0e0  bf89a12c  [stack]
+        bf89a0e4  bf89a1e4  [stack]
+        bf89a0e8  932d1d4a  /data/app/com.example.app-2/base.apk
+        bf89a0ec  b7365206  /system/lib/libc.so (raise+37)
+  #02  bf89a0f0  b77f8c00  /system/bin/linker
+        bf89a0f4  00000006
+        bf89a0f8  b7439468  /system/lib/libc.so
+        bf89a0fc  b743671c  /system/lib/libc.so
+        bf89a100  bf89a12c  [stack]
+        bf89a104  b743671c  /system/lib/libc.so
+        bf89a108  bf89a12c  [stack]
+        bf89a10c  b735e9e5  /system/lib/libc.so (abort+81)
+  #03  bf89a110  00000006
+        bf89a114  bf89a12c  [stack]
+        bf89a118  00000000
+        bf89a11c  b55a3d3b  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::DefaultLogHandler(google::protobuf::LogLevel, char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&)+99)
+        bf89a120  b7439468  /system/lib/libc.so
+        bf89a124  b55ba38d  /system/lib/libprotobuf-cpp-lite.so
+        bf89a128  b55ba408  /system/lib/libprotobuf-cpp-lite.so
+        bf89a12c  ffffffdf
+        bf89a130  0000003d
+        bf89a134  adfedf00  [anon:libc_malloc]
+        bf89a138  bf89a158  [stack]
+  #04  bf89a13c  a0cee7f0  /data/app/com.example.app-2/base.apk
+        bf89a140  b55c1cb0  /system/lib/libprotobuf-cpp-lite.so
+        bf89a144  bf89a1e4  [stack]
+'''
+
+# Expected value of _TEST_STACK after translation of addresses in the APK
+# into offsets into libraries.
+_EXPECTED_STACK = r'''stack:
+        bf89a070  b7439468  /system/lib/libc.so
+        bf89a074  bf89a1e4  [stack]
+        bf89a078  932d4000  /data/app/com.example.app-2/base.apk
+        bf89a07c  b73bfbc9  /system/lib/libc.so (pthread_mutex_lock+65)
+        bf89a080  00000000
+        bf89a084  4000671c  /dev/ashmem/dalvik-main space 1 (deleted)
+        bf89a088  932d1d86  /data/app/com.example.app-2/base.apk
+        bf89a08c  b743671c  /system/lib/libc.so
+        bf89a090  b77f8c00  /system/bin/linker
+        bf89a094  b743cc90
+        bf89a098  932d1d4a  /data/app/com.example.app-2/base.apk
+        bf89a09c  b73bf271  /system/lib/libc.so (__pthread_internal_find(long)+65)
+        bf89a0a0  b743cc90
+        bf89a0a4  bf89a0b0  [stack]
+        bf89a0a8  bf89a0b8  [stack]
+        bf89a0ac  00000008
+        ........  ........
+  #00  bf89a0b0  00000006
+        bf89a0b4  00000002
+        bf89a0b8  b743671c  /system/lib/libc.so
+        bf89a0bc  b73bf5d9  /system/lib/libc.so (pthread_kill+71)
+  #01  bf89a0c0  00006937
+        bf89a0c4  00006937
+        bf89a0c8  00000006
+        bf89a0cc  b77fd3a9  /system/bin/app_process32 (sigprocmask+141)
+        bf89a0d0  00000002
+        bf89a0d4  bf89a0ec  [stack]
+        bf89a0d8  00000000
+        bf89a0dc  b743671c  /system/lib/libc.so
+        bf89a0e0  bf89a12c  [stack]
+        bf89a0e4  bf89a1e4  [stack]
+        bf89a0e8  932d1d4a  /data/app/com.example.app-2/base.apk
+        bf89a0ec  b7365206  /system/lib/libc.so (raise+37)
+  #02  bf89a0f0  b77f8c00  /system/bin/linker
+        bf89a0f4  00000006
+        bf89a0f8  b7439468  /system/lib/libc.so
+        bf89a0fc  b743671c  /system/lib/libc.so
+        bf89a100  bf89a12c  [stack]
+        bf89a104  b743671c  /system/lib/libc.so
+        bf89a108  bf89a12c  [stack]
+        bf89a10c  b735e9e5  /system/lib/libc.so (abort+81)
+  #03  bf89a110  00000006
+        bf89a114  bf89a12c  [stack]
+        bf89a118  00000000
+        bf89a11c  b55a3d3b  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::DefaultLogHandler(google::protobuf::LogLevel, char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&)+99)
+        bf89a120  b7439468  /system/lib/libc.so
+        bf89a124  b55ba38d  /system/lib/libprotobuf-cpp-lite.so
+        bf89a128  b55ba408  /system/lib/libprotobuf-cpp-lite.so
+        bf89a12c  ffffffdf
+        bf89a130  0000003d
+        bf89a134  adfedf00  [anon:libc_malloc]
+        bf89a138  bf89a158  [stack]
+  #04  bf89a13c  a0cee7f0  /data/app/com.example.app-2/base.apk
+        bf89a140  b55c1cb0  /system/lib/libprotobuf-cpp-lite.so
+        bf89a144  bf89a1e4  [stack]
+'''
+
+_TEST_BACKTRACE = r'''backtrace:
+    #00 pc 00084126  /system/lib/libc.so (tgkill+22)
+    #01 pc 000815d8  /system/lib/libc.so (pthread_kill+70)
+    #02 pc 00027205  /system/lib/libc.so (raise+36)
+    #03 pc 000209e4  /system/lib/libc.so (abort+80)
+    #04 pc 0000cf73  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::LogMessage::Finish()+117)
+    #05 pc 0000cf8e  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::LogFinisher::operator=(google::protobuf::internal::LogMessage&)+26)
+    #06 pc 0000d27f  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::VerifyVersion(int, int, char const*)+574)
+    #07 pc 007cd236  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #08 pc 000111a9  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0xbfc2000)
+    #09 pc 00013228  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0xbfc2000)
+    #10 pc 000131de  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0xbfc2000)
+    #11 pc 007cd2d8  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #12 pc 007cd956  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #13 pc 007c2d4a  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #14 pc 009fc9f1  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #15 pc 009fc8ea  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #16 pc 00561c63  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #17 pc 0106fbdb  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #18 pc 004d7371  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #19 pc 004d8159  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #20 pc 004d7b96  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #21 pc 004da4b6  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #22 pc 005ab66c  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #23 pc 005afca2  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #24 pc 0000cae8  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x598d000)
+    #25 pc 00ce864f  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #26 pc 00ce8dfa  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #27 pc 00ce74c6  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #28 pc 00004616  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x961e000)
+    #29 pc 00ce8215  /data/app/com.google.android.apps.chrome-2/base.apk (offset 0x7daa000)
+    #30 pc 0013d8c7  /system/lib/libart.so (art_quick_generic_jni_trampoline+71)
+    #31 pc 00137c52  /system/lib/libart.so (art_quick_invoke_static_stub+418)
+    #32 pc 00143651  /system/lib/libart.so (art::ArtMethod::Invoke(art::Thread*, unsigned int*, unsigned int, art::JValue*, char const*)+353)
+    #33 pc 005e06ae  /system/lib/libart.so (artInterpreterToCompiledCodeBridge+190)
+    #34 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #35 pc 0032cfc0  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)0, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+160)
+    #36 pc 000fc703  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29891)
+    #37 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #38 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #39 pc 0032cfc0  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)0, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+160)
+    #40 pc 000fc703  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29891)
+    #41 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #42 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #43 pc 0032ebf9  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)2, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+297)
+    #44 pc 000fc955  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+30485)
+    #45 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #46 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #47 pc 0033090c  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)4, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+636)
+    #48 pc 000fc67f  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29759)
+    #49 pc 00300700  /system/lib/libart.so (art::interpreter::EnterInterpreterFromEntryPoint(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame*)+128)
+    #50 pc 00667c73  /system/lib/libart.so (artQuickToInterpreterBridge+808)
+    #51 pc 0013d98d  /system/lib/libart.so (art_quick_to_interpreter_bridge+77)
+    #52 pc 7264bc5b  /data/dalvik-cache/x86/system@framework@boot.oat (offset 0x1eb2000)
+'''
+
+_EXPECTED_BACKTRACE = r'''backtrace:
+    #00 pc 00084126  /system/lib/libc.so (tgkill+22)
+    #01 pc 000815d8  /system/lib/libc.so (pthread_kill+70)
+    #02 pc 00027205  /system/lib/libc.so (raise+36)
+    #03 pc 000209e4  /system/lib/libc.so (abort+80)
+    #04 pc 0000cf73  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::LogMessage::Finish()+117)
+    #05 pc 0000cf8e  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::LogFinisher::operator=(google::protobuf::internal::LogMessage&)+26)
+    #06 pc 0000d27f  /system/lib/libprotobuf-cpp-lite.so (google::protobuf::internal::VerifyVersion(int, int, char const*)+574)
+    #07 pc 007cd236  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #08 pc 000111a9  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libprotobuf_lite.cr.so (offset 0x1c000)
+    #09 pc 00013228  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libprotobuf_lite.cr.so (offset 0x1c000)
+    #10 pc 000131de  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libprotobuf_lite.cr.so (offset 0x1c000)
+    #11 pc 007cd2d8  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #12 pc 007cd956  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #13 pc 007c2d4a  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #14 pc 009fc9f1  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #15 pc 009fc8ea  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #16 pc 00561c63  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #17 pc 0106fbdb  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #18 pc 004d7371  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #19 pc 004d8159  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #20 pc 004d7b96  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #21 pc 004da4b6  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #22 pc 005ab66c  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #23 pc 005afca2  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #24 pc 0000cae8  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so (offset 0x90e000)
+    #25 pc 00ce864f  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #26 pc 00ce8dfa  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #27 pc 00ce74c6  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #28 pc 00004616  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libembedder.cr.so (offset 0x28000)
+    #29 pc 00ce8215  /data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so (offset 0xc2d000)
+    #30 pc 0013d8c7  /system/lib/libart.so (art_quick_generic_jni_trampoline+71)
+    #31 pc 00137c52  /system/lib/libart.so (art_quick_invoke_static_stub+418)
+    #32 pc 00143651  /system/lib/libart.so (art::ArtMethod::Invoke(art::Thread*, unsigned int*, unsigned int, art::JValue*, char const*)+353)
+    #33 pc 005e06ae  /system/lib/libart.so (artInterpreterToCompiledCodeBridge+190)
+    #34 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #35 pc 0032cfc0  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)0, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+160)
+    #36 pc 000fc703  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29891)
+    #37 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #38 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #39 pc 0032cfc0  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)0, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+160)
+    #40 pc 000fc703  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29891)
+    #41 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #42 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #43 pc 0032ebf9  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)2, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+297)
+    #44 pc 000fc955  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+30485)
+    #45 pc 00300af7  /system/lib/libart.so (artInterpreterToInterpreterBridge+188)
+    #46 pc 00328b5d  /system/lib/libart.so (bool art::interpreter::DoCall<false, false>(art::ArtMethod*, art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+445)
+    #47 pc 0033090c  /system/lib/libart.so (bool art::interpreter::DoInvoke<(art::InvokeType)4, false, false>(art::Thread*, art::ShadowFrame&, art::Instruction const*, unsigned short, art::JValue*)+636)
+    #48 pc 000fc67f  /system/lib/libart.so (art::JValue art::interpreter::ExecuteGotoImpl<false, false>(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)+29759)
+    #49 pc 00300700  /system/lib/libart.so (art::interpreter::EnterInterpreterFromEntryPoint(art::Thread*, art::DexFile::CodeItem const*, art::ShadowFrame*)+128)
+    #50 pc 00667c73  /system/lib/libart.so (artQuickToInterpreterBridge+808)
+    #51 pc 0013d98d  /system/lib/libart.so (art_quick_to_interpreter_bridge+77)
+    #52 pc 7264bc5b  /data/dalvik-cache/x86/system@framework@boot.oat (offset 0x1eb2000)
+'''
+
+_EXPECTED_BACKTRACE_OFFSETS_MAP = {
+  '/data/app/com.google.android.apps.chrome-2/base.apk!lib/libprotobuf_lite.cr.so':
+      set([
+          0x1c000 + 0x111a9,
+          0x1c000 + 0x13228,
+          0x1c000 + 0x131de,
+      ]),
+
+  '/data/app/com.google.android.apps.chrome-2/base.apk!lib/libchrome.cr.so':
+      set([
+          0x90e000 + 0x7cd236,
+          0x90e000 + 0x7cd2d8,
+          0x90e000 + 0x7cd956,
+          0x90e000 + 0x7c2d4a,
+          0x90e000 + 0x9fc9f1,
+          0x90e000 + 0x9fc8ea,
+          0x90e000 + 0x561c63,
+          0x90e000 + 0x106fbdb,
+          0x90e000 + 0x4d7371,
+          0x90e000 + 0x4d8159,
+          0x90e000 + 0x4d7b96,
+          0x90e000 + 0x4da4b6,
+          0x90e000 + 0xcae8,
+      ]),
+  '/data/app/com.google.android.apps.chrome-2/base.apk!lib/libcontent.cr.so':
+      set([
+          0xc2d000 + 0x5ab66c,
+          0xc2d000 + 0x5afca2,
+          0xc2d000 + 0xce864f,
+          0xc2d000 + 0xce8dfa,
+          0xc2d000 + 0xce74c6,
+          0xc2d000 + 0xce8215,
+      ]),
+  '/data/app/com.google.android.apps.chrome-2/base.apk!lib/libembedder.cr.so':
+      set([
+          0x28000 + 0x4616,
+      ])
+}
+
+# pylint: enable=line-too-long
+
+_ONE_MB = 1024 * 1024
+_TEST_SYMBOL_DATA = {
+  # Regular symbols
+  0: 'mock_sym_for_addr_0 [mock_src/libmock1.so.c:0]',
+  0x1000: 'mock_sym_for_addr_4096 [mock_src/libmock1.so.c:4096]',
+
+  # Symbols without source file path.
+  _ONE_MB: 'mock_sym_for_addr_1048576 [??:0]',
+  _ONE_MB + 0x8234: 'mock_sym_for_addr_1081908 [??:0]',
+
+  # Unknown symbol.
+  2 * _ONE_MB: '?? [??:0]',
+
+  # Inlined symbol.
+  3 * _ONE_MB:
+    'mock_sym_for_addr_3145728_inner [mock_src/libmock1.so.c:3145728]',
+}
+
+@contextlib.contextmanager
+def _TempDir():
+  dirname = tempfile.mkdtemp()
+  try:
+    yield dirname
+  finally:
+    shutil.rmtree(dirname)
+
+
+def _TouchFile(path):
+  # Create parent directories.
+  try:
+    os.makedirs(os.path.dirname(path))
+  except OSError:
+    pass
+  with open(path, 'a'):
+    os.utime(path, None)
+
+class MockApkTranslator(object):
+  """A mock ApkLibraryPathTranslator object used for testing."""
+
+  # Regex that matches the content of APK native library map files generated
+  # with apk_lib_dump.py.
+  _RE_MAP_FILE = re.compile(
+      r'0x(?P<file_start>[0-9a-f]+)\s+' +
+      r'0x(?P<file_end>[0-9a-f]+)\s+' +
+      r'0x(?P<file_size>[0-9a-f]+)\s+' +
+      r'0x(?P<lib_path>[0-9a-f]+)\s+')
+
+  def __init__(self, test_apk_libs=None):
+    """Initialize instance.
+
+    Args:
+      test_apk_libs: Optional list of (file_start, file_end, size, lib_path)
+        tuples, like _TEST_APK_LIBS for example. This will be used to
+        implement TranslatePath().
+    """
+    self._apk_libs = []
+    if test_apk_libs:
+      self._AddLibEntries(test_apk_libs)
+
+  def _AddLibEntries(self, entries):
+    self._apk_libs = sorted(self._apk_libs + entries,
+                            lambda x, y: cmp(x[0], y[0]))
+
+  def ReadMapFile(self, file_path):
+    """Read an .apk.native-libs file that was produced with apk_lib_dump.py.
+
+    Args:
+      file_path: input path to .apk.native-libs file. Its format is
+        essentially: 0x<start>  0x<end> 0x<size> <library-path>
+    """
+    new_libs = []
+    with open(file_path) as f:
+      for line in f.readlines():
+        m = MockApkTranslator._RE_MAP_FILE.match(line)
+        if m:
+          file_start = int(m.group('file_start'), 16)
+          file_end = int(m.group('file_end'), 16)
+          file_size = int(m.group('file_size'), 16)
+          lib_path = m.group('lib_path')
+          # Sanity check
+          if file_start + file_size != file_end:
+            logging.warning('%s: Inconsistent (start, end, size) values '
+                            '(0x%x, 0x%x, 0x%x)',
+                            file_path, file_start, file_end, file_size)
+          else:
+            new_libs.append((file_start, file_end, file_size, lib_path))
+
+    self._AddLibEntries(new_libs)
+
+  def TranslatePath(self, lib_path, lib_offset):
+    """Translate an APK file path + offset into a library path + offset."""
+    min_pos = 0
+    max_pos = len(self._apk_libs)
+    while min_pos < max_pos:
+      mid_pos = (min_pos + max_pos) / 2
+      mid_entry = self._apk_libs[mid_pos]
+      mid_offset = mid_entry[0]
+      mid_size = mid_entry[2]
+      if lib_offset < mid_offset:
+        max_pos = mid_pos
+      elif lib_offset >= mid_offset + mid_size:
+        min_pos = mid_pos + 1
+      else:
+        # Found it
+        new_path = '%s!lib/%s' % (lib_path, mid_entry[3])
+        new_offset = lib_offset - mid_offset
+        return (new_path, new_offset)
+
+    return lib_path, lib_offset
+
+
+class HostLibraryFinderTest(unittest.TestCase):
+
+  def testEmpty(self):
+    finder = symbol_utils.HostLibraryFinder()
+    self.assertIsNone(finder.Find('/data/data/com.example.app-1/lib/libfoo.so'))
+    self.assertIsNone(
+        finder.Find('/data/data/com.example.app-1/base.apk!lib/libfoo.so'))
+
+
+  def testSimpleDirectory(self):
+    finder = symbol_utils.HostLibraryFinder()
+    with _TempDir() as tmp_dir:
+      host_libfoo_path = os.path.join(tmp_dir, 'libfoo.so')
+      host_libbar_path = os.path.join(tmp_dir, 'libbar.so')
+      _TouchFile(host_libfoo_path)
+      _TouchFile(host_libbar_path)
+
+      finder.AddSearchDir(tmp_dir)
+
+      # Regular library path (extracted at installation by the PackageManager).
+      # Note that the extraction path has changed between Android releases,
+      # i.e. it can be /data/app/, /data/data/ or /data/app-lib/ depending
+      # on the system.
+      self.assertEqual(
+          host_libfoo_path,
+          finder.Find('/data/app-lib/com.example.app-1/lib/libfoo.so'))
+
+      # Verify that the path doesn't really matter
+      self.assertEqual(
+          host_libfoo_path,
+          finder.Find('/whatever/what.apk!lib/libfoo.so'))
+
+      self.assertEqual(
+          host_libbar_path,
+          finder.Find('/data/data/com.example.app-1/lib/libbar.so'))
+
+      self.assertIsNone(
+          finder.Find('/data/data/com.example.app-1/lib/libunknown.so'))
+
+
+  def testMultipleDirectories(self):
+    with _TempDir() as tmp_dir:
+      # Create the following files:
+      #   <tmp_dir>/aaa/
+      #      libfoo.so
+      #   <tmp_dir>/bbb/
+      #      libbar.so
+      #      libfoo.so    (this one should never be seen because 'aaa'
+      #                    shall be first in the search path list).
+      #
+      aaa_dir = os.path.join(tmp_dir, 'aaa')
+      bbb_dir = os.path.join(tmp_dir, 'bbb')
+      os.makedirs(aaa_dir)
+      os.makedirs(bbb_dir)
+
+      host_libfoo_path = os.path.join(aaa_dir, 'libfoo.so')
+      host_libbar_path = os.path.join(bbb_dir, 'libbar.so')
+      host_libfoo2_path = os.path.join(bbb_dir, 'libfoo.so')
+
+      _TouchFile(host_libfoo_path)
+      _TouchFile(host_libbar_path)
+      _TouchFile(host_libfoo2_path)
+
+      finder = symbol_utils.HostLibraryFinder()
+      finder.AddSearchDir(aaa_dir)
+      finder.AddSearchDir(bbb_dir)
+
+      self.assertEqual(
+          host_libfoo_path,
+          finder.Find('/data/data/com.example.app-1/lib/libfoo.so'))
+
+      self.assertEqual(
+          host_libfoo_path,
+          finder.Find('/data/whatever/base.apk!lib/libfoo.so'))
+
+      self.assertEqual(
+          host_libbar_path,
+          finder.Find('/data/data/com.example.app-1/lib/libbar.so'))
+
+      self.assertIsNone(
+          finder.Find('/data/data/com.example.app-1/lib/libunknown.so'))
+
+
+class ElfSymbolResolverTest(unittest.TestCase):
+
+  def testCreation(self):
+    resolver = symbol_utils.ElfSymbolResolver(
+        addr2line_path_for_tests=_MOCK_A2L_PATH)
+    self.assertTrue(resolver)
+
+  def testWithSimpleOffsets(self):
+    resolver = symbol_utils.ElfSymbolResolver(
+        addr2line_path_for_tests=_MOCK_A2L_PATH)
+    resolver.SetAndroidAbi('ignored-abi')
+
+    for addr, expected_sym in _TEST_SYMBOL_DATA.iteritems():
+      self.assertEqual(resolver.FindSymbolInfo('/some/path/libmock1.so', addr),
+                       expected_sym)
+
+  def testWithPreResolvedSymbols(self):
+    resolver = symbol_utils.ElfSymbolResolver(
+        addr2line_path_for_tests=_MOCK_A2L_PATH)
+    resolver.SetAndroidAbi('ignored-abi')
+    resolver.AddLibraryOffsets('/some/path/libmock1.so',
+                               _TEST_SYMBOL_DATA.keys())
+
+    resolver.DisallowSymbolizerForTesting()
+
+    for addr, expected_sym in _TEST_SYMBOL_DATA.iteritems():
+      sym_info = resolver.FindSymbolInfo('/some/path/libmock1.so', addr)
+      self.assertIsNotNone(sym_info, 'None symbol info for addr %x' % addr)
+      self.assertEqual(
+          sym_info, expected_sym,
+          'Invalid symbol info for addr %x [%s] expected [%s]' % (
+              addr, sym_info, expected_sym))
+
+
+class MemoryMapTest(unittest.TestCase):
+
+  def testCreation(self):
+    mem_map = symbol_utils.MemoryMap('test-abi32')
+    self.assertIsNone(mem_map.FindSectionForAddress(0))
+
+  def testParseLines(self):
+    mem_map = symbol_utils.MemoryMap('test-abi32')
+    mem_map.ParseLines(_TEST_MEMORY_MAP.splitlines())
+    for exp_addr, exp_size, exp_path, exp_offset in _TEST_MEMORY_MAP_SECTIONS:
+      text = '(addr:%x, size:%x, path:%s, offset=%x)' % (
+          exp_addr, exp_size, exp_path, exp_offset)
+
+      t = mem_map.FindSectionForAddress(exp_addr)
+      self.assertTrue(t, 'Could not find %s' % text)
+      self.assertEqual(t.address, exp_addr)
+      self.assertEqual(t.size, exp_size)
+      self.assertEqual(t.offset, exp_offset)
+      self.assertEqual(t.path, exp_path)
+
+  def testTranslateLine(self):
+    android_abi = 'test-abi'
+    apk_translator = MockApkTranslator(_TEST_APK_LIBS)
+    mem_map = symbol_utils.MemoryMap(android_abi)
+    for line, expected_line in zip(_TEST_MEMORY_MAP.splitlines(),
+                                   _EXPECTED_TEST_MEMORY_MAP.splitlines()):
+      self.assertEqual(mem_map.TranslateLine(line, apk_translator),
+                       expected_line)
+
+class StackTranslatorTest(unittest.TestCase):
+
+  def testSimpleStack(self):
+    android_abi = 'test-abi32'
+    mem_map = symbol_utils.MemoryMap(android_abi)
+    mem_map.ParseLines(_TEST_MEMORY_MAP)
+    apk_translator = MockApkTranslator(_TEST_APK_LIBS)
+    stack_translator = symbol_utils.StackTranslator(android_abi, mem_map,
+                                                    apk_translator)
+    input_stack = _TEST_STACK.splitlines()
+    expected_stack = _EXPECTED_STACK.splitlines()
+    self.assertEqual(len(input_stack), len(expected_stack))
+    for stack_line, expected_line in zip(input_stack, expected_stack):
+      new_line = stack_translator.TranslateLine(stack_line)
+      self.assertEqual(new_line, expected_line)
+
+
+class MockSymbolResolver(symbol_utils.SymbolResolver):
+
+  # A regex matching a symbol definition as it appears in a test symbol file.
+  # Format is: <hex-offset> <whitespace> <symbol-string>
+  _RE_SYMBOL_DEFINITION = re.compile(
+      r'(?P<offset>[0-9a-f]+)\s+(?P<symbol>.*)')
+
+  def __init__(self):
+    super(MockSymbolResolver, self).__init__()
+    self._map = collections.defaultdict(dict)
+
+  def AddTestLibrarySymbols(self, lib_name, offsets_map):
+    """Add a new test entry for a given library name.
+
+    Args:
+      lib_name: Library name (e.g. 'libfoo.so')
+      offsets_map: A mapping from offsets to symbol info strings.
+    """
+    self._map[lib_name] = offsets_map
+
+  def ReadTestFile(self, file_path, lib_name):
+    """Read a single test symbol file, matching a given library.
+
+    Args:
+      file_path: Input file path.
+      lib_name: Library name these symbols correspond to (e.g. 'libfoo.so')
+    """
+    with open(file_path) as f:
+      for line in f.readlines():
+        line = line.rstrip()
+        m = MockSymbolResolver._RE_SYMBOL_DEFINITION.match(line)
+        if m:
+          offset = int(m.group('offset'))
+          symbol = m.group('symbol')
+          self._map[lib_name][offset] = symbol
+
+  def ReadTestFilesInDir(self, dir_path, file_suffix):
+    """Read all symbol test files in a given directory.
+
+    Args:
+      dir_path: Directory path.
+      file_suffix: File suffix used to detect test symbol files.
+    """
+    for filename in os.listdir(dir_path):
+      if filename.endswith(file_suffix):
+        lib_name = filename[:-len(file_suffix)]
+        self.ReadTestFile(os.path.join(dir_path, filename), lib_name)
+
+  def FindSymbolInfo(self, device_path, device_offset):
+    """Implement SymbolResolver.FindSymbolInfo."""
+    lib_name = os.path.basename(device_path)
+    offsets = self._map.get(lib_name)
+    if not offsets:
+      return None
+
+    return offsets.get(device_offset)
+
+
+class BacktraceTranslatorTest(unittest.TestCase):
+
+  def testEmpty(self):
+    android_abi = 'test-abi'
+    apk_translator = MockApkTranslator()
+    backtrace_translator = symbol_utils.BacktraceTranslator(android_abi,
+                                                            apk_translator)
+    self.assertTrue(backtrace_translator)
+
+  def testFindLibraryOffsets(self):
+    android_abi = 'test-abi'
+    apk_translator = MockApkTranslator(_TEST_APK_LIBS)
+    backtrace_translator = symbol_utils.BacktraceTranslator(android_abi,
+                                                            apk_translator)
+    input_backtrace = _EXPECTED_BACKTRACE.splitlines()
+    expected_lib_offsets_map = _EXPECTED_BACKTRACE_OFFSETS_MAP
+    offset_map = backtrace_translator.FindLibraryOffsets(input_backtrace)
+    for lib_path, offsets in offset_map.iteritems():
+      self.assertTrue(lib_path in expected_lib_offsets_map,
+                      '%s is not in expected library-offsets map!' % lib_path)
+      sorted_offsets = sorted(offsets)
+      sorted_expected_offsets = sorted(expected_lib_offsets_map[lib_path])
+      self.assertEqual(sorted_offsets, sorted_expected_offsets,
+                       '%s has invalid offsets %s expected %s' % (
+                          lib_path, sorted_offsets, sorted_expected_offsets))
+
+  def testTranslateLine(self):
+    android_abi = 'test-abi'
+    apk_translator = MockApkTranslator(_TEST_APK_LIBS)
+    backtrace_translator = symbol_utils.BacktraceTranslator(android_abi,
+                                                            apk_translator)
+    input_backtrace = _TEST_BACKTRACE.splitlines()
+    expected_backtrace = _EXPECTED_BACKTRACE.splitlines()
+    self.assertEqual(len(input_backtrace), len(expected_backtrace))
+    for trace_line, expected_line in zip(input_backtrace, expected_backtrace):
+      line = backtrace_translator.TranslateLine(trace_line,
+                                                MockSymbolResolver())
+      self.assertEqual(line, expected_line)
+
+
+if __name__ == '__main__':
+  unittest.main()
diff --git a/build/android/pylib/utils/google_storage_helper.py b/build/android/pylib/utils/google_storage_helper.py
index 3101d71..d184810 100644
--- a/build/android/pylib/utils/google_storage_helper.py
+++ b/build/android/pylib/utils/google_storage_helper.py
@@ -79,10 +79,7 @@
 
   cmd = [_GSUTIL_PATH, '-q', 'stat', gs_path]
   return_code = cmd_helper.RunCmd(cmd)
-  if return_code == 0:
-    return True
-  else:
-    return False
+  return return_code == 0
 
 
 # TODO(jbudorick): Delete this function. Only one user of it.
diff --git a/build/android/pylib/utils/logdog_helper.py b/build/android/pylib/utils/logdog_helper.py
index 58fe781..68a7ba5 100644
--- a/build/android/pylib/utils/logdog_helper.py
+++ b/build/android/pylib/utils/logdog_helper.py
@@ -92,4 +92,3 @@
 def get_logdog_client():
   logging.info('Getting logdog client.')
   return bootstrap.ButlerBootstrap.probe().stream_client()
-
diff --git a/build/android/pylib/valgrind_tools.py b/build/android/pylib/valgrind_tools.py
index 3dc2488..4c27b08 100644
--- a/build/android/pylib/valgrind_tools.py
+++ b/build/android/pylib/valgrind_tools.py
@@ -127,4 +127,3 @@
     print 'Unknown tool %s, available tools: %s' % (
         tool_name, ', '.join(sorted(TOOL_REGISTRY.keys())))
     sys.exit(1)
-
diff --git a/build/android/pylintrc b/build/android/pylintrc
index 8005a5d..2a721bf2 100644
--- a/build/android/pylintrc
+++ b/build/android/pylintrc
@@ -4,7 +4,7 @@
 
 [MESSAGES CONTROL]
 
-disable=abstract-class-not-used,bad-continuation,bad-indentation,duplicate-code,fixme,invalid-name,locally-disabled,locally-enabled,missing-docstring,star-args,too-few-public-methods,too-many-arguments,too-many-branches,too-many-instance-attributes,too-many-lines,too-many-locals,too-many-public-methods,too-many-statements,
+disable=abstract-class-not-used,bad-continuation,bad-indentation,duplicate-code,fixme,invalid-name,locally-disabled,locally-enabled,missing-docstring,star-args,too-few-public-methods,too-many-arguments,too-many-branches,too-many-instance-attributes,too-many-lines,too-many-locals,too-many-public-methods,too-many-statements,wrong-import-position
 
 [REPORTS]
 
diff --git a/build/android/test_runner.py b/build/android/test_runner.py
index a83d9ab..8512acc 100755
--- a/build/android/test_runner.py
+++ b/build/android/test_runner.py
@@ -25,6 +25,7 @@
 # See http://crbug.com/724524 and https://bugs.python.org/issue7980.
 import _strptime  # pylint: disable=unused-import
 
+# pylint: disable=ungrouped-imports
 from pylib.constants import host_paths
 
 if host_paths.DEVIL_PATH not in sys.path:
@@ -49,7 +50,6 @@
 
 from py_utils import contextlib_ext
 
-
 _DEVIL_STATIC_CONFIG_FILE = os.path.abspath(os.path.join(
     host_paths.DIR_SOURCE_ROOT, 'build', 'android', 'devil_config.json'))
 
@@ -212,10 +212,12 @@
 def ProcessCommonOptions(args):
   """Processes and handles all common options."""
   run_tests_helper.SetLogLevel(args.verbose_count, add_handler=False)
+  # pylint: disable=redefined-variable-type
   if args.verbose_count > 0:
     handler = logging_utils.ColorStreamHandler()
   else:
     handler = logging.StreamHandler(sys.stdout)
+  # pylint: enable=redefined-variable-type
   handler.setFormatter(run_tests_helper.CustomFormatter())
   logging.getLogger().addHandler(handler)
 
@@ -429,6 +431,7 @@
     split_arg = arg.split(',')
     if len(split_arg) != 2:
       raise argparse.ArgumentError(
+          arg,
           'Expected two comma-separated strings for --replace-system-package, '
           'received %d' % len(split_arg))
     PackageReplacement = collections.namedtuple('PackageReplacement',
diff --git a/build/check_gn_headers_whitelist.txt b/build/check_gn_headers_whitelist.txt
index 9547f35..d570a03f 100644
--- a/build/check_gn_headers_whitelist.txt
+++ b/build/check_gn_headers_whitelist.txt
@@ -51,7 +51,6 @@
 chrome/browser/download/download_file_icon_extractor.h
 chrome/browser/extensions/api/networking_cast_private/chrome_networking_cast_private_delegate.h
 chrome/browser/extensions/api/omnibox/omnibox_api_testbase.h
-chrome/browser/extensions/api/socket/mock_tcp_client_socket.h
 chrome/browser/mac/bluetooth_utility.h
 chrome/browser/media/router/mojo/media_route_provider_util_win.h
 chrome/browser/media/webrtc/desktop_media_list_ash.h
diff --git a/chrome/android/chrome_public_apk_tmpl.gni b/chrome/android/chrome_public_apk_tmpl.gni
index 6aea38e..7af1b221 100644
--- a/chrome/android/chrome_public_apk_tmpl.gni
+++ b/chrome/android/chrome_public_apk_tmpl.gni
@@ -3,6 +3,7 @@
 # found in the LICENSE file.
 
 import("//base/android/linker/config.gni")
+import("//base/android/proguard/proguard.gni")
 import("//build/config/android/rules.gni")
 import("//build/config/locales.gni")
 import("//build/config/android/extract_unwind_tables.gni")
@@ -13,9 +14,6 @@
 import("channel.gni")
 
 declare_args() {
-  # Enables ProGuard obfuscation of Chromium packages.
-  enable_proguard_obfuscation = true
-
   # Enable multidex in release builds.
   multidex_in_release = false
 }
@@ -105,7 +103,7 @@
             [ "//base/android/proguard/enable_obfuscation.flags" ]
       } else {
         proguard_configs +=
-            [ "//base/android/proguard/disable_chromium_obfuscation.flags" ]
+            [ "//base/android/proguard/disable_all_obfuscation.flags" ]
       }
     }
 
diff --git a/chrome/android/java/res/layout/toolbar_phone_common.xml b/chrome/android/java/res/layout/toolbar_phone_common.xml
index e136f91..f456850 100644
--- a/chrome/android/java/res/layout/toolbar_phone_common.xml
+++ b/chrome/android/java/res/layout/toolbar_phone_common.xml
@@ -44,23 +44,27 @@
             android:layout_width="4dp"
             android:layout_height="match_parent" />
 
-        <ViewStub
-            android:id="@+id/experimental_button_stub"
-            android:inflatedId="@+id/experimental_toolbar_button"
-            android:layout="@layout/experimental_toolbar_button"
-            style="@style/ToolbarButton"
-            android:paddingStart="8dp"
-            android:layout_gravity="top"
-            android:visibility="gone" />
+        <FrameLayout
+            android:layout_width="wrap_content"
+            android:layout_height="wrap_content"
+            android:layout_gravity="top" >
 
-        <ViewStub
-            android:id="@+id/incognito_button_stub"
-            android:inflatedId="@+id/incognito_button"
-            android:layout="@layout/toolbar_phone_incognito_button"
-            style="@style/ToolbarButton"
-            android:layout_gravity="top"
-            android:contentDescription="@string/accessibility_tabstrip_btn_incognito_toggle_standard"
-            android:visibility="gone" />
+            <ViewStub
+                android:id="@+id/experimental_button_stub"
+                android:inflatedId="@+id/experimental_toolbar_button"
+                android:layout="@layout/experimental_toolbar_button"
+                style="@style/ToolbarButton"
+                android:paddingStart="8dp"
+                android:visibility="gone" />
+
+            <ViewStub
+                android:id="@+id/incognito_button_stub"
+                android:inflatedId="@+id/incognito_button"
+                android:layout="@layout/toolbar_phone_incognito_button"
+                style="@style/ToolbarButton"
+                android:contentDescription="@string/accessibility_tabstrip_btn_incognito_toggle_standard"
+                android:visibility="gone" />
+        </FrameLayout>
 
         <ImageButton android:id="@+id/tab_switcher_button"
             style="@style/ToolbarButton"
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/WarmupManager.java b/chrome/android/java/src/org/chromium/chrome/browser/WarmupManager.java
index 13a4a44..fd948bb77 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/WarmupManager.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/WarmupManager.java
@@ -217,9 +217,9 @@
      */
     private void prefetchDnsForUrlInBackground(final String url) {
         mDnsRequestsInFlight.add(url);
-        new AsyncTask<String, Void, Void>() {
+        new AsyncTask<Void, Void, Void>() {
             @Override
-            protected Void doInBackground(String... params) {
+            protected Void doInBackground(Void... params) {
                 try (TraceEvent e =
                                 TraceEvent.scoped("WarmupManager.prefetchDnsForUrlInBackground")) {
                     InetAddress.getByName(new URL(url).getHost());
@@ -242,7 +242,8 @@
                     maybePreconnectUrlAndSubResources(profile, url);
                 }
             }
-        }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, url);
+        }
+                .executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
     }
 
     /** Launches a background DNS query for a given URL if the data reduction proxy is not in use.
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/feedback/AsyncFeedbackSourceAdapter.java b/chrome/android/java/src/org/chromium/chrome/browser/feedback/AsyncFeedbackSourceAdapter.java
index dae4a74d..2ce8121 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/feedback/AsyncFeedbackSourceAdapter.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/feedback/AsyncFeedbackSourceAdapter.java
@@ -23,7 +23,7 @@
 public abstract class AsyncFeedbackSourceAdapter<Result> implements AsyncFeedbackSource {
     private Worker mWorker;
 
-    private class Worker extends AsyncTask<Context, Void, Result> {
+    private class Worker extends AsyncTask<Void, Void, Result> {
         private final Runnable mCallback;
 
         public Worker(Runnable callback) {
@@ -32,14 +32,13 @@
 
         // AsyncTask implementation.
         @Override
-        protected Result doInBackground(Context... params) {
-            return AsyncFeedbackSourceAdapter.this.doInBackground(params[0]);
+        protected Result doInBackground(Void... params) {
+            return AsyncFeedbackSourceAdapter.this.doInBackground(
+                    ContextUtils.getApplicationContext());
         }
 
         @Override
         protected void onPostExecute(Result result) {
-            super.onPostExecute(result);
-
             mCallback.run();
         }
     }
@@ -75,7 +74,6 @@
     public final void start(Runnable callback) {
         if (mWorker != null) return;
         mWorker = new Worker(callback);
-        mWorker.executeOnExecutor(
-                AsyncTask.THREAD_POOL_EXECUTOR, ContextUtils.getApplicationContext());
+        mWorker.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
     }
 }
\ No newline at end of file
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/offlinepages/evaluation/OfflinePageEvaluationBridge.java b/chrome/android/java/src/org/chromium/chrome/browser/offlinepages/evaluation/OfflinePageEvaluationBridge.java
index bb9aafb4..45a70997 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/offlinepages/evaluation/OfflinePageEvaluationBridge.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/offlinepages/evaluation/OfflinePageEvaluationBridge.java
@@ -66,12 +66,18 @@
      * Class used for writing logs to external log file asynchronously to prevent violating strict
      * mode during test.
      */
-    private class LogTask extends AsyncTask<String, Void, Void> {
+    private class LogTask extends AsyncTask<Void, Void, Void> {
+        final String mLogString;
+
+        LogTask(String logString) {
+            mLogString = logString;
+        }
+
         @Override
-        protected Void doInBackground(String... strings) {
+        protected Void doInBackground(Void... params) {
             try {
                 synchronized (mLogOutput) {
-                    mLogOutput.write(strings[0]);
+                    mLogOutput.write(mLogString);
                     mLogOutput.flush();
                 }
             } catch (IOException e) {
@@ -198,9 +204,9 @@
                 new SimpleDateFormat("MM-dd HH:mm:ss.SSS", Locale.getDefault());
         String logString = formatter.format(date) + ": " + sourceTag + " | " + message
                 + System.getProperty("line.separator");
-        LogTask logTask = new LogTask();
+        LogTask logTask = new LogTask(logString);
         Log.d(TAG, logString);
-        logTask.executeOnExecutor(AsyncTask.SERIAL_EXECUTOR, logString);
+        logTask.executeOnExecutor(AsyncTask.SERIAL_EXECUTOR);
     }
 
     public void closeLog() {
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/BitmapScalerTask.java b/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/BitmapScalerTask.java
index 123f0d3b..656b023 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/BitmapScalerTask.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/BitmapScalerTask.java
@@ -17,18 +17,21 @@
 /**
  * A worker task to scale bitmaps in the background.
  */
-class BitmapScalerTask extends AsyncTask<Bitmap, Void, Bitmap> {
+class BitmapScalerTask extends AsyncTask<Void, Void, Bitmap> {
     private final LruCache<String, Bitmap> mCache;
     private final String mFilePath;
     private final int mSize;
+    private final Bitmap mBitmap;
 
     /**
      * A BitmapScalerTask constructor.
      */
-    public BitmapScalerTask(LruCache<String, Bitmap> cache, String filePath, int size) {
+    public BitmapScalerTask(
+            LruCache<String, Bitmap> cache, String filePath, int size, Bitmap bitmap) {
         mCache = cache;
         mFilePath = filePath;
         mSize = size;
+        mBitmap = bitmap;
     }
 
     /**
@@ -37,13 +40,13 @@
      * @return A sorted list of images (by last-modified first).
      */
     @Override
-    protected Bitmap doInBackground(Bitmap... bitmaps) {
+    protected Bitmap doInBackground(Void... params) {
         assert !ThreadUtils.runningOnUiThread();
 
         if (isCancelled()) return null;
 
         long begin = SystemClock.elapsedRealtime();
-        Bitmap bitmap = BitmapUtils.scale(bitmaps[0], mSize, false);
+        Bitmap bitmap = BitmapUtils.scale(mBitmap, mSize, false);
         long scaleTime = SystemClock.elapsedRealtime() - begin;
         RecordHistogram.recordTimesHistogram(
                 "Android.PhotoPicker.BitmapScalerTask", scaleTime, TimeUnit.MILLISECONDS);
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/PickerBitmapViewHolder.java b/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/PickerBitmapViewHolder.java
index 72a39c69..ef22a284 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/PickerBitmapViewHolder.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/photo_picker/PickerBitmapViewHolder.java
@@ -55,8 +55,9 @@
         if (mCategoryView.getLowResBitmaps().get(filePath) == null) {
             Resources resources = mItemView.getContext().getResources();
             new BitmapScalerTask(mCategoryView.getLowResBitmaps(), filePath,
-                    resources.getDimensionPixelSize(R.dimen.photo_picker_grainy_thumbnail_size))
-                    .executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, bitmap);
+                    resources.getDimensionPixelSize(R.dimen.photo_picker_grainy_thumbnail_size),
+                    bitmap)
+                    .executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
         }
 
         if (!TextUtils.equals(mBitmapDetails.getFilePath(), filePath)) {
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/toolbar/ToolbarPhone.java b/chrome/android/java/src/org/chromium/chrome/browser/toolbar/ToolbarPhone.java
index 4803507..8da8a34c 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/toolbar/ToolbarPhone.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/toolbar/ToolbarPhone.java
@@ -868,8 +868,12 @@
         // We set the incognito toggle button's visibility from GONE to VISIBLE when the tab
         // switcher starts to open, but we don't want this to affect the Omnibox's size during the
         // animation, so we have to make an adjustment here.
+        // However, if the experimental button is showing it sits in a FrameLayout with the
+        // incognito button and the omnibox will be appropriately sized without an explicit
+        // adjustment.
         int incognitoButtonWidth = 0;
-        if (mIncognitoToggleButton != null && mIncognitoToggleButton.getVisibility() == VISIBLE) {
+        if (mIncognitoToggleButton != null && mIncognitoToggleButton.getVisibility() == VISIBLE
+                && (mExperimentalButton == null || mExperimentalButton.getVisibility() == GONE)) {
             incognitoButtonWidth += mIncognitoToggleButton.getMeasuredWidth();
         }
         return Math.max(mToolbarSidePadding,
@@ -1392,10 +1396,15 @@
 
         // Draw the experimental button if necessary.
         if (mExperimentalButton != null && mExperimentalButton.getVisibility() != View.GONE) {
+            canvas.save();
+            translateCanvasToView(mToolbarButtonsContainer, mExperimentalButton, canvas);
+
             previousAlpha = mExperimentalButton.getAlpha();
             mExperimentalButton.setAlpha(previousAlpha * floatAlpha);
             drawChild(canvas, mExperimentalButton, SystemClock.uptimeMillis());
             mExperimentalButton.setAlpha(previousAlpha);
+
+            canvas.restore();
         }
 
         // Draw the tab stack button and associated text.
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/vr/VrShellImpl.java b/chrome/android/java/src/org/chromium/chrome/browser/vr/VrShellImpl.java
index 94467d3..3ac2c5c 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/vr/VrShellImpl.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/vr/VrShellImpl.java
@@ -164,12 +164,6 @@
             mActivity.getFindToolbarManager().hideToolbar();
         }
 
-        if (ChromeFeatureList.isEnabled(ChromeFeatureList.VR_BROWSING_NATIVE_ANDROID_UI)) {
-            mNonVrUiWidgetFactory = UiWidgetFactory.getInstance();
-            UiWidgetFactory.setInstance(
-                    new VrUiWidgetFactory(this, mActivity.getModalDialogManager()));
-        }
-
         // This overrides the default intent created by GVR to return to Chrome when the DON flow
         // is triggered by resuming the GvrLayout, which is the usual way Daydream apps enter VR.
         // See VrShellDelegate#getEnterVrPendingIntent for why we need to do this.
@@ -200,6 +194,13 @@
 
         if (mVrBrowsingEnabled) injectVrHostedUiView();
 
+        // This has to happen after VrModalDialogManager is created.
+        if (ChromeFeatureList.isEnabled(ChromeFeatureList.VR_BROWSING_NATIVE_ANDROID_UI)) {
+            mNonVrUiWidgetFactory = UiWidgetFactory.getInstance();
+            UiWidgetFactory.setInstance(
+                    new VrUiWidgetFactory(this, mActivity.getModalDialogManager()));
+        }
+
         mTabRedirectHandler = new TabRedirectHandler(mActivity) {
             @Override
             public boolean shouldStayInChrome(boolean hasExternalProtocol) {
diff --git a/chrome/android/webapk/shell_apk/src/org/chromium/webapk/shell_apk/WebApkUtils.java b/chrome/android/webapk/shell_apk/src/org/chromium/webapk/shell_apk/WebApkUtils.java
index a10297901..695ccbb 100644
--- a/chrome/android/webapk/shell_apk/src/org/chromium/webapk/shell_apk/WebApkUtils.java
+++ b/chrome/android/webapk/shell_apk/src/org/chromium/webapk/shell_apk/WebApkUtils.java
@@ -71,7 +71,7 @@
         try {
             String hostPackage = getHostBrowserPackageName(context);
             return context.getApplicationContext().createPackageContext(
-                    hostPackage, Context.CONTEXT_IGNORE_SECURITY | Context.CONTEXT_INCLUDE_CODE);
+                    hostPackage, 0);
         } catch (NameNotFoundException e) {
             e.printStackTrace();
         }
diff --git a/chrome/app/chromium_strings.grd b/chrome/app/chromium_strings.grd
index d3d5d68a..7d4f3f8c 100644
--- a/chrome/app/chromium_strings.grd
+++ b/chrome/app/chromium_strings.grd
@@ -961,6 +961,9 @@
         </message>
       </if>
       <if expr="chromeos">
+        <message name="IDS_UPDATE_RECOMMENDED_DIALOG_TITLE" desc="The window title for the Update Recommended dialog.">
+          Restart Chromium OS
+        </message>
         <message name="IDS_UPDATE_RECOMMENDED" desc="The main text of the Update Recommended dialog.">
           Chromium OS needs to be restarted to apply the update.
         </message>
@@ -1187,40 +1190,79 @@
         </message>
       </if>
       <!-- Relaunch notification bubble and dialog. -->
-      <if expr="not is_android and not chromeos">
-        <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a browser relaunch is recommended for an update available for some number of days.">
-          {0, plural,
-           =0 {A Chromium update is available}
-           =1 {A Chromium update is available}
-           other {A Chromium update has been available for # days}}
-        </message>
-        <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a browser relaunch is recommended for an update.">
-          Your administrator asks that you relaunch Chromium to apply this update
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the browser must be relaunched within two or more days.">
-          {0, plural,
-           =1 {Relaunch Chromium within a day}
-           other {Relaunch Chromium within # days}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the browser must be relaunched within one or more hours.">
-          {0, plural,
-           =1 {Chromium will relaunch in an hour}
-           other {Chromium will relaunch in # hours}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the browser must be relaunched within one or more minutes.">
-          {0, plural,
-           =1 {Chromium will relaunch in 1 minute}
-           other {Chromium will relaunch in # minutes}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the browser must be relaunched within some number of seconds.">
-          {0, plural,
-           =0 {Chromium will relaunch now}
-           =1 {Chromium will relaunch in 1 second}
-           other {Chromium will relaunch in # seconds}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the browser must be relaunched.">
-          Your administrator requires that you relaunch Chromium to apply an update
-        </message>
+      <if expr="not is_android">
+        <if expr="not chromeos">
+          <then>
+            <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a browser relaunch is recommended for an update available for some number of days.">
+              {0, plural,
+               =0 {A Chromium update is available}
+               =1 {A Chromium update is available}
+               other {A Chromium update has been available for # days}}
+            </message>
+            <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a browser relaunch is recommended for an update.">
+              Your administrator asks that you relaunch Chromium to apply this update
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the browser must be relaunched within two or more days.">
+              {0, plural,
+               =1 {Relaunch Chromium within a day}
+               other {Relaunch Chromium within # days}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the browser must be relaunched within one or more hours.">
+              {0, plural,
+               =1 {Chromium will relaunch in an hour}
+               other {Chromium will relaunch in # hours}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the browser must be relaunched within one or more minutes.">
+              {0, plural,
+               =1 {Chromium will relaunch in 1 minute}
+               other {Chromium will relaunch in # minutes}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the browser must be relaunched within some number of seconds.">
+              {0, plural,
+               =0 {Chromium will relaunch now}
+               =1 {Chromium will relaunch in 1 second}
+               other {Chromium will relaunch in # seconds}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the browser must be relaunched.">
+              Your administrator requires that you relaunch Chromium to apply an update
+            </message>
+          </then>
+          <else>
+            <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a device restart is recommended for an update available for some number of days.">
+              {0, plural,
+               =0 {A Chromium OS update is available}
+               =1 {A Chromium OS update is available}
+               other {A Chromium OS update has been available for # days}}
+            </message>
+            <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a device restart is recommended for an update.">
+              Your administrator asks that you restart Chromium OS to apply this update
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the device must be restarted within two or more days.">
+              {0, plural,
+               =1 {Restart Chromium OS within a day}
+               other {Restart Chromium OS within # days}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the device must be restarted within one or more hours.">
+              {0, plural,
+               =1 {Chromium OS will restart in an hour}
+               other {Chromium OS will restart in # hours}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the device must be restarted within one or more minutes.">
+              {0, plural,
+               =1 {Chromium OS will restart in 1 minute}
+               other {Chromium OS will restart in # minutes}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the device must be restarted within some number of seconds.">
+              {0, plural,
+               =0 {Chromium OS will restart now}
+               =1 {Chromium OS will restart in 1 second}
+               other {Chromium OS will restart in # seconds}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the device must be restarted.">
+              Your administrator requires that you restart Chromium OS to apply an update
+            </message>
+          </else>
+        </if>
       </if>
       <!-- Chroium launch blocking dialog. -->
       <if expr="not is_android and not chromeos">
diff --git a/chrome/app/generated_resources.grd b/chrome/app/generated_resources.grd
index 7c9aa1a..7356de0 100644
--- a/chrome/app/generated_resources.grd
+++ b/chrome/app/generated_resources.grd
@@ -9018,10 +9018,17 @@
       Upload
     </message>
     <!-- Relaunch notification bubble and dialog. -->
-    <if expr="not is_android and not chromeos">
-      <message name="IDS_RELAUNCH_ACCEPT_BUTTON" desc="The text in the accept button of a dialog that causes the browser to be relaunched immediately.">
-        Relaunch now
-      </message>
+    <if expr="not is_android">
+      <if expr="not chromeos">
+        <message name="IDS_RELAUNCH_ACCEPT_BUTTON" desc="The text in the accept button of a dialog that causes the browser to be relaunched immediately.">
+          Relaunch now
+        </message>
+      </if>
+      <if expr="chromeos">
+        <message name="IDS_RELAUNCH_ACCEPT_BUTTON" desc="The text in the accept button of a dialog that causes the device to be restarted immediately.">
+          Restart now
+        </message>
+      </if>
       <message name="IDS_RELAUNCH_REQUIRED_CANCEL_BUTTON" desc="The text in the cancel button that dismisses a dialog telling users the browser must be relaunched.">
         Got it
       </message>
diff --git a/chrome/app/google_chrome_strings.grd b/chrome/app/google_chrome_strings.grd
index 4552b231..ec8b999c9 100644
--- a/chrome/app/google_chrome_strings.grd
+++ b/chrome/app/google_chrome_strings.grd
@@ -979,6 +979,9 @@
         </message>
       </if>
       <if expr="chromeos">
+        <message name="IDS_UPDATE_RECOMMENDED_DIALOG_TITLE" desc="The window title for the Update Recommended dialog.">
+          Restart Chrome OS
+        </message>
         <message name="IDS_UPDATE_RECOMMENDED" desc="The main text of the Update Recommended dialog.">
           Chrome OS needs to be restarted to apply the update.
         </message>
@@ -1205,40 +1208,79 @@
         </message>
       </if>
       <!-- Relaunch notification bubble and dialog. -->
-      <if expr="not is_android and not chromeos">
-        <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a browser relaunch is recommended for an update available for some number of days.">
-          {0, plural,
-           =0 {A Chrome update is available}
-           =1 {A Chrome update is available}
-           other {A Chrome update has been available for # days}}
-        </message>
-        <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a browser relaunch is recommended for an update.">
-          Your administrator asks that you relaunch Chrome to apply this update
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the browser must be relaunched within two or more days.">
-          {0, plural,
-           =1 {Relaunch Chrome within a day}
-           other {Relaunch Chrome within # days}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the browser must be relaunched within one or more hours.">
-          {0, plural,
-           =1 {Chrome will relaunch in an hour}
-           other {Chrome will relaunch in # hours}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the browser must be relaunched within one or more minutes.">
-          {0, plural,
-           =1 {Chrome will relaunch in 1 minute}
-           other {Chrome will relaunch in # minutes}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the browser must be relaunched within some number of seconds.">
-          {0, plural,
-           =0 {Chrome will relaunch now}
-           =1 {Chrome will relaunch in 1 second}
-           other {Chrome will relaunch in # seconds}}
-        </message>
-        <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the browser must be relaunched.">
-          Your administrator requires that you relaunch Chrome to apply an update
-        </message>
+      <if expr="not is_android">
+        <if expr="not chromeos">
+          <then>
+            <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a browser relaunch is recommended for an update available for some number of days.">
+              {0, plural,
+               =0 {A Chrome update is available}
+               =1 {A Chrome update is available}
+               other {A Chrome update has been available for # days}}
+            </message>
+            <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a browser relaunch is recommended for an update.">
+              Your administrator asks that you relaunch Chrome to apply this update
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the browser must be relaunched within two or more days.">
+              {0, plural,
+               =1 {Relaunch Chrome within a day}
+               other {Relaunch Chrome within # days}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the browser must be relaunched within one or more hours.">
+              {0, plural,
+               =1 {Chrome will relaunch in an hour}
+               other {Chrome will relaunch in # hours}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the browser must be relaunched within one or more minutes.">
+              {0, plural,
+               =1 {Chrome will relaunch in 1 minute}
+               other {Chrome will relaunch in # minutes}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the browser must be relaunched within some number of seconds.">
+              {0, plural,
+               =0 {Chrome will relaunch now}
+               =1 {Chrome will relaunch in 1 second}
+               other {Chrome will relaunch in # seconds}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the browser must be relaunched.">
+              Your administrator requires that you relaunch Chrome to apply an update
+            </message>
+          </then>
+          <else>
+            <message name="IDS_RELAUNCH_RECOMMENDED_TITLE" desc="The title of a dialog that tells users that a device restart is recommended for an update available for some number of days.">
+              {0, plural,
+               =0 {A Chrome OS update is available}
+               =1 {A Chrome OS update is available}
+               other {A Chrome OS update has been available for # days}}
+            </message>
+            <message name="IDS_RELAUNCH_RECOMMENDED_BODY" desc="The body text of a dialog that tells users that a device restart is recommended for an update.">
+              Your administrator asks that you restart Chrome OS to apply this update
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_DAYS" desc="The title of a dialog that tells users the device must be restarted within two or more days.">
+              {0, plural,
+               =1 {Restart Chrome OS within a day}
+               other {Restart Chrome OS within # days}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_HOURS" desc="The title of a dialog that tells users the device must be restarted within one or more hours.">
+              {0, plural,
+               =1 {Chrome OS will restart in an hour}
+               other {Chrome OS will restart in # hours}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_MINUTES" desc="The title of a dialog that tells users the device must be restarted within one or more minutes.">
+              {0, plural,
+               =1 {Chrome OS will restart in 1 minute}
+               other {Chrome OS will restart in # minutes}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_TITLE_SECONDS" desc="The title of a dialog that tells users the device must be restarted within some number of seconds.">
+              {0, plural,
+               =0 {Chrome OS will restart now}
+               =1 {Chrome OS will restart in 1 second}
+               other {Chrome OS will restart in # seconds}}
+            </message>
+            <message name="IDS_RELAUNCH_REQUIRED_BODY" desc="The body text of a dialog that tells users the device must be restarted.">
+              Your administrator requires that you restart Chrome OS to apply an update
+            </message>
+          </else>
+        </if>
       </if>
 
       <!-- Chrome launch blocking dialog. -->
diff --git a/chrome/browser/about_flags.cc b/chrome/browser/about_flags.cc
index ddb5ad4b..0aaade1f2 100644
--- a/chrome/browser/about_flags.cc
+++ b/chrome/browser/about_flags.cc
@@ -1678,6 +1678,11 @@
      flag_descriptions::kNewPasswordFormParsingName,
      flag_descriptions::kNewPasswordFormParsingDescription, kOsAll,
      FEATURE_VALUE_TYPE(password_manager::features::kNewPasswordFormParsing)},
+    {"new-password-form-parsing-for-saving",
+     flag_descriptions::kNewPasswordFormParsingForSavingName,
+     flag_descriptions::kNewPasswordFormParsingForSavingDescription, kOsAll,
+     FEATURE_VALUE_TYPE(
+         password_manager::features::kNewPasswordFormParsingForSaving)},
     {"enable-show-autofill-signatures",
      flag_descriptions::kShowAutofillSignaturesName,
      flag_descriptions::kShowAutofillSignaturesDescription, kOsAll,
diff --git a/chrome/browser/android/vr/arcore_device/arcore_device.cc b/chrome/browser/android/vr/arcore_device/arcore_device.cc
index d49ff76..0b1b6f9e 100644
--- a/chrome/browser/android/vr/arcore_device/arcore_device.cc
+++ b/chrome/browser/android/vr/arcore_device/arcore_device.cc
@@ -277,7 +277,7 @@
 }
 
 void ARCoreDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   TRACE_EVENT0("gpu", __FUNCTION__);
   DCHECK(IsOnMainThread());
   // We should not be able to reach this point if we are not initialized.
@@ -312,7 +312,7 @@
 
 void ARCoreDevice::RequestHitTest(
     mojom::XRRayPtr ray,
-    mojom::VRMagicWindowProvider::RequestHitTestCallback callback) {
+    mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback) {
   DCHECK(IsOnMainThread());
 
   PostTaskToGlThread(base::BindOnce(
@@ -441,16 +441,19 @@
         &ARCoreGl::Resume, arcore_gl_thread_->GetARCoreGl()->GetWeakPtr()));
   }
 
-  // TODO(offenwanger) When the XRMagicWindowProvider or equivalent is returned
-  // here, clean out this dummy code.
-  auto connection = mojom::XRPresentationConnection::New();
-  mojom::VRSubmitFrameClientPtr submit_client;
-  connection->client_request = mojo::MakeRequest(&submit_client);
-  mojom::VRPresentationProviderPtr provider;
-  mojo::MakeRequest(&provider);
-  connection->provider = provider.PassInterface();
-  connection->transport_options = mojom::VRDisplayFrameTransportOptions::New();
-  std::move(callback).Run(std::move(connection), nullptr);
+  auto session = mojom::XRSession::New();
+
+  mojom::XRFrameDataProviderPtr data_provider;
+  mojom::XREnviromentIntegrationProviderPtr enviroment_provider;
+  mojom::XRSessionControllerPtr controller;
+  magic_window_sessions_.push_back(std::make_unique<VRDisplayImpl>(
+      this, mojo::MakeRequest(&data_provider),
+      mojo::MakeRequest(&enviroment_provider), mojo::MakeRequest(&controller)));
+
+  session->data_provider = data_provider.PassInterface();
+  session->enviroment_provider = enviroment_provider.PassInterface();
+
+  std::move(callback).Run(std::move(session), std::move(controller));
 }
 
 void ARCoreDevice::OnRequestAndroidCameraPermissionResult(
diff --git a/chrome/browser/android/vr/arcore_device/arcore_device.h b/chrome/browser/android/vr/arcore_device/arcore_device.h
index ad09a36..d3ecff7 100644
--- a/chrome/browser/android/vr/arcore_device/arcore_device.h
+++ b/chrome/browser/android/vr/arcore_device/arcore_device.h
@@ -52,10 +52,11 @@
   // VRDeviceBase implementation
   bool ShouldPauseTrackingWhenFrameDataRestricted() override;
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
   void RequestHitTest(
       mojom::XRRayPtr ray,
-      mojom::VRMagicWindowProvider::RequestHitTestCallback callback) override;
+      mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback)
+      override;
 
   void OnMailboxBridgeReady();
   void OnARCoreGlThreadInitialized();
diff --git a/chrome/browser/android/vr/arcore_device/arcore_gl.cc b/chrome/browser/android/vr/arcore_device/arcore_gl.cc
index 891ed17..a7dd48da 100644
--- a/chrome/browser/android/vr/arcore_device/arcore_gl.cc
+++ b/chrome/browser/android/vr/arcore_device/arcore_gl.cc
@@ -83,7 +83,7 @@
   ARCoreHitTestRequest() = default;
   ~ARCoreHitTestRequest() = default;
   mojom::XRRayPtr ray;
-  mojom::VRMagicWindowProvider::RequestHitTestCallback callback;
+  mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback;
 
  private:
   DISALLOW_COPY_AND_ASSIGN(ARCoreHitTestRequest);
@@ -176,7 +176,7 @@
 void ARCoreGl::ProduceFrame(
     const gfx::Size& frame_size,
     display::Display::Rotation display_rotation,
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   TRACE_EVENT0("gpu", __FUNCTION__);
   DCHECK(IsOnGlThread());
   DCHECK(is_initialized_);
@@ -252,7 +252,7 @@
 
 void ARCoreGl::RequestHitTest(
     mojom::XRRayPtr ray,
-    mojom::VRMagicWindowProvider::RequestHitTestCallback callback) {
+    mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback) {
   DCHECK(IsOnGlThread());
   DCHECK(is_initialized_);
 
@@ -266,7 +266,7 @@
 void ARCoreGl::ProcessFrame(
     mojom::XRFrameDataPtr frame_data,
     const gfx::Size& frame_size,
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   DCHECK(IsOnGlThread());
   DCHECK(is_initialized_);
 
diff --git a/chrome/browser/android/vr/arcore_device/arcore_gl.h b/chrome/browser/android/vr/arcore_device/arcore_gl.h
index d190193..2262ca97 100644
--- a/chrome/browser/android/vr/arcore_device/arcore_gl.h
+++ b/chrome/browser/android/vr/arcore_device/arcore_gl.h
@@ -50,7 +50,7 @@
 
   void ProduceFrame(const gfx::Size& frame_size,
                     display::Display::Rotation display_rotation,
-                    mojom::VRMagicWindowProvider::GetFrameDataCallback);
+                    mojom::XRFrameDataProvider::GetFrameDataCallback);
   void Pause();
   void Resume();
 
@@ -58,17 +58,17 @@
     return gl_thread_task_runner_;
   }
 
-  void RequestHitTest(mojom::XRRayPtr,
-                      mojom::VRMagicWindowProvider::RequestHitTestCallback);
+  void RequestHitTest(
+      mojom::XRRayPtr,
+      mojom::XREnviromentIntegrationProvider::RequestHitTestCallback);
 
   base::WeakPtr<ARCoreGl> GetWeakPtr();
 
  private:
   // TODO(https://crbug/835948): remove frame_size.
-  void ProcessFrame(
-      mojom::XRFrameDataPtr frame_data,
-      const gfx::Size& frame_size,
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback);
+  void ProcessFrame(mojom::XRFrameDataPtr frame_data,
+                    const gfx::Size& frame_size,
+                    mojom::XRFrameDataProvider::GetFrameDataCallback callback);
 
   bool InitializeGl();
   bool IsOnGlThread() const;
diff --git a/chrome/browser/android/vr/gl_browser_interface.h b/chrome/browser/android/vr/gl_browser_interface.h
index 77959c2..0e03d914 100644
--- a/chrome/browser/android/vr/gl_browser_interface.h
+++ b/chrome/browser/android/vr/gl_browser_interface.h
@@ -32,11 +32,8 @@
   virtual void ContentOverlaySurfaceCreated(jobject surface,
                                             gl::SurfaceTexture* texture) = 0;
   virtual void GvrDelegateReady(gvr::ViewerType viewer_type) = 0;
-  virtual void SendRequestPresentReply(
-      bool success,
-      device::mojom::VRSubmitFrameClientRequest,
-      device::mojom::VRPresentationProviderPtr,
-      device::mojom::VRDisplayFrameTransportOptionsPtr) = 0;
+  // XRSessionPtr is optional, if null, the request failed.
+  virtual void SendRequestPresentReply(device::mojom::XRSessionPtr) = 0;
   virtual void DialogSurfaceCreated(jobject surface,
                                     gl::SurfaceTexture* texture) = 0;
   virtual void UpdateGamepadData(device::GvrGamepadData) = 0;
diff --git a/chrome/browser/android/vr/vr_gl_thread.cc b/chrome/browser/android/vr/vr_gl_thread.cc
index 0e88057..22b57b5 100644
--- a/chrome/browser/android/vr/vr_gl_thread.cc
+++ b/chrome/browser/android/vr/vr_gl_thread.cc
@@ -131,17 +131,11 @@
       base::BindOnce(&VrShell::GvrDelegateReady, weak_vr_shell_, viewer_type));
 }
 
-void VrGLThread::SendRequestPresentReply(
-    bool success,
-    device::mojom::VRSubmitFrameClientRequest request,
-    device::mojom::VRPresentationProviderPtr provider,
-    device::mojom::VRDisplayFrameTransportOptionsPtr transport_options) {
+void VrGLThread::SendRequestPresentReply(device::mojom::XRSessionPtr session) {
   DCHECK(OnGlThread());
   main_thread_task_runner_->PostTask(
-      FROM_HERE,
-      base::BindOnce(&VrShell::SendRequestPresentReply, weak_vr_shell_, success,
-                     std::move(request), provider.PassInterface(),
-                     std::move(transport_options)));
+      FROM_HERE, base::BindOnce(&VrShell::SendRequestPresentReply,
+                                weak_vr_shell_, std::move(session)));
 }
 
 void VrGLThread::UpdateGamepadData(device::GvrGamepadData pad) {
diff --git a/chrome/browser/android/vr/vr_gl_thread.h b/chrome/browser/android/vr/vr_gl_thread.h
index 127aef7..2a490eb 100644
--- a/chrome/browser/android/vr/vr_gl_thread.h
+++ b/chrome/browser/android/vr/vr_gl_thread.h
@@ -65,11 +65,7 @@
   void ContentOverlaySurfaceCreated(jobject surface,
                                     gl::SurfaceTexture* texture) override;
   void GvrDelegateReady(gvr::ViewerType viewer_type) override;
-  void SendRequestPresentReply(
-      bool success,
-      device::mojom::VRSubmitFrameClientRequest,
-      device::mojom::VRPresentationProviderPtr,
-      device::mojom::VRDisplayFrameTransportOptionsPtr) override;
+  void SendRequestPresentReply(device::mojom::XRSessionPtr) override;
   void DialogSurfaceCreated(jobject surface,
                             gl::SurfaceTexture* texture) override;
   void UpdateGamepadData(device::GvrGamepadData) override;
diff --git a/chrome/browser/android/vr/vr_shell.cc b/chrome/browser/android/vr/vr_shell.cc
index f6ac330f..8da7fd7 100644
--- a/chrome/browser/android/vr/vr_shell.cc
+++ b/chrome/browser/android/vr/vr_shell.cc
@@ -92,9 +92,6 @@
 constexpr base::TimeDelta kPollCapturingStateInterval =
     base::TimeDelta::FromSecondsD(0.2);
 
-constexpr base::TimeDelta kExitVrDueToUnsupportedModeDelay =
-    base::TimeDelta::FromSeconds(5);
-
 constexpr base::TimeDelta kAssetsComponentWaitDelay =
     base::TimeDelta::FromSeconds(2);
 
@@ -754,17 +751,8 @@
   delegate_provider_->SetDelegate(this, viewer_type);
 }
 
-void VrShell::SendRequestPresentReply(
-    bool success,
-    device::mojom::VRSubmitFrameClientRequest request,
-    device::mojom::VRPresentationProviderPtrInfo provider_info,
-    device::mojom::VRDisplayFrameTransportOptionsPtr transport_options) {
-  device::mojom::VRPresentationProviderPtr provider;
-  provider.Bind(std::move(provider_info));
-
-  delegate_provider_->SendRequestPresentReply(success, std::move(request),
-                                              std::move(provider),
-                                              std::move(transport_options));
+void VrShell::SendRequestPresentReply(device::mojom::XRSessionPtr session) {
+  delegate_provider_->SendRequestPresentReply(std::move(session));
 }
 
 void VrShell::BufferBoundsChanged(JNIEnv* env,
@@ -866,28 +854,15 @@
                             UiUnsupportedMode::kCount);
 }
 
-void VrShell::ExitVrDueToUnsupportedMode(UiUnsupportedMode mode) {
-  ui_->SetIsExiting();
-  PostToGlThread(FROM_HERE, base::BindOnce(&VrShellGl::set_is_exiting,
-                                           gl_thread_->GetVrShellGl(), true));
-  main_thread_task_runner_->PostDelayedTask(
-      FROM_HERE,
-      base::BindOnce(&VrShell::ForceExitVr, weak_ptr_factory_.GetWeakPtr()),
-      kExitVrDueToUnsupportedModeDelay);
-  LogUnsupportedModeUserMetric(mode);
-}
-
 content::WebContents* VrShell::GetNonNativePageWebContents() const {
   return !web_contents_is_native_page_ ? web_contents_ : nullptr;
 }
 
 void VrShell::OnUnsupportedMode(UiUnsupportedMode mode) {
   switch (mode) {
-    case UiUnsupportedMode::kUnhandledCodePoint:  // Fall through.
-    case UiUnsupportedMode::kUnhandledCertificateInfo:
-    case UiUnsupportedMode::kUnhandledConnectionSecurityInfo:
-    case UiUnsupportedMode::kGenericUnsupportedFeature:
-      ExitVrDueToUnsupportedMode(mode);
+    case UiUnsupportedMode::kUnhandledCodePoint:
+      // We should never have this case.
+      CHECK(false);
       return;
     case UiUnsupportedMode::kVoiceSearchNeedsRecordAudioOsPermission: {
       JNIEnv* env = base::android::AttachCurrentThread();
@@ -899,20 +874,21 @@
       Java_VrShellImpl_onNeedsKeyboardUpdate(env, j_vr_shell_);
       return;
     }
-    // Is not sent by the UI anymore. Enum value still exists to show correct
-    // exit prompt if vr-browsing-native-android-ui flag is false.
+    // These modes are not sent by the UI anymore. Enum values still exist to
+    // show correct exit prompt if vr-browsing-native-android-ui flag is false.
     case UiUnsupportedMode::kUnhandledPageInfo:
+    case UiUnsupportedMode::kUnhandledCertificateInfo:
+    case UiUnsupportedMode::kUnhandledConnectionSecurityInfo:
+    case UiUnsupportedMode::kGenericUnsupportedFeature:
     // kSearchEnginePromo should directly DOFF without showing a promo. So it
     // should never be used from VR ui thread.
     case UiUnsupportedMode::kSearchEnginePromo:
-    // Should never be used as a mode.
     case UiUnsupportedMode::kCount:
       NOTREACHED();
       return;
   }
 
   NOTREACHED();
-  ExitVrDueToUnsupportedMode(mode);
 }
 
 void VrShell::OnExitVrPromptResult(UiUnsupportedMode reason,
diff --git a/chrome/browser/android/vr/vr_shell.h b/chrome/browser/android/vr/vr_shell.h
index da2d1e9..fb7a271 100644
--- a/chrome/browser/android/vr/vr_shell.h
+++ b/chrome/browser/android/vr/vr_shell.h
@@ -181,11 +181,7 @@
   void ContentOverlaySurfaceCreated(jobject surface,
                                     gl::SurfaceTexture* texture);
   void GvrDelegateReady(gvr::ViewerType viewer_type);
-  void SendRequestPresentReply(
-      bool success,
-      device::mojom::VRSubmitFrameClientRequest,
-      device::mojom::VRPresentationProviderPtrInfo,
-      device::mojom::VRDisplayFrameTransportOptionsPtr);
+  void SendRequestPresentReply(device::mojom::XRSessionPtr);
 
   void DialogSurfaceCreated(jobject surface, gl::SurfaceTexture* texture);
 
@@ -302,8 +298,6 @@
 
   bool HasDaydreamSupport(JNIEnv* env);
 
-  void ExitVrDueToUnsupportedMode(UiUnsupportedMode mode);
-
   content::WebContents* GetNonNativePageWebContents() const;
 
   void LoadAssets();
diff --git a/chrome/browser/android/vr/vr_shell_delegate.cc b/chrome/browser/android/vr/vr_shell_delegate.cc
index f843dd95..f754485 100644
--- a/chrome/browser/android/vr/vr_shell_delegate.cc
+++ b/chrome/browser/android/vr/vr_shell_delegate.cc
@@ -187,7 +187,7 @@
 
   if (!vr_shell_) {
     // We have to wait until the GL thread is ready since we have to get the
-    // VRSubmitFrameClient.
+    // XRPresentationClient.
     pending_successful_present_request_ = true;
     on_present_result_callback_ = base::BindOnce(
         &VrShellDelegate::OnPresentResult, base::Unretained(this),
@@ -211,30 +211,15 @@
 }
 
 void VrShellDelegate::SendRequestPresentReply(
-    bool success,
-    device::mojom::VRSubmitFrameClientRequest request,
-    device::mojom::VRPresentationProviderPtr provider,
-    device::mojom::VRDisplayFrameTransportOptionsPtr transport_options) {
+    device::mojom::XRSessionPtr session) {
   DVLOG(1) << __FUNCTION__;
   if (!request_present_response_callback_) {
     DLOG(ERROR) << __FUNCTION__ << ": ERROR: no callback";
     return;
   }
 
-  if (success) {
-    auto connection = device::mojom::XRPresentationConnection::New();
-    connection->client_request = std::move(request);
-    connection->provider = provider.PassInterface();
-    connection->transport_options = std::move(transport_options);
-
-    device::mojom::XRSessionPtr xr_session = device::mojom::XRSession::New();
-    xr_session->connection = std::move(connection);
-
-    base::ResetAndReturn(&request_present_response_callback_)
-        .Run(std::move(xr_session));
-  } else {
-    base::ResetAndReturn(&request_present_response_callback_).Run(nullptr);
-  }
+  base::ResetAndReturn(&request_present_response_callback_)
+      .Run(std::move(session));
 }
 
 void VrShellDelegate::DisplayActivate(JNIEnv* env,
diff --git a/chrome/browser/android/vr/vr_shell_delegate.h b/chrome/browser/android/vr/vr_shell_delegate.h
index 859f616..b523ac0 100644
--- a/chrome/browser/android/vr/vr_shell_delegate.h
+++ b/chrome/browser/android/vr/vr_shell_delegate.h
@@ -68,11 +68,7 @@
 
   device::GvrDevice* GetDevice();
 
-  void SendRequestPresentReply(
-      bool success,
-      device::mojom::VRSubmitFrameClientRequest request,
-      device::mojom::VRPresentationProviderPtr provider,
-      device::mojom::VRDisplayFrameTransportOptionsPtr);
+  void SendRequestPresentReply(device::mojom::XRSessionPtr session);
 
   // device::GvrDelegateProvider implementation.
   void ExitWebVRPresent() override;
diff --git a/chrome/browser/android/vr/vr_shell_gl.cc b/chrome/browser/android/vr/vr_shell_gl.cc
index 9a1d110..e4c9811 100644
--- a/chrome/browser/android/vr/vr_shell_gl.cc
+++ b/chrome/browser/android/vr/vr_shell_gl.cc
@@ -338,7 +338,8 @@
       daydream_support_(daydream_support),
       content_paused_(pause_content),
       task_runner_(base::ThreadTaskRunnerHandle::Get()),
-      binding_(this),
+      presentation_binding_(this),
+      frame_data_binding_(this),
       browser_(browser_interface),
       vr_ui_fps_meter_(),
       webvr_fps_meter_(),
@@ -681,7 +682,8 @@
     DVLOG(1) << __FUNCTION__ << ": wrong frame index, got " << frame_index
              << ", expected " << animating_frame->index;
     mojo::ReportBadMessage("SubmitFrame called with wrong frame index");
-    binding_.Close();
+    presentation_binding_.Close();
+    frame_data_binding_.Close();
     return false;
   }
 
@@ -846,8 +848,10 @@
     device::mojom::XRDeviceRuntimeSessionOptionsPtr options) {
   ClosePresentationBindings();
 
-  device::mojom::VRPresentationProviderPtr provider;
-  binding_.Bind(mojo::MakeRequest(&provider));
+  device::mojom::XRPresentationProviderPtr presentation_provider;
+  presentation_binding_.Bind(mojo::MakeRequest(&presentation_provider));
+  device::mojom::XRFrameDataProviderPtr frame_data_provider;
+  frame_data_binding_.Bind(mojo::MakeRequest(&frame_data_provider));
 
   gfx::Size webvr_size(
       display_info->leftEye->renderWidth + display_info->rightEye->renderWidth,
@@ -857,7 +861,7 @@
 
   // Decide which transport mechanism we want to use. This sets
   // the webxr_use_* options as a side effect.
-  device::mojom::VRDisplayFrameTransportOptionsPtr transport_options =
+  device::mojom::XRPresentationTransportOptionsPtr transport_options =
       GetWebVrFrameTransportOptions(options);
 
   if (webxr_use_shared_buffer_draw_) {
@@ -877,9 +881,16 @@
 
   ScheduleOrCancelWebVrFrameTimeout();
 
-  browser_->SendRequestPresentReply(true, mojo::MakeRequest(&submit_client_),
-                                    std::move(provider),
-                                    std::move(transport_options));
+  auto submit_frame_sink = device::mojom::XRPresentationConnection::New();
+  submit_frame_sink->client_request = mojo::MakeRequest(&submit_client_);
+  submit_frame_sink->provider = presentation_provider.PassInterface();
+  submit_frame_sink->transport_options = std::move(transport_options);
+
+  auto session = device::mojom::XRSession::New();
+  session->data_provider = frame_data_provider.PassInterface();
+  session->submit_frame_sink = std::move(submit_frame_sink);
+
+  browser_->SendRequestPresentReply(std::move(session));
 }
 
 void VrShellGl::OnSwapContents(int new_content_id) {
@@ -1077,7 +1088,7 @@
   }
 }
 
-device::mojom::VRDisplayFrameTransportOptionsPtr
+device::mojom::XRPresentationTransportOptionsPtr
 VrShellGl::GetWebVrFrameTransportOptions(
     const device::mojom::XRDeviceRuntimeSessionOptionsPtr& options) {
   DVLOG(1) << __FUNCTION__;
@@ -1120,19 +1131,19 @@
   DVLOG(1) << __FUNCTION__ << ": render_path=" << static_cast<int>(render_path);
   MetricsUtilAndroid::LogXrRenderPathUsed(render_path);
 
-  device::mojom::VRDisplayFrameTransportOptionsPtr transport_options =
-      device::mojom::VRDisplayFrameTransportOptions::New();
+  device::mojom::XRPresentationTransportOptionsPtr transport_options =
+      device::mojom::XRPresentationTransportOptions::New();
   // Only set boolean options that we need. Default is false, and we should be
   // able to safely ignore ones that our implementation doesn't care about.
   transport_options->wait_for_transfer_notification = true;
   if (webxr_use_shared_buffer_draw_) {
     transport_options->transport_method =
-        device::mojom::VRDisplayFrameTransportMethod::DRAW_INTO_TEXTURE_MAILBOX;
+        device::mojom::XRPresentationTransportMethod::DRAW_INTO_TEXTURE_MAILBOX;
     DCHECK(webxr_use_gpu_fence_);
     transport_options->wait_for_gpu_fence = true;
   } else {
     transport_options->transport_method =
-        device::mojom::VRDisplayFrameTransportMethod::SUBMIT_AS_MAILBOX_HOLDER;
+        device::mojom::XRPresentationTransportMethod::SUBMIT_AS_MAILBOX_HOLDER;
     transport_options->wait_for_transfer_notification = true;
     if (webxr_use_gpu_fence_) {
       transport_options->wait_for_gpu_fence = true;
@@ -1244,13 +1255,6 @@
                                       const RenderInfo& render_info,
                                       base::TimeTicks current_time) {
   gfx::Vector3dF head_direction = GetForwardVector(render_info.head_pose);
-  if (is_exiting_) {
-    // When we're exiting, we don't show the reticle and the only input
-    // processing we do is to handle immediate exits.
-    SendImmediateExitRequestIfNecessary();
-    return;
-  }
-
   gfx::Vector3dF ergo_neutral_pose;
   if (!controller_->IsConnected()) {
     // No controller detected, set up a gaze cursor that tracks the
@@ -1323,19 +1327,6 @@
   ui_->OnControllerUpdated(controller_model, reticle_model);
 }
 
-void VrShellGl::SendImmediateExitRequestIfNecessary() {
-  gvr::ControllerButton buttons[] = {
-      gvr::kControllerButtonClick, gvr::kControllerButtonApp,
-      gvr::kControllerButtonHome,
-  };
-  for (size_t i = 0; i < arraysize(buttons); ++i) {
-    if (controller_->ButtonUpHappened(buttons[i]) ||
-        controller_->ButtonDownHappened(buttons[i])) {
-      browser_->ForceExitVr();
-    }
-  }
-}
-
 bool VrShellGl::ResizeForWebVR(int16_t frame_index) {
   // Process all pending_bounds_ changes targeted for before this frame, being
   // careful of wrapping frame indices.
@@ -2252,7 +2243,7 @@
 }
 
 void VrShellGl::GetFrameData(
-    device::mojom::VRPresentationProvider::GetFrameDataCallback callback) {
+    device::mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   TRACE_EVENT0("gpu", __FUNCTION__);
   if (!get_frame_data_callback_.is_null()) {
     DLOG(WARNING) << ": previous get_frame_data_callback_ was not used yet";
@@ -2286,14 +2277,16 @@
                                   const gfx::Size& source_size) {
   if (!ValidateRect(left_bounds) || !ValidateRect(right_bounds)) {
     mojo::ReportBadMessage("UpdateLayerBounds called with invalid bounds");
-    binding_.Close();
+    presentation_binding_.Close();
+    frame_data_binding_.Close();
     return;
   }
 
   if (frame_index >= 0 && !webxr_->HaveAnimatingFrame()) {
     // The optional UpdateLayerBounds call must happen before SubmitFrame.
     mojo::ReportBadMessage("UpdateLayerBounds called without animating frame");
-    binding_.Close();
+    presentation_binding_.Close();
+    frame_data_binding_.Close();
     return;
   }
 
@@ -2467,7 +2460,8 @@
     // the connection is closing.
     base::ResetAndReturn(&get_frame_data_callback_).Run(nullptr);
   }
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
 }
 
 device::mojom::XRInputSourceStatePtr VrShellGl::GetGazeInputSourceState() {
diff --git a/chrome/browser/android/vr/vr_shell_gl.h b/chrome/browser/android/vr/vr_shell_gl.h
index e227dcd..619a81e 100644
--- a/chrome/browser/android/vr/vr_shell_gl.h
+++ b/chrome/browser/android/vr/vr_shell_gl.h
@@ -264,7 +264,8 @@
 
 // This class manages all GLThread owned objects and GL rendering for VrShell.
 // It is not threadsafe and must only be used on the GL thread.
-class VrShellGl : public device::mojom::VRPresentationProvider {
+class VrShellGl : public device::mojom::XRPresentationProvider,
+                  public device::mojom::XRFrameDataProvider {
  public:
   VrShellGl(GlBrowserInterface* browser_interface,
             std::unique_ptr<UiInterface> ui,
@@ -309,8 +310,6 @@
       device::mojom::VRDisplayInfoPtr display_info,
       device::mojom::XRDeviceRuntimeSessionOptionsPtr options);
 
-  void set_is_exiting(bool exiting) { is_exiting_ = exiting; }
-
   void OnSwapContents(int new_content_id);
 
   void EnableAlertDialog(PlatformInputHandler* input_handler,
@@ -333,7 +332,7 @@
  private:
   void GvrInit(gvr_context* gvr_api);
 
-  device::mojom::VRDisplayFrameTransportOptionsPtr
+  device::mojom::XRPresentationTransportOptionsPtr
   GetWebVrFrameTransportOptions(
       const device::mojom::XRDeviceRuntimeSessionOptionsPtr&);
 
@@ -362,7 +361,6 @@
   void UpdateController(const RenderInfo& render_info,
                         base::TimeTicks current_time);
 
-  void SendImmediateExitRequestIfNecessary();
   void HandleControllerInput(const gfx::Point3F& laser_origin,
                              const RenderInfo& render_info,
                              base::TimeTicks current_time);
@@ -386,9 +384,11 @@
 
   bool IsSubmitFrameExpected(int16_t frame_index);
 
-  // VRPresentationProvider
-  void GetFrameData(device::mojom::VRPresentationProvider::GetFrameDataCallback
+  // XRFrameDataProvider
+  void GetFrameData(device::mojom::XRFrameDataProvider::GetFrameDataCallback
                         callback) override;
+
+  // XRPresentationProvider
   void SubmitFrameMissing(int16_t frame_index, const gpu::SyncToken&) override;
   void SubmitFrame(int16_t frame_index,
                    const gpu::MailboxHolder& mailbox,
@@ -530,7 +530,6 @@
   bool paused_ = true;
   const bool surfaceless_rendering_;
   bool daydream_support_;
-  bool is_exiting_ = false;
   bool content_paused_;
   bool cardboard_trigger_pressed_ = false;
   bool cardboard_trigger_clicked_ = false;
@@ -550,11 +549,12 @@
   // updated in OnVSync and used as the rAF animation timer in SendVSync.
   base::TimeTicks pending_time_;
   bool pending_vsync_ = false;
-  device::mojom::VRPresentationProvider::GetFrameDataCallback
+  device::mojom::XRFrameDataProvider::GetFrameDataCallback
       get_frame_data_callback_;
 
-  mojo::Binding<device::mojom::VRPresentationProvider> binding_;
-  device::mojom::VRSubmitFrameClientPtr submit_client_;
+  mojo::Binding<device::mojom::XRPresentationProvider> presentation_binding_;
+  mojo::Binding<device::mojom::XRFrameDataProvider> frame_data_binding_;
+  device::mojom::XRPresentationClientPtr submit_client_;
 
   GlBrowserInterface* browser_;
 
diff --git a/chrome/browser/browser_resources.grd b/chrome/browser/browser_resources.grd
index b0625b33..e22fae5 100644
--- a/chrome/browser/browser_resources.grd
+++ b/chrome/browser/browser_resources.grd
@@ -143,8 +143,8 @@
       <include name="IDR_DOWNLOAD_INTERNALS_JS" file="resources\download_internals\download_internals.js" type="BINDATA" compress="gzip" />
       <include name="IDR_DOWNLOAD_INTERNALS_BROWSER_PROXY_JS" file="resources\download_internals\download_internals_browser_proxy.js" type="BINDATA" compress="gzip" />
       <include name="IDR_DOWNLOAD_INTERNALS_VISUALS_JS" file="resources\download_internals\download_internals_visuals.js" type="BINDATA" compress="gzip" />
-      <include name="IDR_UKM_INTERNALS_HTML" file="resources/ukm/ukm_internals.html" flattenhtml="true" allowexternalscript="true" compress="gzip" type="BINDATA" />
-      <include name="IDR_UKM_INTERNALS_JS" file="resources/ukm/ukm_internals.js" flattenhtml="true" compress="gzip" type="BINDATA" />
+      <include name="IDR_UKM_INTERNALS_HTML" file="../../components/ukm/debug/ukm_internals.html" flattenhtml="true" allowexternalscript="true" compress="gzip" type="BINDATA" />
+      <include name="IDR_UKM_INTERNALS_JS" file="../../components/ukm/debug/ukm_internals.js" flattenhtml="true" compress="gzip" type="BINDATA" />
       <if expr="not is_android">
         <include name="IDR_MD_DOWNLOADS_1X_INCOGNITO_MARKER_PNG" file="resources\md_downloads\1x\incognito_marker.png" type="BINDATA" />
         <include name="IDR_MD_DOWNLOADS_2X_INCOGNITO_MARKER_PNG" file="resources\md_downloads\2x\incognito_marker.png" type="BINDATA" />
diff --git a/chrome/browser/extensions/api/socket/combined_socket_unittest.cc b/chrome/browser/extensions/api/socket/combined_socket_unittest.cc
deleted file mode 100644
index 6feb6fe9..0000000
--- a/chrome/browser/extensions/api/socket/combined_socket_unittest.cc
+++ /dev/null
@@ -1,145 +0,0 @@
-// Copyright 2016 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include <memory>
-
-#include "chrome/browser/extensions/api/socket/mock_tcp_client_socket.h"
-#include "extensions/browser/api/socket/socket.h"
-#include "extensions/browser/api/socket/tcp_socket.h"
-#include "extensions/browser/api/socket/tls_socket.h"
-#include "net/socket/stream_socket.h"
-#include "testing/gmock/include/gmock/gmock.h"
-#include "testing/gtest/include/gtest/gtest.h"
-
-namespace extensions {
-
-const int kBufferLength = 10;
-
-template <typename T>
-std::unique_ptr<T> CreateTestSocket(
-    std::unique_ptr<MockTCPClientSocket> stream);
-
-template <>
-std::unique_ptr<TCPSocket> CreateTestSocket(
-    std::unique_ptr<MockTCPClientSocket> stream) {
-  return std::make_unique<TCPSocket>(std::move(stream), "fake id",
-                                     true /* is_connected */);
-}
-
-template <>
-std::unique_ptr<TLSSocket> CreateTestSocket(
-    std::unique_ptr<MockTCPClientSocket> stream) {
-  return std::make_unique<TLSSocket>(std::move(stream), "fake id");
-}
-
-class CombinedSocketTest : public testing::Test {
- public:
-  CombinedSocketTest() : count_(0), io_buffer_(nullptr) {}
-
-  // Strict test for synchronous (immediate) read behavior
-  template <typename T>
-  void TestRead() {
-    net::IOBuffer* buffer = nullptr;
-
-    std::unique_ptr<MockTCPClientSocket> stream(
-        new testing::StrictMock<MockTCPClientSocket>());
-    EXPECT_CALL(*stream, Read(testing::NotNull(), kBufferLength, testing::_))
-        .WillOnce(DoAll(testing::SaveArg<0>(&buffer),
-                        testing::Return(kBufferLength)));
-    EXPECT_CALL(*stream, Disconnect());
-
-    std::unique_ptr<T> socket = CreateTestSocket<T>(std::move(stream));
-    ReadCompletionCallback read_callback =
-        base::Bind(&CombinedSocketTest::OnRead, base::Unretained(this));
-    socket->Read(kBufferLength, read_callback);
-    EXPECT_EQ(kBufferLength, count_);
-    EXPECT_NE(nullptr, buffer);
-    EXPECT_EQ(buffer, io_buffer_);
-  }
-
-  // Strict test for async read behavior (read returns PENDING)
-  template <typename T>
-  void TestReadPending() {
-    net::IOBuffer* buffer = nullptr;
-    net::CompletionCallback socket_cb;
-
-    std::unique_ptr<MockTCPClientSocket> stream(
-        new testing::StrictMock<MockTCPClientSocket>());
-    EXPECT_CALL(*stream, Read(testing::NotNull(), kBufferLength, testing::_))
-        .WillOnce(DoAll(testing::SaveArg<0>(&buffer),
-                        testing::SaveArg<2>(&socket_cb),
-                        testing::Return(net::ERR_IO_PENDING)));
-    EXPECT_CALL(*stream, Disconnect());
-
-    std::unique_ptr<T> socket = CreateTestSocket<T>(std::move(stream));
-    ReadCompletionCallback read_callback =
-        base::Bind(&CombinedSocketTest::OnRead, base::Unretained(this));
-    socket->Read(kBufferLength, read_callback);
-    EXPECT_EQ(0, count_);
-    EXPECT_EQ(nullptr, io_buffer_);
-    socket_cb.Run(kBufferLength);
-    EXPECT_EQ(kBufferLength, count_);
-    EXPECT_NE(nullptr, buffer);
-    EXPECT_EQ(buffer, io_buffer_);
-  }
-
-  // Even if the socket is closed, it may still have data left to read.
-  template <typename T>
-  void TestReadAfterDisconnect() {
-    net::IOBuffer* buffer = nullptr;
-
-    std::unique_ptr<MockTCPClientSocket> stream(
-        new testing::NiceMock<MockTCPClientSocket>());
-    EXPECT_CALL(*stream, Read(testing::NotNull(), kBufferLength, testing::_))
-        .WillOnce(DoAll(testing::SaveArg<0>(&buffer),
-                        testing::Return(kBufferLength)));
-    ON_CALL(*stream, IsConnected()).WillByDefault(testing::Return(false));
-    EXPECT_CALL(*stream, Disconnect());
-
-    std::unique_ptr<T> socket = CreateTestSocket<T>(std::move(stream));
-    ReadCompletionCallback read_callback =
-        base::Bind(&CombinedSocketTest::OnRead, base::Unretained(this));
-    socket->Read(kBufferLength, read_callback);
-    EXPECT_EQ(kBufferLength, count_);
-    EXPECT_NE(nullptr, buffer);
-    EXPECT_EQ(buffer, io_buffer_);
-  }
-
-  void OnRead(int count,
-              scoped_refptr<net::IOBuffer> io_buffer,
-              bool socket_destroying) {
-    count_ = count;
-    io_buffer_ = io_buffer.get();
-  }
-
- protected:
-  int count_;
-  net::IOBuffer* io_buffer_;
-};
-
-TEST_F(CombinedSocketTest, TlsRead) {
-  TestRead<TLSSocket>();
-}
-
-TEST_F(CombinedSocketTest, TcpRead) {
-  TestRead<TCPSocket>();
-}
-
-TEST_F(CombinedSocketTest, TlsReadPending) {
-  TestReadPending<TLSSocket>();
-}
-
-TEST_F(CombinedSocketTest, TcpReadPending) {
-  TestReadPending<TCPSocket>();
-}
-
-TEST_F(CombinedSocketTest, TlsReadAfterDisconnect) {
-  TestReadAfterDisconnect<TLSSocket>();
-}
-
-TEST_F(CombinedSocketTest, TcpReadAfterDisconnect) {
-  TestReadAfterDisconnect<TCPSocket>();
-}
-
-}  //  namespace extensions
diff --git a/chrome/browser/extensions/api/socket/mock_tcp_client_socket.h b/chrome/browser/extensions/api/socket/mock_tcp_client_socket.h
deleted file mode 100644
index fddad10..0000000
--- a/chrome/browser/extensions/api/socket/mock_tcp_client_socket.h
+++ /dev/null
@@ -1,82 +0,0 @@
-// Copyright 2016 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_EXTENSIONS_API_SOCKET_MOCK_TCP_CLIENT_SOCKET_H_
-#define CHROME_BROWSER_EXTENSIONS_API_SOCKET_MOCK_TCP_CLIENT_SOCKET_H_
-
-#include "base/callback_helpers.h"
-#include "net/log/net_log_source.h"
-#include "net/log/net_log_with_source.h"
-#include "net/socket/tcp_client_socket.h"
-#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
-#include "testing/gmock/include/gmock/gmock.h"
-
-namespace extensions {
-class MockTCPClientSocket : public net::TCPClientSocket {
- public:
-  MockTCPClientSocket();
-  ~MockTCPClientSocket() override;
-
-  int Read(net::IOBuffer* buffer,
-           int bytes,
-           net::CompletionOnceCallback callback) override {
-    return Read(buffer, bytes,
-                base::AdaptCallbackForRepeating(std::move(callback)));
-  }
-
-  int Write(net::IOBuffer* buffer,
-            int bytes,
-            net::CompletionOnceCallback callback,
-            const net::NetworkTrafficAnnotationTag& tag) override {
-    return Write(buffer, bytes,
-                 base::AdaptCallbackForRepeating(std::move(callback)), tag);
-  }
-
-  int Connect(net::CompletionOnceCallback callback) override {
-    return Connect(base::AdaptCallbackForRepeating(std::move(callback)));
-  }
-
-  MOCK_METHOD3(Read, int(net::IOBuffer*, int, const net::CompletionCallback&));
-  MOCK_METHOD4(Write,
-               int(net::IOBuffer*,
-                   int,
-                   const net::CompletionCallback&,
-                   const net::NetworkTrafficAnnotationTag&));
-  MOCK_METHOD1(SetReceiveBufferSize, int(int32_t));
-  MOCK_METHOD1(SetSendBufferSize, int(int32_t));
-  MOCK_METHOD1(Connect, int(const net::CompletionCallback&));
-  MOCK_METHOD0(Disconnect, void());
-  MOCK_CONST_METHOD0(IsConnected, bool());
-  MOCK_CONST_METHOD0(IsConnectedAndIdle, bool());
-  MOCK_CONST_METHOD1(GetPeerAddress, int(net::IPEndPoint*));
-  MOCK_CONST_METHOD1(GetLocalAddress, int(net::IPEndPoint*));
-  MOCK_CONST_METHOD0(NetLog, const net::NetLogWithSource&());
-  MOCK_CONST_METHOD0(WasEverUsed, bool());
-  MOCK_CONST_METHOD0(UsingTCPFastOpen, bool());
-  MOCK_CONST_METHOD0(NumBytesRead, int64_t());
-  MOCK_CONST_METHOD0(GetConnectTimeMicros, base::TimeDelta());
-  MOCK_CONST_METHOD0(WasAlpnNegotiated, bool());
-  MOCK_CONST_METHOD0(GetNegotiatedProtocol, net::NextProto());
-  MOCK_METHOD1(GetSSLInfo, bool(net::SSLInfo*));
-  MOCK_CONST_METHOD1(GetConnectionAttempts, void(net::ConnectionAttempts*));
-  MOCK_METHOD0(ClearConnectionAttempts, void());
-  MOCK_METHOD1(AddConnectionAttempts, void(const net::ConnectionAttempts&));
-  MOCK_CONST_METHOD0(GetTotalReceivedBytes, int64_t());
-
-  // Methods specific to MockTCPClientSocket
-  MOCK_METHOD1(Bind, int(const net::IPEndPoint&));
-  MOCK_METHOD2(SetKeepAlive, bool(bool, int));
-  MOCK_METHOD1(SetNoDelay, bool(bool));
-};
-
-MockTCPClientSocket::MockTCPClientSocket()
-    : TCPClientSocket(net::AddressList(),
-                      nullptr,
-                      nullptr,
-                      net::NetLogSource()) {}
-MockTCPClientSocket::~MockTCPClientSocket() {}
-
-}  // namespace extensions
-
-#endif  // CHROME_BROWSER_EXTENSIONS_API_SOCKET_MOCK_TCP_CLIENT_SOCKET_H_
diff --git a/chrome/browser/extensions/api/socket/tcp_socket_unittest.cc b/chrome/browser/extensions/api/socket/tcp_socket_unittest.cc
index 6351e3d..90c7937 100644
--- a/chrome/browser/extensions/api/socket/tcp_socket_unittest.cc
+++ b/chrome/browser/extensions/api/socket/tcp_socket_unittest.cc
@@ -4,274 +4,717 @@
 
 #include <memory>
 
-#include "base/callback_helpers.h"
 #include "base/macros.h"
+#include "base/memory/scoped_refptr.h"
+#include "base/test/bind_test_util.h"
+#include "chrome/browser/extensions/extension_service_test_base.h"
+#include "chrome/test/base/testing_profile.h"
+#include "content/public/browser/browser_context.h"
+#include "content/public/test/test_storage_partition.h"
 #include "extensions/browser/api/socket/tcp_socket.h"
 #include "net/base/address_list.h"
 #include "net/base/completion_callback.h"
 #include "net/base/io_buffer.h"
 #include "net/base/net_errors.h"
-#include "net/base/rand_callback.h"
-#include "net/log/net_log_source.h"
-#include "net/socket/tcp_client_socket.h"
-#include "net/socket/tcp_server_socket.h"
+#include "net/base/test_completion_callback.h"
+#include "net/socket/socket_test_util.h"
 #include "net/traffic_annotation/network_traffic_annotation.h"
-#include "testing/gmock/include/gmock/gmock.h"
-
-using testing::_;
-using testing::DoAll;
-using testing::Return;
-using testing::SaveArg;
+#include "net/url_request/url_request_test_util.h"
+#include "services/network/network_context.h"
+#include "services/network/public/mojom/network_context.mojom.h"
 
 namespace extensions {
 
-class MockTCPSocket : public net::TCPClientSocket {
- public:
-  explicit MockTCPSocket(const net::AddressList& address_list)
-      : net::TCPClientSocket(address_list, NULL, NULL, net::NetLogSource()) {}
+namespace {
 
-  int Read(net::IOBuffer* buffer,
-           int bytes,
-           net::CompletionOnceCallback callback) override {
-    return Read(buffer, bytes,
-                base::AdaptCallbackForRepeating(std::move(callback)));
-  }
-
-  int Write(net::IOBuffer* buffer,
-            int bytes,
-            net::CompletionOnceCallback callback,
-            const net::NetworkTrafficAnnotationTag& tag) override {
-    return Write(buffer, bytes,
-                 base::AdaptCallbackForRepeating(std::move(callback)), tag);
-  }
-
-  MOCK_METHOD3(Read, int(net::IOBuffer* buf, int buf_len,
-                         const net::CompletionCallback& callback));
-  MOCK_METHOD4(Write,
-               int(net::IOBuffer* buf,
-                   int buf_len,
-                   const net::CompletionCallback& callback,
-                   const net::NetworkTrafficAnnotationTag&));
-  MOCK_METHOD2(SetKeepAlive, bool(bool enable, int delay));
-  MOCK_METHOD1(SetNoDelay, bool(bool no_delay));
-  bool IsConnected() const override {
-    return true;
-  }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockTCPSocket);
-};
-
-class MockTCPServerSocket : public net::TCPServerSocket {
- public:
-  MockTCPServerSocket() : net::TCPServerSocket(NULL, net::NetLogSource()) {}
-  MOCK_METHOD2(Listen, int(const net::IPEndPoint& address, int backlog));
-  MOCK_METHOD2(Accept,
-               int(std::unique_ptr<net::StreamSocket>* socket,
-                   const net::CompletionCallback& callback));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockTCPServerSocket);
-};
-
-class CompleteHandler {
- public:
-  CompleteHandler() {}
-  MOCK_METHOD1(OnComplete, void(int result_code));
-  MOCK_METHOD3(OnReadComplete,
-               void(int result_code,
-                    scoped_refptr<net::IOBuffer> io_buffer,
-                    bool socket_destroying));
-
-  // MOCK_METHOD cannot mock a scoped_ptr argument.
-  MOCK_METHOD2(OnAcceptMock, void(int, net::TCPClientSocket*));
-  void OnAccept(int count, std::unique_ptr<net::TCPClientSocket> socket) {
-    OnAcceptMock(count, socket.get());
-  }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(CompleteHandler);
-};
+const char kTestMsg[] = "abcdefghij";
+const int kTestMsgLength = strlen(kTestMsg);
 
 const char FAKE_ID[] = "abcdefghijklmnopqrst";
 
-TEST(SocketTest, TestTCPSocketRead) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
-  CompleteHandler handler;
+class TCPSocketUnitTestBase : public extensions::ExtensionServiceTestBase {
+ public:
+  TCPSocketUnitTestBase()
+      : url_request_context_(true /* delay_initialization */) {}
+  ~TCPSocketUnitTestBase() override {}
 
-  EXPECT_CALL(*tcp_client_socket, Read(_, _, _))
-      .Times(1);
-  EXPECT_CALL(handler, OnReadComplete(_, _, _)).Times(1);
+  std::unique_ptr<TCPSocket> CreateSocket() {
+    auto socket = std::make_unique<TCPSocket>(&profile_, FAKE_ID);
+    socket->SetStoragePartitionForTest(&partition_);
+    return socket;
+  }
 
-  std::unique_ptr<TCPSocket> socket(TCPSocket::CreateSocketForTesting(
-      std::move(tcp_client_socket), FAKE_ID, true));
+  std::unique_ptr<TCPSocket> CreateAndConnectSocketWithAddress(
+      const net::IPEndPoint& ip_end_point) {
+    auto socket = CreateSocket();
+    net::AddressList address(ip_end_point);
+    net::TestCompletionCallback callback;
+    socket->Connect(address, callback.callback());
+    EXPECT_EQ(net::OK, callback.WaitForResult());
+    return socket;
+  }
+
+  std::unique_ptr<TCPSocket> CreateAndConnectSocket() {
+    net::IPEndPoint ip_end_point(net::IPAddress::IPv4Localhost(), 1234);
+    return CreateAndConnectSocketWithAddress(ip_end_point);
+  }
+
+  // Reads data from |socket| and compares it with |expected_data|.
+  void ReadData(Socket* socket, const std::string& expected_data) {
+    std::string received_data;
+    const int count = 512;
+    while (true) {
+      base::RunLoop run_loop;
+      int net_error = net::ERR_FAILED;
+      socket->Read(count,
+                   base::BindLambdaForTesting(
+                       [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                           bool socket_destroying) {
+                         net_error = result;
+                         EXPECT_FALSE(socket_destroying);
+                         if (result > 0)
+                           received_data.append(io_buffer->data(), result);
+                         run_loop.Quit();
+                       }));
+      run_loop.Run();
+      if (net_error <= 0)
+        break;
+    }
+    EXPECT_EQ(expected_data, received_data);
+  }
+
+ protected:
+  // extensions::ExtensionServiceTestBase implementation.
+  void SetUp() override { InitializeEmptyExtensionService(); }
+
+  void Initialize() {
+    url_request_context_.Init();
+    network_context_ = std::make_unique<network::NetworkContext>(
+        nullptr, mojo::MakeRequest(&network_context_ptr_),
+        &url_request_context_);
+    partition_.set_network_context(network_context_ptr_.get());
+  }
+
+  net::TestURLRequestContext url_request_context_;
+
+ private:
+  TestingProfile profile_;
+  content::TestStoragePartition partition_;
+  std::unique_ptr<network::NetworkContext> network_context_;
+  network::mojom::NetworkContextPtr network_context_ptr_;
+};
+
+}  // namespace
+
+class TCPSocketUnitTest : public TCPSocketUnitTestBase,
+                          public ::testing::WithParamInterface<net::IoMode> {
+ public:
+  TCPSocketUnitTest() : TCPSocketUnitTestBase() {
+    mock_client_socket_factory_.set_enable_read_if_ready(true);
+    url_request_context_.set_client_socket_factory(
+        &mock_client_socket_factory_);
+    Initialize();
+  }
+  ~TCPSocketUnitTest() override {}
+
+  net::MockClientSocketFactory* mock_client_socket_factory() {
+    return &mock_client_socket_factory_;
+  }
+
+ private:
+  net::MockClientSocketFactory mock_client_socket_factory_;
+};
+
+INSTANTIATE_TEST_CASE_P(/* no prefix */,
+                        TCPSocketUnitTest,
+                        testing::Values(net::SYNCHRONOUS, net::ASYNC));
+
+TEST_F(TCPSocketUnitTest, SocketConnectError) {
+  net::IPEndPoint ip_end_point(net::IPAddress::IPv4Localhost(), 1234);
+  net::StaticSocketDataProvider data_provider((base::span<net::MockRead>()),
+                                              base::span<net::MockWrite>());
+  data_provider.set_connect_data(
+      net::MockConnect(net::ASYNC, net::ERR_FAILED, ip_end_point));
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateSocket();
+
+  net::AddressList address(ip_end_point);
+  net::TestCompletionCallback callback;
+  socket->Connect(address, callback.callback());
+  EXPECT_EQ(net::ERR_FAILED, callback.WaitForResult());
+}
+
+TEST_P(TCPSocketUnitTest, SocketConnectAfterDisconnect) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {net::MockRead(io_mode, net::OK)};
+  net::StaticSocketDataProvider data_provider1(kReads,
+                                               base::span<net::MockWrite>());
+  net::StaticSocketDataProvider data_provider2(kReads,
+                                               base::span<net::MockWrite>());
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider1);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider2);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+  socket->Disconnect(false /* socket_destroying */);
+  net::TestCompletionCallback callback2;
+  net::IPEndPoint ip_end_point(net::IPAddress::IPv4Localhost(), 1234);
+  net::AddressList address(ip_end_point);
+  socket->Connect(address, callback2.callback());
+  EXPECT_EQ(net::OK, callback2.WaitForResult());
+
+  EXPECT_TRUE(data_provider1.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider1.AllWriteDataConsumed());
+  EXPECT_TRUE(data_provider2.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider2.AllWriteDataConsumed());
+}
+
+TEST_F(TCPSocketUnitTest, DestroyWhileReadPending) {
+  const net::MockRead kReads[] = {
+      net::MockRead(net::SYNCHRONOUS, net::ERR_IO_PENDING)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  int net_result = net::ERR_FAILED;
+  base::RunLoop run_loop;
+  int count = 1;
+  // Read one byte, and it should be pending.
+  socket->Read(count,
+               base::BindLambdaForTesting(
+                   [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                       bool socket_destroying) {
+                     net_result = result;
+                     // |socket_destroying| should correctly denote that this
+                     // read callback is invoked through the destructor of
+                     // TCPSocket.
+                     EXPECT_TRUE(socket_destroying);
+                     run_loop.Quit();
+                   }));
+  // Destroy socket.
+  socket = nullptr;
+  // Wait for read callback.
+  run_loop.Run();
+  EXPECT_EQ(net::ERR_CONNECTION_CLOSED, net_result);
+}
+
+TEST_P(TCPSocketUnitTest, Read) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(io_mode, kTestMsg, kTestMsgLength),
+      net::MockRead(io_mode, net::OK)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  ReadData(socket.get(), kTestMsg);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+// Tests the case where a message is split over two separate socket reads.
+TEST_P(TCPSocketUnitTest, SocketMultipleRead) {
+  const char kFirstHalfTestMsg[] = "abcde";
+  const char kSecondHalfTestMsg[] = "fghij";
+  EXPECT_EQ(kTestMsg, std::string(kFirstHalfTestMsg) + kSecondHalfTestMsg);
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(io_mode, kFirstHalfTestMsg, strlen(kFirstHalfTestMsg)),
+      net::MockRead(io_mode, kSecondHalfTestMsg, strlen(kSecondHalfTestMsg)),
+      net::MockRead(io_mode, net::OK)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  ReadData(socket.get(), kTestMsg);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+// Tests the case where read size is smaller than the actual message.
+TEST_P(TCPSocketUnitTest, SocketPartialRead) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(io_mode, kTestMsg, kTestMsgLength),
+      net::MockRead(io_mode, net::OK)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  int count = 1;
+  std::string received_data;
+  while (true) {
+    int net_result = net::ERR_FAILED;
+    base::RunLoop run_loop;
+    socket->Read(count,
+                 base::BindLambdaForTesting(
+                     [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                         bool socket_destroying) {
+                       net_result = result;
+                       EXPECT_FALSE(socket_destroying);
+                       if (result > 0)
+                         received_data.append(io_buffer->data(), result);
+                       run_loop.Quit();
+                     }));
+    run_loop.Run();
+    if (net_result <= 0)
+      break;
+    // Double the read size in the next iteration.
+    count *= 2;
+  }
+  EXPECT_EQ(kTestMsg, received_data);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+TEST_P(TCPSocketUnitTest, ReadError) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(io_mode, kTestMsg, kTestMsgLength),
+      net::MockRead(io_mode, net::ERR_INSUFFICIENT_RESOURCES)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
 
   const int count = 512;
-  socket->Read(count, base::Bind(&CompleteHandler::OnReadComplete,
-        base::Unretained(&handler)));
-}
-
-TEST(SocketTest, TestTCPSocketWrite) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
-  CompleteHandler handler;
-
-  net::CompletionCallback callback;
-  EXPECT_CALL(*tcp_client_socket, Write(_, _, _, _))
-      .Times(2)
-      .WillRepeatedly(testing::DoAll(SaveArg<2>(&callback), Return(128)));
-  EXPECT_CALL(handler, OnComplete(_))
-      .Times(1);
-
-  std::unique_ptr<TCPSocket> socket(TCPSocket::CreateSocketForTesting(
-      std::move(tcp_client_socket), FAKE_ID, true));
-
-  scoped_refptr<net::IOBufferWithSize> io_buffer(
-      new net::IOBufferWithSize(256));
-  socket->Write(io_buffer.get(), io_buffer->size(),
-      base::Bind(&CompleteHandler::OnComplete, base::Unretained(&handler)));
-}
-
-TEST(SocketTest, TestTCPSocketBlockedWrite) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
-  CompleteHandler handler;
-
-  net::CompletionCallback callback;
-  EXPECT_CALL(*tcp_client_socket, Write(_, _, _, _))
-      .Times(2)
-      .WillRepeatedly(
-          testing::DoAll(SaveArg<2>(&callback), Return(net::ERR_IO_PENDING)));
-
-  std::unique_ptr<TCPSocket> socket(TCPSocket::CreateSocketForTesting(
-      std::move(tcp_client_socket), FAKE_ID, true));
-
-  scoped_refptr<net::IOBufferWithSize> io_buffer(new net::IOBufferWithSize(42));
-  socket->Write(io_buffer.get(), io_buffer->size(),
-      base::Bind(&CompleteHandler::OnComplete, base::Unretained(&handler)));
-
-  // Good. Original call came back unable to complete. Now pretend the socket
-  // finished, and confirm that we passed the error back.
-  EXPECT_CALL(handler, OnComplete(42))
-      .Times(1);
-  callback.Run(40);
-  callback.Run(2);
-}
-
-TEST(SocketTest, TestTCPSocketBlockedWriteReentry) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
-  CompleteHandler handlers[5];
-
-  net::CompletionCallback callback;
-  EXPECT_CALL(*tcp_client_socket, Write(_, _, _, _))
-      .Times(5)
-      .WillRepeatedly(
-          testing::DoAll(SaveArg<2>(&callback), Return(net::ERR_IO_PENDING)));
-
-  std::unique_ptr<TCPSocket> socket(TCPSocket::CreateSocketForTesting(
-      std::move(tcp_client_socket), FAKE_ID, true));
-
-  scoped_refptr<net::IOBufferWithSize> io_buffers[5];
-  int i;
-  for (i = 0; i < 5; i++) {
-    io_buffers[i] = new net::IOBufferWithSize(128 + i * 50);
-    scoped_refptr<net::IOBufferWithSize> io_buffer1(
-        new net::IOBufferWithSize(42));
-    socket->Write(io_buffers[i].get(), io_buffers[i]->size(),
-        base::Bind(&CompleteHandler::OnComplete,
-            base::Unretained(&handlers[i])));
-
-    EXPECT_CALL(handlers[i], OnComplete(io_buffers[i]->size()))
-        .Times(1);
+  int net_error = net::OK;
+  while (true) {
+    base::RunLoop run_loop;
+    socket->Read(count,
+                 base::BindLambdaForTesting(
+                     [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                         bool socket_destroying) {
+                       net_error = result;
+                       EXPECT_FALSE(socket_destroying);
+                       if (result <= 0) {
+                         EXPECT_FALSE(socket->IsConnected());
+                         EXPECT_EQ(nullptr, io_buffer);
+                       } else {
+                         EXPECT_TRUE(socket->IsConnected());
+                       }
+                       run_loop.Quit();
+                     }));
+    run_loop.Run();
+    if (net_error < 0)
+      break;
   }
-
-  for (i = 0; i < 5; i++) {
-    callback.Run(128 + i * 50);
-  }
+  // Note that TCPSocket only detects that receive pipe is broken and propagates
+  // it as a net::ERR_FAILED. It doesn't know the specific net error code. To do
+  // that, it needs to register itself as a network::mojom::SocketObserver.
+  EXPECT_EQ(net::ERR_FAILED, net_error);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
 }
 
-TEST(SocketTest, TestTCPSocketSetNoDelay) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
+TEST_P(TCPSocketUnitTest, Write) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::SYNCHRONOUS, net::ERR_IO_PENDING)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, kTestMsg, kTestMsgLength)};
 
-  bool no_delay = false;
+  net::StaticSocketDataProvider data_provider(kReads, kWrites);
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  net::TestCompletionCallback write_callback;
+  socket->Write(io_buffer.get(), kTestMsgLength, write_callback.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback.WaitForResult());
+}
+
+// Tests the case where a message is split over two separate socket writes.
+TEST_P(TCPSocketUnitTest, MultipleWrite) {
+  const char kFirstHalfTestMsg[] = "abcde";
+  const char kSecondHalfTestMsg[] = "fghij";
+  EXPECT_EQ(kTestMsg, std::string(kFirstHalfTestMsg) + kSecondHalfTestMsg);
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::SYNCHRONOUS, net::ERR_IO_PENDING)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, kFirstHalfTestMsg, strlen(kFirstHalfTestMsg)),
+      net::MockWrite(io_mode, kSecondHalfTestMsg, strlen(kSecondHalfTestMsg))};
+
+  net::StaticSocketDataProvider data_provider(kReads, kWrites);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  int num_bytes_written = 0;
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  auto drainable_io_buffer = base::MakeRefCounted<net::DrainableIOBuffer>(
+      io_buffer.get(), kTestMsgLength);
+  while (num_bytes_written < kTestMsgLength) {
+    net::TestCompletionCallback write_callback;
+    socket->Write(drainable_io_buffer.get(), kTestMsgLength - num_bytes_written,
+                  write_callback.callback());
+    int result = write_callback.WaitForResult();
+    ASSERT_GT(result, net::OK);
+    drainable_io_buffer->DidConsume(result);
+    num_bytes_written += result;
+    // Flushes the write.
+    base::RunLoop().RunUntilIdle();
+  }
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+TEST_P(TCPSocketUnitTest, PartialWrite) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::SYNCHRONOUS, net::ERR_IO_PENDING)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, "a", 1), net::MockWrite(io_mode, "bc", 2),
+      net::MockWrite(io_mode, "defg", 4), net::MockWrite(io_mode, "hij", 3)};
+
+  net::StaticSocketDataProvider data_provider(kReads, kWrites);
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  // Start with writing one byte, and double that in the next iteration.
+  int num_bytes_to_write = 1;
+  int num_bytes_written = 0;
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  auto drainable_io_buffer = base::MakeRefCounted<net::DrainableIOBuffer>(
+      io_buffer.get(), kTestMsgLength);
+  while (num_bytes_written < kTestMsgLength) {
+    net::TestCompletionCallback write_callback;
+    socket->Write(
+        drainable_io_buffer.get(),
+        std::max(kTestMsgLength - num_bytes_written, num_bytes_to_write),
+        write_callback.callback());
+    int result = write_callback.WaitForResult();
+    ASSERT_GT(result, net::OK);
+    drainable_io_buffer->DidConsume(result);
+    num_bytes_written += result;
+    num_bytes_to_write *= 2;
+    // Flushes the write.
+    base::RunLoop().RunUntilIdle();
+  }
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+TEST_P(TCPSocketUnitTest, WriteError) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::SYNCHRONOUS, net::ERR_IO_PENDING)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, net::ERR_INSUFFICIENT_RESOURCES)};
+
+  net::StaticSocketDataProvider data_provider(kReads, kWrites);
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+
+  // Mojo data pipe might buffer some write data, so continue writing until the
+  // write error is received.
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  int net_error = net::OK;
+  while (true) {
+    base::RunLoop run_loop;
+    socket->Write(io_buffer.get(), kTestMsgLength,
+                  base::BindLambdaForTesting([&](int result) {
+                    EXPECT_EQ(result > 0, socket->IsConnected());
+                    net_error = result;
+                    run_loop.Quit();
+                  }));
+    run_loop.Run();
+    if (net_error <= 0)
+      break;
+  }
+  // Note that TCPSocket only detects that send pipe is broken and propagates
+  // it as a net::ERR_FAILED. It doesn't know the specific net error code. To do
+  // that, it needs to register itself as a network::mojom::SocketObserver.
+  EXPECT_EQ(net::ERR_FAILED, net_error);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+}
+
+namespace {
+
+class ExtensionsMockClientSocket : public net::MockTCPClientSocket {
+ public:
+  ExtensionsMockClientSocket(net::SocketDataProvider* provider, bool success)
+      : MockTCPClientSocket(
+            net::AddressList(
+                net::IPEndPoint(net::IPAddress::IPv4Localhost(), 1234)),
+            nullptr /* netlog */,
+            provider),
+        success_(success) {
+    this->set_enable_read_if_ready(true);
+  }
+  ~ExtensionsMockClientSocket() override {}
+
+  bool SetNoDelay(bool no_delay) override { return success_; }
+  bool SetKeepAlive(bool enable, int delay) override { return success_; }
+
+ private:
+  // Whether to return success for SetNoDelay() and SetKeepAlive().
+  const bool success_;
+
+  DISALLOW_COPY_AND_ASSIGN(ExtensionsMockClientSocket);
+};
+
+static const net::MockRead kMockReads[] = {net::MockRead(net::ASYNC, net::OK)};
+
+// A ClientSocketFactory to create sockets that simulate SetNoDelay and
+// SetKeepAlive success and failures.
+class TestSocketFactory : public net::ClientSocketFactory {
+ public:
+  explicit TestSocketFactory(bool success) : success_(success) {}
+  ~TestSocketFactory() override = default;
+
+  std::unique_ptr<net::DatagramClientSocket> CreateDatagramClientSocket(
+      net::DatagramSocket::BindType,
+      net::NetLog*,
+      const net::NetLogSource&) override {
+    NOTIMPLEMENTED();
+    return nullptr;
+  }
+  std::unique_ptr<net::TransportClientSocket> CreateTransportClientSocket(
+      const net::AddressList&,
+      std::unique_ptr<net::SocketPerformanceWatcher>,
+      net::NetLog*,
+      const net::NetLogSource&) override {
+    providers_.push_back(std::make_unique<net::StaticSocketDataProvider>(
+        kMockReads, base::span<net::MockWrite>()));
+    return std::make_unique<ExtensionsMockClientSocket>(providers_.back().get(),
+                                                        success_);
+  }
+  std::unique_ptr<net::SSLClientSocket> CreateSSLClientSocket(
+      std::unique_ptr<net::ClientSocketHandle>,
+      const net::HostPortPair&,
+      const net::SSLConfig&,
+      const net::SSLClientSocketContext&) override {
+    NOTIMPLEMENTED();
+    return std::unique_ptr<net::SSLClientSocket>();
+  }
+  std::unique_ptr<net::ProxyClientSocket> CreateProxyClientSocket(
+      std::unique_ptr<net::ClientSocketHandle> transport_socket,
+      const std::string& user_agent,
+      const net::HostPortPair& endpoint,
+      net::HttpAuthController* http_auth_controller,
+      bool tunnel,
+      bool using_spdy,
+      net::NextProto negotiated_protocol,
+      bool is_https_proxy,
+      const net::NetworkTrafficAnnotationTag& traffic_annotation) override {
+    NOTIMPLEMENTED();
+    return nullptr;
+  }
+  void ClearSSLSessionCache() override { NOTIMPLEMENTED(); }
+
+ private:
+  std::vector<std::unique_ptr<net::StaticSocketDataProvider>> providers_;
+  // Whether to return success for net::TransportClientSocket::SetNoDelay() and
+  // SetKeepAlive().
+  const bool success_;
+
+  DISALLOW_COPY_AND_ASSIGN(TestSocketFactory);
+};
+
+}  // namespace
+
+class TCPSocketSettingsTest : public TCPSocketUnitTestBase,
+                              public ::testing::WithParamInterface<bool> {
+ public:
+  TCPSocketSettingsTest()
+      : TCPSocketUnitTestBase(), client_socket_factory_(GetParam()) {
+    url_request_context_.set_client_socket_factory(&client_socket_factory_);
+    Initialize();
+  }
+  ~TCPSocketSettingsTest() override {}
+
+ private:
+  TestSocketFactory client_socket_factory_;
+};
+
+INSTANTIATE_TEST_CASE_P(/* no prefix */,
+                        TCPSocketSettingsTest,
+                        testing::Bool());
+
+TEST_P(TCPSocketSettingsTest, SetNoDelay) {
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+  bool expected_success = GetParam();
   {
-    testing::InSequence dummy;
-    EXPECT_CALL(*tcp_client_socket, SetNoDelay(_))
-        .WillOnce(testing::DoAll(SaveArg<0>(&no_delay), Return(true)));
-    EXPECT_CALL(*tcp_client_socket, SetNoDelay(_))
-        .WillOnce(testing::DoAll(SaveArg<0>(&no_delay), Return(false)));
+    base::RunLoop run_loop;
+    socket->SetNoDelay(true, base::BindLambdaForTesting([&](bool success) {
+                         EXPECT_EQ(expected_success, success);
+                         run_loop.Quit();
+                       }));
+    run_loop.Run();
   }
 
-  std::unique_ptr<TCPSocket> socket(
-      TCPSocket::CreateSocketForTesting(std::move(tcp_client_socket), FAKE_ID));
-
-  EXPECT_FALSE(no_delay);
-  int result = socket->SetNoDelay(true);
-  EXPECT_TRUE(result);
-  EXPECT_TRUE(no_delay);
-
-  result = socket->SetNoDelay(false);
-  EXPECT_FALSE(result);
-  EXPECT_FALSE(no_delay);
-}
-
-TEST(SocketTest, TestTCPSocketSetKeepAlive) {
-  net::AddressList address_list;
-  std::unique_ptr<MockTCPSocket> tcp_client_socket(
-      new MockTCPSocket(address_list));
-
-  bool enable = false;
-  int delay = 0;
   {
-    testing::InSequence dummy;
-    EXPECT_CALL(*tcp_client_socket, SetKeepAlive(_, _))
-        .WillOnce(testing::DoAll(SaveArg<0>(&enable), SaveArg<1>(&delay),
-                                 Return(true)));
-    EXPECT_CALL(*tcp_client_socket, SetKeepAlive(_, _))
-        .WillOnce(testing::DoAll(SaveArg<0>(&enable), SaveArg<1>(&delay),
-                                 Return(false)));
+    base::RunLoop run_loop;
+    socket->SetNoDelay(false, base::BindLambdaForTesting([&](bool success) {
+                         EXPECT_EQ(expected_success, success);
+                         run_loop.Quit();
+                       }));
+    run_loop.Run();
   }
-
-  std::unique_ptr<TCPSocket> socket(
-      TCPSocket::CreateSocketForTesting(std::move(tcp_client_socket), FAKE_ID));
-
-  EXPECT_FALSE(enable);
-  int result = socket->SetKeepAlive(true, 4500);
-  EXPECT_TRUE(result);
-  EXPECT_TRUE(enable);
-  EXPECT_EQ(4500, delay);
-
-  result = socket->SetKeepAlive(false, 0);
-  EXPECT_FALSE(result);
-  EXPECT_FALSE(enable);
-  EXPECT_EQ(0, delay);
 }
 
-TEST(SocketTest, TestTCPServerSocketListenAccept) {
-  std::unique_ptr<MockTCPServerSocket> tcp_server_socket(
-      new MockTCPServerSocket());
-  CompleteHandler handler;
+TEST_P(TCPSocketSettingsTest, SetKeepAlive) {
+  std::unique_ptr<TCPSocket> socket = CreateAndConnectSocket();
+  bool expected_success = GetParam();
+  {
+    base::RunLoop run_loop;
+    socket->SetKeepAlive(true /* enable */, 123 /* delay */,
+                         base::BindLambdaForTesting([&](bool success) {
+                           EXPECT_EQ(expected_success, success);
+                           run_loop.Quit();
+                         }));
+    run_loop.Run();
+  }
 
-  EXPECT_CALL(*tcp_server_socket, Accept(_, _)).Times(1);
-  EXPECT_CALL(*tcp_server_socket, Listen(_, _)).Times(1);
+  {
+    base::RunLoop run_loop;
+    socket->SetKeepAlive(false /* enable */, 123 /* delay */,
+                         base::BindLambdaForTesting([&](bool success) {
+                           EXPECT_EQ(expected_success, success);
+                           run_loop.Quit();
+                         }));
+    run_loop.Run();
+  }
+}
 
-  std::unique_ptr<TCPSocket> socket(TCPSocket::CreateServerSocketForTesting(
-      std::move(tcp_server_socket), FAKE_ID));
+class TCPSocketServerTest : public TCPSocketUnitTestBase {
+ public:
+  TCPSocketServerTest() : TCPSocketUnitTestBase() { Initialize(); }
+  ~TCPSocketServerTest() override {}
 
-  EXPECT_CALL(handler, OnAcceptMock(_, _));
+ private:
+  net::MockClientSocketFactory mock_client_socket_factory_;
+};
 
-  std::string err_msg;
-  EXPECT_EQ(net::OK, socket->Listen("127.0.0.1", 9999, 10, &err_msg));
-  socket->Accept(base::Bind(&CompleteHandler::OnAccept,
-        base::Unretained(&handler)));
+TEST_F(TCPSocketServerTest, ListenAccept) {
+  // Create a server socket.
+  std::unique_ptr<TCPSocket> socket = CreateSocket();
+  net::TestCompletionCallback callback;
+  base::RunLoop run_loop;
+  socket->Listen(
+      "127.0.0.1", 0 /* port */, 1 /* backlog */,
+      base::BindLambdaForTesting([&](int result, const std::string& error_msg) {
+        EXPECT_EQ(net::OK, result);
+        run_loop.Quit();
+      }));
+  run_loop.Run();
+  net::IPEndPoint server_addr;
+  EXPECT_TRUE(socket->GetLocalAddress(&server_addr));
+
+  base::RunLoop accept_run_loop;
+  net::IPEndPoint accept_client_addr;
+  socket->Accept(base::BindLambdaForTesting(
+      [&](int result, network::mojom::TCPConnectedSocketPtr accepted_socket,
+          const base::Optional<net::IPEndPoint>& remote_addr,
+          mojo::ScopedDataPipeConsumerHandle receive_handle,
+          mojo::ScopedDataPipeProducerHandle send_handle) {
+        EXPECT_EQ(net::OK, result);
+        accept_client_addr = remote_addr.value();
+        accept_run_loop.Quit();
+      }));
+
+  // Create a client socket to talk to the server socket.
+  auto client_socket = CreateAndConnectSocketWithAddress(server_addr);
+  accept_run_loop.Run();
+
+  net::IPEndPoint peer_addr;
+  EXPECT_TRUE(client_socket->GetPeerAddress(&peer_addr));
+  net::IPEndPoint client_addr;
+  EXPECT_TRUE(client_socket->GetLocalAddress(&client_addr));
+  EXPECT_EQ(server_addr, peer_addr);
+  EXPECT_EQ(client_addr, accept_client_addr);
+}
+
+TEST_F(TCPSocketServerTest, ReadAndWrite) {
+  // Create a server socket.
+  std::unique_ptr<TCPSocket> socket = CreateSocket();
+  net::TestCompletionCallback callback;
+  base::RunLoop run_loop;
+  socket->Listen(
+      "127.0.0.1", 0 /* port */, 1 /* backlog */,
+      base::BindLambdaForTesting([&](int result, const std::string& error_msg) {
+        EXPECT_EQ(net::OK, result);
+        run_loop.Quit();
+      }));
+  run_loop.Run();
+  net::IPEndPoint server_addr;
+  EXPECT_TRUE(socket->GetLocalAddress(&server_addr));
+
+  base::RunLoop accept_run_loop;
+  std::unique_ptr<TCPSocket> accepted_socket;
+
+  socket->Accept(base::BindLambdaForTesting(
+      [&](int result, network::mojom::TCPConnectedSocketPtr connected_socket,
+          const base::Optional<net::IPEndPoint>& remote_addr,
+          mojo::ScopedDataPipeConsumerHandle receive_handle,
+          mojo::ScopedDataPipeProducerHandle send_handle) {
+        EXPECT_EQ(net::OK, result);
+        accepted_socket = std::make_unique<TCPSocket>(
+            std::move(connected_socket), std::move(receive_handle),
+            std::move(send_handle), remote_addr, FAKE_ID);
+        accept_run_loop.Quit();
+      }));
+
+  // Create a client socket to talk to the server socket.
+  auto client_socket = CreateAndConnectSocketWithAddress(server_addr);
+  net::TestCompletionCallback connect_callback;
+  accept_run_loop.Run();
+
+  // Send data from the client to the server.
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  net::TestCompletionCallback write_callback;
+  client_socket->Write(io_buffer.get(), kTestMsgLength,
+                       write_callback.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback.WaitForResult());
+
+  std::string received_contents;
+  while (received_contents.size() < kTestMsgLength) {
+    base::RunLoop run_loop;
+    accepted_socket->Read(
+        kTestMsgLength,
+        base::BindLambdaForTesting([&](int result,
+                                       scoped_refptr<net::IOBuffer> io_buffer,
+                                       bool socket_destroying) {
+          ASSERT_GT(result, 0);
+          EXPECT_FALSE(socket_destroying);
+          received_contents.append(std::string(io_buffer->data(), result));
+          run_loop.Quit();
+        }));
+    run_loop.Run();
+  }
+  EXPECT_EQ(kTestMsg, received_contents);
+
+  // Send data from the server to the client.
+  net::TestCompletionCallback write_callback2;
+  accepted_socket->Write(io_buffer.get(), kTestMsgLength,
+                         write_callback2.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback2.WaitForResult());
+
+  std::string sent_contents;
+  while (sent_contents.size() < kTestMsgLength) {
+    base::RunLoop run_loop;
+    client_socket->Read(
+        kTestMsgLength,
+        base::BindLambdaForTesting([&](int result,
+                                       scoped_refptr<net::IOBuffer> io_buffer,
+                                       bool socket_destroying) {
+          ASSERT_GT(result, 0);
+          EXPECT_FALSE(socket_destroying);
+          sent_contents.append(std::string(io_buffer->data(), result));
+          run_loop.Quit();
+        }));
+    run_loop.Run();
+  }
+  EXPECT_EQ(kTestMsg, sent_contents);
 }
 
 }  // namespace extensions
diff --git a/chrome/browser/extensions/api/socket/tls_socket_unittest.cc b/chrome/browser/extensions/api/socket/tls_socket_unittest.cc
index 678b630..5e25fcf0 100644
--- a/chrome/browser/extensions/api/socket/tls_socket_unittest.cc
+++ b/chrome/browser/extensions/api/socket/tls_socket_unittest.cc
@@ -2,372 +2,482 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include <stddef.h>
-#include <stdint.h>
-
 #include <memory>
 #include <utility>
 
-#include "base/callback_helpers.h"
-#include "base/containers/circular_deque.h"
 #include "base/macros.h"
-#include "base/strings/string_piece.h"
+#include "base/test/bind_test_util.h"
+#include "chrome/browser/extensions/extension_service_test_base.h"
+#include "chrome/test/base/testing_profile.h"
+#include "content/public/browser/browser_context.h"
+#include "content/public/test/test_storage_partition.h"
 #include "extensions/browser/api/socket/tls_socket.h"
 #include "net/base/address_list.h"
 #include "net/base/completion_callback.h"
 #include "net/base/io_buffer.h"
 #include "net/base/net_errors.h"
-#include "net/base/rand_callback.h"
-#include "net/log/net_log_source.h"
-#include "net/log/net_log_with_source.h"
-#include "net/socket/next_proto.h"
-#include "net/socket/socket_tag.h"
-#include "net/socket/ssl_client_socket.h"
-#include "net/socket/tcp_client_socket.h"
-#include "net/traffic_annotation/network_traffic_annotation.h"
-#include "testing/gmock/include/gmock/gmock.h"
-
-using testing::_;
-using testing::DoAll;
-using testing::Invoke;
-using testing::Gt;
-using testing::Return;
-using testing::SaveArg;
-using testing::WithArgs;
-using base::StringPiece;
+#include "net/base/test_completion_callback.h"
+#include "net/socket/socket_test_util.h"
+#include "net/url_request/url_request_test_util.h"
+#include "services/network/network_context.h"
+#include "services/network/public/mojom/network_context.mojom.h"
 
 namespace extensions {
 
-class MockSSLClientSocket : public net::SSLClientSocket {
+namespace {
+
+const char kTestMsg[] = "abcdefghij";
+const int kTestMsgLength = strlen(kTestMsg);
+
+const char FAKE_ID[] = "abcdefghijklmnopqrst";
+
+class TLSSocketTestBase : public extensions::ExtensionServiceTestBase {
  public:
-  MockSSLClientSocket() {}
-  int Read(net::IOBuffer* buffer,
-           int bytes,
-           net::CompletionOnceCallback callback) override {
-    return Read(buffer, bytes,
-                base::AdaptCallbackForRepeating(std::move(callback)));
+  TLSSocketTestBase() : url_request_context_(true) {}
+  ~TLSSocketTestBase() override {}
+
+  // Creates a TCP socket.
+  std::unique_ptr<TCPSocket> CreateTCPSocket() {
+    auto socket = std::make_unique<TCPSocket>(&profile_, FAKE_ID);
+    socket->SetStoragePartitionForTest(&partition_);
+    net::TestCompletionCallback connect_callback;
+    net::IPEndPoint ip_end_point(net::IPAddress::IPv4Localhost(), kPort);
+    socket->Connect(net::AddressList(ip_end_point),
+                    connect_callback.callback());
+    if (net::OK != connect_callback.WaitForResult()) {
+      return nullptr;
+    }
+    return socket;
   }
 
-  int Write(net::IOBuffer* buffer,
-            int bytes,
-            net::CompletionOnceCallback callback,
-            const net::NetworkTrafficAnnotationTag& tag) override {
-    return Write(buffer, bytes,
-                 base::AdaptCallbackForRepeating(std::move(callback)), tag);
+  // Create a TCP socket and upgrade it to TLS.
+  std::unique_ptr<TLSSocket> CreateSocket() {
+    auto socket = CreateTCPSocket();
+    if (!socket)
+      return nullptr;
+    base::RunLoop run_loop;
+    net::HostPortPair host_port_pair("example.com", kPort);
+    net::IPEndPoint local_addr;
+    net::IPEndPoint peer_addr;
+    if (!socket->GetLocalAddress(&local_addr) ||
+        !socket->GetPeerAddress(&peer_addr)) {
+      return nullptr;
+    }
+    std::unique_ptr<TLSSocket> tls_socket;
+    socket->UpgradeToTLS(
+        nullptr /* options */,
+        base::BindLambdaForTesting(
+            [&](int result, network::mojom::TLSClientSocketPtr tls_socket_ptr,
+                const net::IPEndPoint& local_addr,
+                const net::IPEndPoint& peer_addr,
+                mojo::ScopedDataPipeConsumerHandle receive_handle,
+                mojo::ScopedDataPipeProducerHandle send_handle) {
+              if (net::OK == result) {
+                tls_socket = std::make_unique<TLSSocket>(
+                    std::move(tls_socket_ptr), local_addr, peer_addr,
+                    std::move(receive_handle), std::move(send_handle), FAKE_ID);
+              }
+              run_loop.Quit();
+            }));
+    run_loop.Run();
+    return tls_socket;
   }
 
-  int Connect(net::CompletionOnceCallback callback) override {
-    return Connect(base::AdaptCallbackForRepeating(std::move(callback)));
-  }
-
-  MOCK_METHOD0(Disconnect, void());
-  MOCK_METHOD3(Read,
-               int(net::IOBuffer* buf,
-                   int buf_len,
-                   const net::CompletionCallback& callback));
-  MOCK_METHOD4(Write,
-               int(net::IOBuffer* buf,
-                   int buf_len,
-                   const net::CompletionCallback& callback,
-                   const net::NetworkTrafficAnnotationTag&));
-  MOCK_METHOD1(SetReceiveBufferSize, int(int32_t));
-  MOCK_METHOD1(SetSendBufferSize, int(int32_t));
-  MOCK_METHOD1(Connect, int(const CompletionCallback&));
-  MOCK_CONST_METHOD0(IsConnectedAndIdle, bool());
-  MOCK_CONST_METHOD1(GetPeerAddress, int(net::IPEndPoint*));
-  MOCK_CONST_METHOD1(GetLocalAddress, int(net::IPEndPoint*));
-  MOCK_CONST_METHOD0(NetLog, const net::NetLogWithSource&());
-  MOCK_CONST_METHOD0(WasEverUsed, bool());
-  MOCK_CONST_METHOD0(UsingTCPFastOpen, bool());
-  MOCK_CONST_METHOD0(WasAlpnNegotiated, bool());
-  MOCK_CONST_METHOD0(GetNegotiatedProtocol, net::NextProto());
-  MOCK_METHOD1(GetSSLInfo, bool(net::SSLInfo*));
-  MOCK_CONST_METHOD1(GetConnectionAttempts, void(net::ConnectionAttempts*));
-  MOCK_METHOD0(ClearConnectionAttempts, void());
-  MOCK_METHOD1(AddConnectionAttempts, void(const net::ConnectionAttempts&));
-  MOCK_CONST_METHOD0(GetTotalReceivedBytes, int64_t());
-  MOCK_METHOD1(ApplySocketTag, void(const net::SocketTag&));
-  MOCK_METHOD5(ExportKeyingMaterial,
-               int(const StringPiece&,
-                   bool,
-                   const StringPiece&,
-                   unsigned char*,
-                   unsigned int));
-  MOCK_CONST_METHOD1(GetSSLCertRequestInfo, void(net::SSLCertRequestInfo*));
-  MOCK_CONST_METHOD0(GetUnverifiedServerCertificateChain,
-                     scoped_refptr<net::X509Certificate>());
-  MOCK_CONST_METHOD0(GetChannelIDService, net::ChannelIDService*());
-  MOCK_METHOD3(GetTokenBindingSignature,
-               net::Error(crypto::ECPrivateKey*,
-                          net::TokenBindingType,
-                          std::vector<uint8_t>*));
-  MOCK_CONST_METHOD0(GetChannelIDKey, crypto::ECPrivateKey*());
-  bool IsConnected() const override { return true; }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockSSLClientSocket);
-};
-
-class MockTCPSocket : public net::TCPClientSocket {
- public:
-  explicit MockTCPSocket(const net::AddressList& address_list)
-      : net::TCPClientSocket(address_list, NULL, NULL, net::NetLogSource()) {}
-
-  int Read(net::IOBuffer* buffer,
-           int bytes,
-           net::CompletionOnceCallback callback) override {
-    return Read(buffer, bytes,
-                base::AdaptCallbackForRepeating(std::move(callback)));
-  }
-
-  int Write(net::IOBuffer* buffer,
-            int bytes,
-            net::CompletionOnceCallback callback,
-            const net::NetworkTrafficAnnotationTag& tag) override {
-    return Write(buffer, bytes,
-                 base::AdaptCallbackForRepeating(std::move(callback)), tag);
-  }
-
-  MOCK_METHOD3(Read,
-               int(net::IOBuffer* buf,
-                   int buf_len,
-                   const net::CompletionCallback& callback));
-  MOCK_METHOD4(Write,
-               int(net::IOBuffer* buf,
-                   int buf_len,
-                   const net::CompletionCallback& callback,
-                   const net::NetworkTrafficAnnotationTag&));
-  MOCK_METHOD2(SetKeepAlive, bool(bool enable, int delay));
-  MOCK_METHOD1(SetNoDelay, bool(bool no_delay));
-
-  bool IsConnected() const override { return true; }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockTCPSocket);
-};
-
-class CompleteHandler {
- public:
-  CompleteHandler() {}
-  MOCK_METHOD1(OnComplete, void(int result_code));
-  MOCK_METHOD3(OnReadComplete,
-               void(int result_code,
-                    scoped_refptr<net::IOBuffer> io_buffer,
-                    bool socket_destroying));
-  MOCK_METHOD2(OnAccept, void(int, net::TCPClientSocket*));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(CompleteHandler);
-};
-
-class TLSSocketTest : public ::testing::Test {
- public:
-  TLSSocketTest() {}
-
-  void SetUp() override {
-    net::AddressList address_list;
-    // |ssl_socket_| is owned by |socket_|. TLSSocketTest keeps a pointer to
-    // it to expect invocations from TLSSocket to |ssl_socket_|.
-    std::unique_ptr<MockSSLClientSocket> ssl_sock(new MockSSLClientSocket);
-    ssl_socket_ = ssl_sock.get();
-    socket_.reset(new TLSSocket(std::move(ssl_sock), "test_extension_id"));
-    EXPECT_CALL(*ssl_socket_, Disconnect()).Times(1);
-  };
-
-  void TearDown() override {
-    ssl_socket_ = NULL;
-    socket_.reset();
-  };
-
  protected:
-  MockSSLClientSocket* ssl_socket_;
-  std::unique_ptr<TLSSocket> socket_;
+  // extensions::ExtensionServiceTestBase implementation.
+  void SetUp() override { InitializeEmptyExtensionService(); }
+
+  void Initialize() {
+    url_request_context_.Init();
+    network_context_ = std::make_unique<network::NetworkContext>(
+        nullptr, mojo::MakeRequest(&network_context_ptr_),
+        &url_request_context_);
+    partition_.set_network_context(network_context_ptr_.get());
+  }
+
+  net::TestURLRequestContext url_request_context_;
+
+ private:
+  static const int kPort = 1234;
+  TestingProfile profile_;
+  content::TestStoragePartition partition_;
+  std::unique_ptr<network::NetworkContext> network_context_;
+  network::mojom::NetworkContextPtr network_context_ptr_;
 };
 
-// Verify that a Read() on TLSSocket will pass through into a Read() on
-// |ssl_socket_| and invoke its completion callback.
-TEST_F(TLSSocketTest, TestTLSSocketRead) {
-  CompleteHandler handler;
+}  // namespace
 
-  EXPECT_CALL(*ssl_socket_, Read(_, _, _)).Times(1);
-  EXPECT_CALL(handler, OnReadComplete(_, _, _)).Times(1);
+class TLSSocketTest : public TLSSocketTestBase,
+                      public ::testing::WithParamInterface<net::IoMode> {
+ public:
+  TLSSocketTest() : TLSSocketTestBase() {
+    mock_client_socket_factory_.set_enable_read_if_ready(true);
+    url_request_context_.set_client_socket_factory(
+        &mock_client_socket_factory_);
+    Initialize();
+  }
+  ~TLSSocketTest() override {}
+
+  net::MockClientSocketFactory* mock_client_socket_factory() {
+    return &mock_client_socket_factory_;
+  }
+
+ private:
+  net::MockClientSocketFactory mock_client_socket_factory_;
+};
+
+TEST_F(TLSSocketTest, DestroyWhileReadPending) {
+  const net::MockRead kReads[] = {net::MockRead(net::SYNCHRONOUS, net::OK, 1)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(net::ASYNC, kTestMsg, kTestMsgLength, 0)};
+  net::StaticSocketDataProvider data_provider(kReads, kWrites);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  net::SSLSocketDataProvider ssl_socket(net::ASYNC, net::OK);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
+
+  int net_result = net::ERR_FAILED;
+  base::RunLoop run_loop;
+  int count = 1;
+  // Read one byte, and it should be pending because it is blocked on the mock
+  // write.
+  socket->Read(count,
+               base::BindLambdaForTesting(
+                   [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                       bool socket_destroying) {
+                     net_result = result;
+                     // |socket_destroying| should correctly denote that this
+                     // read callback is invoked through the destructor of
+                     // TLSSocket.
+                     EXPECT_TRUE(socket_destroying);
+                     run_loop.Quit();
+                   }));
+  // Destroy socket.
+  socket = nullptr;
+  // Wait for read callback.
+  run_loop.Run();
+  EXPECT_EQ(net::ERR_CONNECTION_CLOSED, net_result);
+}
+
+// UpgradeToTLS() fails when there is a pending read.
+TEST_F(TLSSocketTest, UpgradeToTLSWhilePendingRead) {
+  const net::MockRead kReads[] = {
+      net::MockRead(net::ASYNC, net::ERR_IO_PENDING)};
+  net::StaticSocketDataProvider data_provider(kReads,
+                                              base::span<net::MockWrite>());
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  auto socket = CreateTCPSocket();
+  // This read will be pending when UpgradeToTLS() is called.
+  socket->Read(1 /* count */, base::DoNothing());
+  network::mojom::TLSClientSocketPtr tls_socket_ptr;
+  base::RunLoop run_loop;
+  socket->UpgradeToTLS(
+      nullptr /* options */,
+      base::BindLambdaForTesting(
+          [&](int result, network::mojom::TLSClientSocketPtr tls_socket_ptr,
+              const net::IPEndPoint& local_addr,
+              const net::IPEndPoint& peer_addr,
+              mojo::ScopedDataPipeConsumerHandle receive_handle,
+              mojo::ScopedDataPipeProducerHandle send_handle) {
+            EXPECT_EQ(net::ERR_FAILED, result);
+            run_loop.Quit();
+          }));
+  run_loop.Run();
+}
+
+TEST_F(TLSSocketTest, UpgradeToTLSWithCustomOptions) {
+  // Mock data are not consumed. These are here so that net::StreamSocket::Read
+  // is always pending and blocked on the write. Otherwise, mock socket data
+  // will complains that there aren't any data to read.
+  const net::MockRead kReads[] = {
+      net::MockRead(net::ASYNC, kTestMsg, kTestMsgLength, 1),
+      net::MockRead(net::ASYNC, net::OK, 2)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(net::ASYNC, kTestMsg, kTestMsgLength, 0)};
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(net::ASYNC, net::OK);
+  ssl_socket.expected_ssl_version_min = net::SSL_PROTOCOL_VERSION_TLS1_1;
+  ssl_socket.expected_ssl_version_max = net::SSL_PROTOCOL_VERSION_TLS1_2;
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+
+  auto socket = CreateTCPSocket();
+  network::mojom::TLSClientSocketPtr tls_socket_ptr;
+  api::socket::SecureOptions options;
+  options.tls_version = std::make_unique<api::socket::TLSVersionConstraints>();
+  options.tls_version->min = std::make_unique<std::string>("tls1.1");
+  options.tls_version->max = std::make_unique<std::string>("tls1.2");
+  int net_error = net::ERR_FAILED;
+  base::RunLoop run_loop;
+  socket->UpgradeToTLS(
+      &options,
+      base::BindLambdaForTesting(
+          [&](int result, network::mojom::TLSClientSocketPtr tls_socket_ptr,
+              const net::IPEndPoint& local_addr,
+              const net::IPEndPoint& peer_addr,
+              mojo::ScopedDataPipeConsumerHandle receive_handle,
+              mojo::ScopedDataPipeProducerHandle send_handle) {
+            net_error = result;
+            run_loop.Quit();
+          }));
+  run_loop.Run();
+  EXPECT_EQ(net::OK, net_error);
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
+}
+
+INSTANTIATE_TEST_CASE_P(/* no prefix */,
+                        TLSSocketTest,
+                        testing::Values(net::SYNCHRONOUS, net::ASYNC));
+
+TEST_P(TLSSocketTest, ReadWrite) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::ASYNC, kTestMsg, kTestMsgLength, 1),
+      net::MockRead(io_mode, net::OK, 2)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(net::SYNCHRONOUS, kTestMsg, kTestMsgLength, 0)};
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
+
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
+
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  net::TestCompletionCallback write_callback;
+  socket->Write(io_buffer.get(), kTestMsgLength, write_callback.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback.WaitForResult());
+
+  std::string received_data;
+  int count = 512;
+  while (true) {
+    base::RunLoop run_loop;
+    int net_error = net::ERR_FAILED;
+    socket->Read(count,
+                 base::BindLambdaForTesting(
+                     [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                         bool socket_destroying) {
+                       net_error = result;
+                       EXPECT_FALSE(socket_destroying);
+                       if (result > 0)
+                         received_data.append(io_buffer->data(), result);
+                       run_loop.Quit();
+                     }));
+    run_loop.Run();
+    if (net_error <= 0)
+      break;
+  }
+  EXPECT_EQ(kTestMsg, received_data);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
+}
+
+// Tests the case where read size is smaller than the actual message.
+TEST_P(TLSSocketTest, PartialRead) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::ASYNC, kTestMsg, kTestMsgLength, 1),
+      net::MockRead(io_mode, net::OK, 2)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(net::SYNCHRONOUS, kTestMsg, kTestMsgLength, 0)};
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
+
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  net::TestCompletionCallback write_callback;
+  socket->Write(io_buffer.get(), kTestMsgLength, write_callback.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback.WaitForResult());
+
+  int count = 1;
+  std::string received_data;
+  while (true) {
+    int net_result = net::ERR_FAILED;
+    base::RunLoop run_loop;
+    socket->Read(count,
+                 base::BindLambdaForTesting(
+                     [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                         bool socket_destroying) {
+                       net_result = result;
+                       EXPECT_FALSE(socket_destroying);
+                       if (result > 0)
+                         received_data.append(io_buffer->data(), result);
+                       run_loop.Quit();
+                     }));
+    run_loop.Run();
+    if (net_result <= 0)
+      break;
+    // Double the read size in the next iteration.
+    count *= 2;
+  }
+  EXPECT_EQ(kTestMsg, received_data);
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
+}
+
+TEST_P(TLSSocketTest, ReadError) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {
+      net::MockRead(net::ASYNC, net::ERR_INSUFFICIENT_RESOURCES, 1)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(net::SYNCHRONOUS, kTestMsg, kTestMsgLength, 0)};
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
+
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  net::TestCompletionCallback write_callback;
+  socket->Write(io_buffer.get(), kTestMsgLength, write_callback.callback());
+  EXPECT_EQ(kTestMsgLength, write_callback.WaitForResult());
 
   const int count = 512;
-  socket_->Read(
-      count,
-      base::Bind(&CompleteHandler::OnReadComplete, base::Unretained(&handler)));
+  int net_error = net::OK;
+  while (true) {
+    base::RunLoop run_loop;
+    socket->Read(count,
+                 base::BindLambdaForTesting(
+                     [&](int result, scoped_refptr<net::IOBuffer> io_buffer,
+                         bool socket_destroying) {
+                       net_error = result;
+                       EXPECT_FALSE(socket_destroying);
+                       if (result <= 0) {
+                         EXPECT_FALSE(socket->IsConnected());
+                         EXPECT_EQ(nullptr, io_buffer);
+                       } else {
+                         EXPECT_TRUE(socket->IsConnected());
+                       }
+                       run_loop.Quit();
+                     }));
+    run_loop.Run();
+    if (net_error < 0)
+      break;
+  }
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
 }
 
-// Verify that a Write() on a TLSSocket will pass through to Write()
-// invocations on |ssl_socket_|, handling partial writes correctly, and calls
-// the completion callback correctly.
-TEST_F(TLSSocketTest, TestTLSSocketWrite) {
-  CompleteHandler handler;
-  net::CompletionCallback callback;
+// Tests the case where a message is split over two separate socket writes.
+TEST_P(TLSSocketTest, MultipleWrite) {
+  const char kFirstHalfTestMsg[] = "abcde";
+  const char kSecondHalfTestMsg[] = "fghij";
+  EXPECT_EQ(kTestMsg, std::string(kFirstHalfTestMsg) + kSecondHalfTestMsg);
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {net::MockRead(net::ASYNC, net::OK, 2)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, kFirstHalfTestMsg, strlen(kFirstHalfTestMsg), 0),
+      net::MockWrite(io_mode, kSecondHalfTestMsg, strlen(kSecondHalfTestMsg),
+                     1)};
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
 
-  EXPECT_CALL(*ssl_socket_, Write(_, _, _, _))
-      .Times(2)
-      .WillRepeatedly(DoAll(SaveArg<2>(&callback), Return(128)));
-  EXPECT_CALL(handler, OnComplete(_)).Times(1);
-
-  scoped_refptr<net::IOBufferWithSize> io_buffer(
-      new net::IOBufferWithSize(256));
-  socket_->Write(
-      io_buffer.get(),
-      io_buffer->size(),
-      base::Bind(&CompleteHandler::OnComplete, base::Unretained(&handler)));
+  int num_bytes_written = 0;
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  auto drainable_io_buffer = base::MakeRefCounted<net::DrainableIOBuffer>(
+      io_buffer.get(), kTestMsgLength);
+  while (num_bytes_written < kTestMsgLength) {
+    net::TestCompletionCallback write_callback;
+    socket->Write(drainable_io_buffer.get(), kTestMsgLength - num_bytes_written,
+                  write_callback.callback());
+    int result = write_callback.WaitForResult();
+    ASSERT_GT(result, net::OK);
+    drainable_io_buffer->DidConsume(result);
+    num_bytes_written += result;
+    // Flushes the write.
+    base::RunLoop().RunUntilIdle();
+  }
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
 }
 
-// Simulate a blocked Write, and verify that, when simulating the Write going
-// through, the callback gets invoked.
-TEST_F(TLSSocketTest, TestTLSSocketBlockedWrite) {
-  CompleteHandler handler;
-  net::CompletionCallback callback;
+TEST_P(TLSSocketTest, PartialWrite) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {net::MockRead(net::ASYNC, net::OK, 4)};
+  const net::MockWrite kWrites[] = {net::MockWrite(io_mode, "a", 1, 0),
+                                    net::MockWrite(io_mode, "bc", 2, 1),
+                                    net::MockWrite(io_mode, "defg", 4, 2),
+                                    net::MockWrite(io_mode, "hij", 3, 3)};
 
-  // Return ERR_IO_PENDING to say the Write()'s blocked. Save the |callback|
-  // Write()'s passed.
-  EXPECT_CALL(*ssl_socket_, Write(_, _, _, _))
-      .Times(2)
-      .WillRepeatedly(
-          DoAll(SaveArg<2>(&callback), Return(net::ERR_IO_PENDING)));
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
 
-  scoped_refptr<net::IOBufferWithSize> io_buffer(new net::IOBufferWithSize(42));
-  socket_->Write(
-      io_buffer.get(),
-      io_buffer->size(),
-      base::Bind(&CompleteHandler::OnComplete, base::Unretained(&handler)));
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
 
-  // After the simulated asynchronous writes come back (via calls to
-  // callback.Run()), hander's OnComplete() should get invoked with the total
-  // amount written.
-  EXPECT_CALL(handler, OnComplete(42)).Times(1);
-  callback.Run(40);
-  callback.Run(2);
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
+
+  // Start with writing one byte, and double that in the next iteration.
+  int num_bytes_to_write = 1;
+  int num_bytes_written = 0;
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  auto drainable_io_buffer = base::MakeRefCounted<net::DrainableIOBuffer>(
+      io_buffer.get(), kTestMsgLength);
+  while (num_bytes_written < kTestMsgLength) {
+    net::TestCompletionCallback write_callback;
+    socket->Write(
+        drainable_io_buffer.get(),
+        std::max(kTestMsgLength - num_bytes_written, num_bytes_to_write),
+        write_callback.callback());
+    int result = write_callback.WaitForResult();
+    ASSERT_GT(result, net::OK);
+    drainable_io_buffer->DidConsume(result);
+    num_bytes_written += result;
+    num_bytes_to_write *= 2;
+    // Flushes the write.
+    base::RunLoop().RunUntilIdle();
+  }
+  EXPECT_TRUE(data_provider.AllReadDataConsumed());
+  EXPECT_TRUE(data_provider.AllWriteDataConsumed());
+  EXPECT_TRUE(ssl_socket.ConnectDataConsumed());
 }
 
-// Simulate multiple blocked Write()s.
-TEST_F(TLSSocketTest, TestTLSSocketBlockedWriteReentry) {
-  const int kNumIOs = 5;
-  CompleteHandler handlers[kNumIOs];
-  net::CompletionCallback callback;
-  scoped_refptr<net::IOBufferWithSize> io_buffers[kNumIOs];
+TEST_P(TLSSocketTest, WriteError) {
+  net::IoMode io_mode = GetParam();
+  const net::MockRead kReads[] = {net::MockRead(net::ASYNC, net::OK, 1)};
+  const net::MockWrite kWrites[] = {
+      net::MockWrite(io_mode, net::ERR_INSUFFICIENT_RESOURCES, 0)};
 
-  // The implementation of TLSSocket::Write() is inherited from
-  // Socket::Write(), which implements an internal write queue that wraps
-  // TLSSocket::WriteImpl(). Each call from TLSSocket::WriteImpl() will invoke
-  // |ssl_socket_|'s Write() (mocked here). Save the |callback| (assume they
-  // will all be equivalent), and return ERR_IO_PENDING, to indicate a blocked
-  // request. The mocked SSLClientSocket::Write() will get one request per
-  // TLSSocket::Write() request invoked on |socket_| below.
-  EXPECT_CALL(*ssl_socket_, Write(_, _, _, _))
-      .Times(kNumIOs)
-      .WillRepeatedly(
-          DoAll(SaveArg<2>(&callback), Return(net::ERR_IO_PENDING)));
+  net::SequencedSocketData data_provider(kReads, kWrites);
+  net::SSLSocketDataProvider ssl_socket(io_mode, net::OK);
 
-  // Send out |kNuMIOs| requests, each with a different size.
-  for (int i = 0; i < kNumIOs; i++) {
-    io_buffers[i] = new net::IOBufferWithSize(128 + i * 50);
-    socket_->Write(io_buffers[i].get(),
-                   io_buffers[i]->size(),
-                   base::Bind(&CompleteHandler::OnComplete,
-                              base::Unretained(&handlers[i])));
+  mock_client_socket_factory()->AddSocketDataProvider(&data_provider);
+  mock_client_socket_factory()->AddSSLSocketDataProvider(&ssl_socket);
+  std::unique_ptr<TLSSocket> socket = CreateSocket();
 
-    // Set up expectations on all |kNumIOs| handlers.
-    EXPECT_CALL(handlers[i], OnComplete(io_buffers[i]->size())).Times(1);
+  // Mojo data pipe might buffer some write data, so continue writing until the
+  // write error is received.
+  auto io_buffer = base::MakeRefCounted<net::StringIOBuffer>(kTestMsg);
+  int net_error = net::OK;
+  while (true) {
+    base::RunLoop run_loop;
+    socket->Write(io_buffer.get(), kTestMsgLength,
+                  base::BindLambdaForTesting([&](int result) {
+                    if (result == net::ERR_FAILED)
+                      EXPECT_FALSE(socket->IsConnected());
+                    net_error = result;
+                    run_loop.Quit();
+                  }));
+    run_loop.Run();
+    if (net_error <= 0)
+      break;
   }
-
-  // Finish each pending I/O. This should satisfy the expectations on the
-  // handlers.
-  for (int i = 0; i < kNumIOs; i++) {
-    callback.Run(128 + i * 50);
-  }
-}
-
-typedef std::pair<net::CompletionCallback, int> PendingCallback;
-
-class CallbackList : public base::circular_deque<PendingCallback> {
- public:
-  void append(const net::CompletionCallback& cb, int arg) {
-    push_back(std::make_pair(cb, arg));
-  }
-};
-
-// Simulate Write()s above and below a SSLClientSocket size limit.
-TEST_F(TLSSocketTest, TestTLSSocketLargeWrites) {
-  const int kSizeIncrement = 4096;
-  const int kNumIncrements = 10;
-  const int kFragmentIncrement = 4;
-  const int kSizeLimit = kSizeIncrement * kFragmentIncrement;
-  net::CompletionCallback callback;
-  CompleteHandler handler;
-  scoped_refptr<net::IOBufferWithSize> io_buffers[kNumIncrements];
-  CallbackList pending_callbacks;
-  size_t total_bytes_requested = 0;
-  size_t total_bytes_written = 0;
-
-  // Some implementations of SSLClientSocket may have write-size limits (e.g,
-  // max 1 TLS record, which is 16k). This test mocks a size limit at
-  // |kSizeIncrement| and calls Write() above and below that limit. It
-  // simulates SSLClientSocket::Write() behavior in only writing up to the size
-  // limit, requiring additional calls for the remaining data to be sent.
-  // Socket::Write() (and supporting methods) execute the additional calls as
-  // needed. This test verifies that this inherited implementation does
-  // properly issue additional calls, and that the total amount returned from
-  // all mocked SSLClientSocket::Write() calls is the same as originally
-  // requested.
-
-  // |ssl_socket_|'s Write() will write at most |kSizeLimit| bytes. The
-  // inherited Socket::Write() will repeatedly call |ssl_socket_|'s Write()
-  // until the entire original request is sent. Socket::Write() will queue any
-  // additional write requests until the current request is complete. A
-  // request is complete when the callback passed to Socket::WriteImpl() is
-  // invoked with an argument equal to the original number of bytes requested
-  // from Socket::Write(). If the callback is invoked with a smaller number,
-  // Socket::WriteImpl() will get repeatedly invoked until the sum of the
-  // callbacks' arguments is equal to the original requested amount.
-  EXPECT_CALL(*ssl_socket_, Write(_, _, _, _))
-      .WillRepeatedly(DoAll(
-          WithArgs<2, 1>(Invoke(&pending_callbacks, &CallbackList::append)),
-          Return(net::ERR_IO_PENDING)));
-
-  // Observe what comes back from Socket::Write() here.
-  EXPECT_CALL(handler, OnComplete(Gt(0))).Times(kNumIncrements);
-
-  // Send out |kNumIncrements| requests, each with a different size. The
-  // last request is the same size as the first, and the ones in the middle
-  // are monotonically increasing from the first.
-  for (int i = 0; i < kNumIncrements; i++) {
-    const bool last = i == (kNumIncrements - 1);
-    io_buffers[i] = new net::IOBufferWithSize(last ? kSizeIncrement
-                                                   : kSizeIncrement * (i + 1));
-    total_bytes_requested += io_buffers[i]->size();
-
-    // Invoke Socket::Write(). This will invoke |ssl_socket_|'s Write(), which
-    // this test mocks out. That mocked Write() is in an asynchronous waiting
-    // state until the passed callback (saved in the EXPECT_CALL for
-    // |ssl_socket_|'s Write()) is invoked.
-    socket_->Write(
-        io_buffers[i].get(),
-        io_buffers[i]->size(),
-        base::Bind(&CompleteHandler::OnComplete, base::Unretained(&handler)));
-  }
-
-  // Invoke callbacks for pending I/Os. These can synchronously invoke more of
-  // |ssl_socket_|'s Write() as needed. The callback checks how much is left
-  // in the request, and then starts issuing any queued Socket::Write()
-  // invocations.
-  while (!pending_callbacks.empty()) {
-    PendingCallback cb = pending_callbacks.front();
-    pending_callbacks.pop_front();
-
-    int amount_written_invocation = std::min(kSizeLimit, cb.second);
-    total_bytes_written += amount_written_invocation;
-    cb.first.Run(amount_written_invocation);
-  }
-
-  ASSERT_EQ(total_bytes_requested, total_bytes_written)
-      << "There should be exactly as many bytes written as originally "
-      << "requested to Write().";
+  // Note that TCPSocket only detects that send pipe is broken and propagates
+  // it as a net::ERR_FAILED. It doesn't know the specific net error code. To do
+  // that, it needs to register itself as a network::mojom::SocketObserver.
+  EXPECT_EQ(net::ERR_FAILED, net_error);
 }
 
 }  // namespace extensions
diff --git a/chrome/browser/flag_descriptions.cc b/chrome/browser/flag_descriptions.cc
index d5fccec..09755bb5 100644
--- a/chrome/browser/flag_descriptions.cc
+++ b/chrome/browser/flag_descriptions.cc
@@ -1134,11 +1134,19 @@
 const char kNewBookmarkAppsDescription[] =
     "Enables the new system for creating bookmark apps.";
 
-const char kNewPasswordFormParsingName[] = "New password form parsing";
+const char kNewPasswordFormParsingName[] =
+    "New password form parsing for filling passwords";
 const char kNewPasswordFormParsingDescription[] =
-    "Replaces existing form parsing in password manager with a new version, "
-    "currently under development. WARNING: when enabled Password Manager might "
-    "stop working";
+    "Replaces existing form parsing for filling in password manager with a new "
+    "version, currently under development. WARNING: when enabled, Password "
+    "Manager might stop working";
+
+const char kNewPasswordFormParsingForSavingName[] =
+    "New password form parsing for saving passwords";
+const char kNewPasswordFormParsingForSavingDescription[] =
+    "Replaces existing form parsing for saving in password manager with a new "
+    "version, currently under development. WARNING: when enabled, Password "
+    "Manager might stop working";
 
 const char kNewRemotePlaybackPipelineName[] =
     "Enable the new remote playback pipeline.";
diff --git a/chrome/browser/flag_descriptions.h b/chrome/browser/flag_descriptions.h
index dd9999c..3a55f66 100644
--- a/chrome/browser/flag_descriptions.h
+++ b/chrome/browser/flag_descriptions.h
@@ -711,6 +711,9 @@
 extern const char kNewPasswordFormParsingName[];
 extern const char kNewPasswordFormParsingDescription[];
 
+extern const char kNewPasswordFormParsingForSavingName[];
+extern const char kNewPasswordFormParsingForSavingDescription[];
+
 extern const char kNewRemotePlaybackPipelineName[];
 extern const char kNewRemotePlaybackPipelineDescription[];
 
diff --git a/chrome/browser/history/top_sites_factory.cc b/chrome/browser/history/top_sites_factory.cc
index 5ca3310..a8746a8 100644
--- a/chrome/browser/history/top_sites_factory.cc
+++ b/chrome/browser/history/top_sites_factory.cc
@@ -30,6 +30,7 @@
 #include "components/history/core/browser/top_sites_impl.h"
 #include "components/history/core/browser/top_sites_provider.h"
 #include "components/keyed_service/content/browser_context_dependency_manager.h"
+#include "components/ntp_tiles/constants.h"
 #include "components/pref_registry/pref_registry_syncable.h"
 #include "content/public/browser/browser_thread.h"
 #include "ui/base/l10n/l10n_util.h"
@@ -44,23 +45,22 @@
 }
 
 struct RawPrepopulatedPage {
-  int url_id;        // The resource for the page URL.
-  int title_id;      // The resource for the page title.
-  int favicon_id;    // The raw data resource for the favicon.
-  int thumbnail_id;  // The raw data resource for the thumbnail.
-  SkColor color;     // The best color to highlight the page (should roughly
-                     // match favicon).
+  int url_id;            // The resource for the page URL.
+  int title_id;          // The resource for the page title.
+  int favicon_id;        // The raw data resource for the favicon.
+  int large_favicon_id;  // The raw data resource for the larger favicon.
+  int thumbnail_id;      // The raw data resource for the thumbnail.
+  SkColor color;         // The best color to highlight the page (should
+                         // roughly match favicon).
 };
 
 #if !defined(OS_ANDROID)
 // Android does not use prepopulated pages.
 const RawPrepopulatedPage kRawPrepopulatedPages[] = {
     {
-     IDS_WEBSTORE_URL,
-     IDS_EXTENSION_WEB_STORE_TITLE,
-     IDR_WEBSTORE_ICON_16,
-     IDR_NEWTAB_WEBSTORE_THUMBNAIL,
-     SkColorSetRGB(63, 132, 197),
+        IDS_WEBSTORE_URL, IDS_EXTENSION_WEB_STORE_TITLE, IDR_WEBSTORE_ICON_16,
+        IDR_WEBSTORE_ICON_32, IDR_NEWTAB_WEBSTORE_THUMBNAIL,
+        SkColorSetRGB(63, 132, 197),
     },
 };
 #endif
@@ -74,7 +74,8 @@
     const RawPrepopulatedPage& page = kRawPrepopulatedPages[i];
     prepopulated_pages->push_back(history::PrepopulatedPage(
         GURL(l10n_util::GetStringUTF8(page.url_id)),
-        l10n_util::GetStringUTF16(page.title_id), page.favicon_id,
+        l10n_util::GetStringUTF16(page.title_id),
+        ntp_tiles::IsMDIconsEnabled() ? page.large_favicon_id : page.favicon_id,
         page.thumbnail_id, page.color));
   }
 #endif
diff --git a/chrome/browser/media/router/BUILD.gn b/chrome/browser/media/router/BUILD.gn
index 880e6ae4..498fae2 100644
--- a/chrome/browser/media/router/BUILD.gn
+++ b/chrome/browser/media/router/BUILD.gn
@@ -14,6 +14,7 @@
     "//components/keyed_service/core",
     "//content/public/browser",
     "//content/public/common",
+    "//crypto",
     "//net",
     "//third_party/icu",
     "//url",
@@ -93,6 +94,8 @@
       "providers/cast/cast_app_availability_tracker.h",
       "providers/cast/cast_app_discovery_service.cc",
       "providers/cast/cast_app_discovery_service.h",
+      "providers/cast/cast_internal_message_util.cc",
+      "providers/cast/cast_internal_message_util.h",
       "providers/cast/cast_media_route_provider.cc",
       "providers/cast/cast_media_route_provider.h",
       "providers/cast/cast_media_route_provider_metrics.cc",
diff --git a/chrome/browser/media/router/media_router_feature.cc b/chrome/browser/media/router/media_router_feature.cc
index 487e2f34..8f9385c 100644
--- a/chrome/browser/media/router/media_router_feature.cc
+++ b/chrome/browser/media/router/media_router_feature.cc
@@ -4,10 +4,13 @@
 
 #include "chrome/browser/media/router/media_router_feature.h"
 
+#include "base/base64.h"
 #include "base/feature_list.h"
+#include "base/strings/string_util.h"
 #include "build/build_config.h"
 #include "chrome/browser/profiles/profile.h"
 #include "content/public/browser/browser_context.h"
+#include "crypto/random.h"
 #include "extensions/buildflags/buildflags.h"
 #include "ui/base/ui_features.h"
 
@@ -73,6 +76,12 @@
                                 PrefRegistry::PUBLIC);
 }
 
+void RegisterProfilePrefs(PrefRegistrySimple* registry) {
+  // TODO(imcheng): Migrate existing Media Router prefs to here.
+  registry->RegisterStringPref(prefs::kMediaRouterReceiverIdHashToken, "",
+                               PrefRegistry::PUBLIC);
+}
+
 const base::Feature kCastAllowAllIPsFeature{"CastAllowAllIPs",
                                             base::FEATURE_DISABLED_BY_DEFAULT};
 
@@ -90,6 +99,19 @@
   return allow_all_ips;
 }
 
+std::string GetReceiverIdHashToken(PrefService* pref_service) {
+  static constexpr size_t kHashTokenSize = 64;
+  std::string token =
+      pref_service->GetString(prefs::kMediaRouterReceiverIdHashToken);
+  if (token.empty()) {
+    crypto::RandBytes(base::WriteInto(&token, kHashTokenSize + 1),
+                      kHashTokenSize);
+    base::Base64Encode(token, &token);
+    pref_service->SetString(prefs::kMediaRouterReceiverIdHashToken, token);
+  }
+  return token;
+}
+
 bool DialMediaRouteProviderEnabled() {
   return base::FeatureList::IsEnabled(kDialMediaRouteProvider);
 }
diff --git a/chrome/browser/media/router/media_router_feature.h b/chrome/browser/media/router/media_router_feature.h
index 73791ef..400912d3 100644
--- a/chrome/browser/media/router/media_router_feature.h
+++ b/chrome/browser/media/router/media_router_feature.h
@@ -25,11 +25,18 @@
 // Pref name for the enterprise policy for allowing Cast devices on all IPs.
 constexpr char kMediaRouterCastAllowAllIPs[] =
     "media_router.cast_allow_all_ips";
+// Pref name for the per-profile randomly generated token to include with the
+// hash when externalizing MediaSink IDs.
+constexpr char kMediaRouterReceiverIdHashToken[] =
+    "media_router.receiver_id_hash_token";
 }  // namespace prefs
 
 // Registers |kMediaRouterCastAllowAllIPs| with local state pref |registry|.
 void RegisterLocalStatePrefs(PrefRegistrySimple* registry);
 
+// Registers Media Router related preferences with per-profile pref |registry|.
+void RegisterProfilePrefs(PrefRegistrySimple* registry);
+
 // If enabled, allows Media Router to connect to Cast devices on all IP
 // addresses, not just RFC1918/RFC4913 private addresses. Workaround for
 // https://crbug.com/813974.
@@ -39,6 +46,11 @@
 // all IPs, as determined by local state |pref_service| / feature flag.
 bool GetCastAllowAllIPsPref(PrefService* pref_service);
 
+// Returns the hash token to use for externalizing MediaSink IDs from
+// |pref_service|. If the token does not exist, the token will be created from a
+// randomly generated string and stored in |pref_service|.
+std::string GetReceiverIdHashToken(PrefService* pref_service);
+
 extern const base::Feature kEnableDialSinkQuery;
 extern const base::Feature kEnableCastDiscovery;
 extern const base::Feature kCastMediaRouteProvider;
diff --git a/chrome/browser/media/router/media_router_feature_unittest.cc b/chrome/browser/media/router/media_router_feature_unittest.cc
index 16ed5b7..e61e9d4 100644
--- a/chrome/browser/media/router/media_router_feature_unittest.cc
+++ b/chrome/browser/media/router/media_router_feature_unittest.cc
@@ -30,4 +30,16 @@
   EXPECT_FALSE(GetCastAllowAllIPsPref(pref_service.get()));
 }
 
+TEST(MediaRouterFeatureTest, GetReceiverIdHashToken) {
+  auto pref_service = std::make_unique<TestingPrefServiceSimple>();
+  pref_service->registry()->RegisterStringPref(
+      prefs::kMediaRouterReceiverIdHashToken, "");
+
+  std::string token = GetReceiverIdHashToken(pref_service.get());
+  EXPECT_FALSE(token.empty());
+
+  // Token stays the same on subsequent invocation.
+  EXPECT_EQ(token, GetReceiverIdHashToken(pref_service.get()));
+}
+
 }  // namespace media_router
diff --git a/chrome/browser/media/router/providers/cast/cast_app_availability_tracker_unittest.cc b/chrome/browser/media/router/providers/cast/cast_app_availability_tracker_unittest.cc
index 591ef24..f2fd454 100644
--- a/chrome/browser/media/router/providers/cast/cast_app_availability_tracker_unittest.cc
+++ b/chrome/browser/media/router/providers/cast/cast_app_availability_tracker_unittest.cc
@@ -76,7 +76,7 @@
 
 TEST_F(CastAppAvailabilityTrackerTest, MultipleAppIdsAlreadyTrackingOne) {
   // One of the mirroring app IDs.
-  auto source1 = CastMediaSource::From("cast:0F5096E8");
+  auto source1 = CastMediaSource::From("cast:0F5096E8?clientId=123");
   ASSERT_TRUE(source1);
 
   base::flat_set<std::string> new_app_ids = {"0F5096E8"};
diff --git a/chrome/browser/media/router/providers/cast/cast_internal_message_util.cc b/chrome/browser/media/router/providers/cast/cast_internal_message_util.cc
new file mode 100644
index 0000000..a73dfe7
--- /dev/null
+++ b/chrome/browser/media/router/providers/cast/cast_internal_message_util.cc
@@ -0,0 +1,394 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/media/router/providers/cast/cast_internal_message_util.h"
+
+#include "base/base64url.h"
+#include "base/json/json_writer.h"
+#include "base/memory/ptr_util.h"
+#include "base/sha1.h"
+#include "chrome/common/media_router/discovery/media_sink_internal.h"
+#include "chrome/common/media_router/providers/cast/cast_media_source.h"
+#include "components/cast_channel/cast_socket.h"
+#include "components/cast_channel/proto/cast_channel.pb.h"
+#include "net/base/escape.h"
+
+namespace media_router {
+
+namespace {
+
+constexpr char kClientConnect[] = "client_connect";
+constexpr char kAppMessage[] = "app_message";
+constexpr char kReceiverAction[] = "receiver_action";
+constexpr char kNewSession[] = "new_session";
+
+bool GetString(const base::Value& value,
+               const std::string& key,
+               std::string* out) {
+  const base::Value* string_value =
+      value.FindKeyOfType(key, base::Value::Type::STRING);
+  if (!string_value)
+    return false;
+
+  *out = string_value->GetString();
+  return !out->empty();
+}
+
+void CopyValueWithDefault(const base::Value& from,
+                          const std::string& key,
+                          base::Value default_value,
+                          base::Value* to) {
+  const base::Value* value = from.FindKey(key);
+  to->SetKey(key, value ? value->Clone() : std::move(default_value));
+}
+
+void CopyValue(const base::Value& from,
+               const std::string& key,
+               base::Value* to) {
+  const base::Value* value = from.FindKey(key);
+  if (value)
+    to->SetKey(key, value->Clone());
+}
+
+CastInternalMessage::Type CastInternalMessageTypeFromString(
+    const std::string& type) {
+  if (type == kClientConnect)
+    return CastInternalMessage::Type::kClientConnect;
+  if (type == kAppMessage)
+    return CastInternalMessage::Type::kAppMessage;
+  if (type == kReceiverAction)
+    return CastInternalMessage::Type::kReceiverAction;
+  if (type == kNewSession)
+    return CastInternalMessage::Type::kNewSession;
+
+  return CastInternalMessage::Type::kOther;
+}
+
+std::string CastInternalMessageTypeToString(CastInternalMessage::Type type) {
+  switch (type) {
+    case CastInternalMessage::Type::kClientConnect:
+      return kClientConnect;
+    case CastInternalMessage::Type::kAppMessage:
+      return kAppMessage;
+    case CastInternalMessage::Type::kReceiverAction:
+      return kReceiverAction;
+    case CastInternalMessage::Type::kNewSession:
+      return kNewSession;
+    case CastInternalMessage::Type::kOther:
+      NOTREACHED();
+      return "";
+  }
+  NOTREACHED();
+  return "";
+}
+
+// Possible types in a receiver_action message.
+constexpr char kReceiverActionTypeCast[] = "cast";
+constexpr char kReceiverActionTypeStop[] = "stop";
+
+base::ListValue CapabilitiesToListValue(uint8_t capabilities) {
+  base::ListValue value;
+  auto& storage = value.GetList();
+  if (capabilities & cast_channel::VIDEO_OUT)
+    storage.emplace_back("video_out");
+  if (capabilities & cast_channel::VIDEO_IN)
+    storage.emplace_back("video_in");
+  if (capabilities & cast_channel::AUDIO_OUT)
+    storage.emplace_back("audio_out");
+  if (capabilities & cast_channel::AUDIO_IN)
+    storage.emplace_back("audio_in");
+  if (capabilities & cast_channel::MULTIZONE_GROUP)
+    storage.emplace_back("multizone_group");
+  return value;
+}
+
+base::Value CreateReceiver(const MediaSinkInternal& sink,
+                           const std::string& hash_token) {
+  base::Value receiver(base::Value::Type::DICTIONARY);
+
+  std::string label = base::SHA1HashString(sink.sink().id() + hash_token);
+  base::Base64UrlEncode(label, base::Base64UrlEncodePolicy::OMIT_PADDING,
+                        &label);
+  receiver.SetKey("label", base::Value(label));
+
+  receiver.SetKey("friendlyName",
+                  base::Value(net::EscapeForHTML(sink.sink().name())));
+  receiver.SetKey("capabilities",
+                  CapabilitiesToListValue(sink.cast_data().capabilities));
+  receiver.SetKey("volume", base::Value());
+  receiver.SetKey("isActiveInput", base::Value());
+  receiver.SetKey("displayStatus", base::Value());
+
+  receiver.SetKey("receiverType", base::Value("cast"));
+  return receiver;
+}
+
+base::Value CreateReceiverActionMessage(const std::string& client_id,
+                                        const MediaSinkInternal& sink,
+                                        const std::string& hash_token,
+                                        const char* action_type) {
+  base::Value message(base::Value::Type::DICTIONARY);
+  message.SetKey("receiver", CreateReceiver(sink, hash_token));
+  message.SetKey("action", base::Value(action_type));
+
+  base::Value value(base::Value::Type::DICTIONARY);
+  value.SetKey("type", base::Value(CastInternalMessageTypeToString(
+                           CastInternalMessage::Type::kReceiverAction)));
+  value.SetKey("message", std::move(message));
+  value.SetKey("sequenceNumber", base::Value(-1));
+  value.SetKey("timeoutMillis", base::Value(0));
+  value.SetKey("clientId", base::Value(client_id));
+  return value;
+}
+
+base::Value CreateAppMessageBody(
+    const std::string& session_id,
+    const cast_channel::CastMessage& cast_message) {
+  // TODO(https://crbug.com/862532): Investigate whether it is possible to move
+  // instead of copying the contents of |cast_message|. Right now copying is
+  // done because the message is passed as a const ref at the
+  // CastSocket::Observer level.
+  base::Value message(base::Value::Type::DICTIONARY);
+  message.SetKey("sessionId", base::Value(session_id));
+  message.SetKey("namespaceName", base::Value(cast_message.namespace_()));
+  switch (cast_message.payload_type()) {
+    case cast_channel::CastMessage_PayloadType_STRING:
+      message.SetKey("message", base::Value(cast_message.payload_utf8()));
+      break;
+    case cast_channel::CastMessage_PayloadType_BINARY: {
+      const auto& payload = cast_message.payload_binary();
+      message.SetKey("message",
+                     base::Value(base::Value::BlobStorage(
+                         payload.front(), payload.front() + payload.size())));
+      break;
+    }
+    default:
+      NOTREACHED();
+      break;
+  }
+  return message;
+}
+
+}  // namespace
+
+// static
+std::unique_ptr<CastInternalMessage> CastInternalMessage::From(
+    base::Value message) {
+  if (!message.is_dict()) {
+    DVLOG(2) << "Failed to read JSON message: " << message;
+    return nullptr;
+  }
+
+  std::string str_type;
+  if (!GetString(message, "type", &str_type)) {
+    DVLOG(2) << "Missing type value, message: " << message;
+    return nullptr;
+  }
+
+  CastInternalMessage::Type message_type =
+      CastInternalMessageTypeFromString(str_type);
+  if (message_type == CastInternalMessage::Type::kOther) {
+    DVLOG(2) << __func__ << ": Unsupported message type: " << str_type
+             << ", message: " << message;
+    return nullptr;
+  }
+
+  std::string client_id;
+  if (!GetString(message, "clientId", &client_id)) {
+    DVLOG(2) << "Missing clientId, message: " << message;
+    return nullptr;
+  }
+
+  base::Value* message_body_value = message.FindKey("message");
+  if (!message_body_value ||
+      (!message_body_value->is_dict() && !message_body_value->is_string())) {
+    DVLOG(2) << "Missing message body, message: " << message;
+    return nullptr;
+  }
+
+  auto internal_message =
+      std::make_unique<CastInternalMessage>(message_type, client_id);
+
+  base::Value* sequence_number_value =
+      message.FindKeyOfType("sequenceNumber", base::Value::Type::INTEGER);
+  if (sequence_number_value)
+    internal_message->sequence_number = sequence_number_value->GetInt();
+
+  if (message_type == CastInternalMessage::Type::kAppMessage) {
+    if (!message_body_value->is_dict())
+      return nullptr;
+
+    if (!GetString(*message_body_value, "namespaceName",
+                   &internal_message->app_message_namespace) ||
+        !GetString(*message_body_value, "sessionId",
+                   &internal_message->app_message_session_id)) {
+      DVLOG(2) << "Missing namespace or session ID, message: " << message;
+      return nullptr;
+    }
+
+    base::Value* app_message_value = message_body_value->FindKey("message");
+    if (!app_message_value ||
+        (!app_message_value->is_dict() && !app_message_value->is_string())) {
+      DVLOG(2) << "Missing app message, message: " << message;
+      return nullptr;
+    }
+    internal_message->app_message_body = std::move(*app_message_value);
+  }
+
+  return internal_message;
+}
+
+CastInternalMessage::CastInternalMessage(CastInternalMessage::Type type,
+                                         const std::string& client_id)
+    : type(type), client_id(client_id) {}
+
+CastInternalMessage::~CastInternalMessage() = default;
+
+blink::mojom::PresentationConnectionMessagePtr
+CreatePresentationConnectionMessage(const base::Value& message) {
+  std::string str;
+  CHECK(base::JSONWriter::Write(message, &str));
+  return blink::mojom::PresentationConnectionMessage::NewMessage(str);
+}
+
+blink::mojom::PresentationConnectionMessagePtr CreateReceiverActionCastMessage(
+    const std::string& client_id,
+    const MediaSinkInternal& sink,
+    const std::string& hash_token) {
+  return CreatePresentationConnectionMessage(CreateReceiverActionMessage(
+      client_id, sink, hash_token, kReceiverActionTypeCast));
+}
+
+blink::mojom::PresentationConnectionMessagePtr CreateReceiverActionStopMessage(
+    const std::string& client_id,
+    const MediaSinkInternal& sink,
+    const std::string& hash_token) {
+  return CreatePresentationConnectionMessage(CreateReceiverActionMessage(
+      client_id, sink, hash_token, kReceiverActionTypeStop));
+}
+
+// static
+std::unique_ptr<CastSession> CastSession::From(
+    const MediaSinkInternal& sink,
+    const std::string& hash_token,
+    const base::Value& receiver_status) {
+  // There should be only 1 app on |receiver_status|.
+  const base::Value* app_list_value =
+      receiver_status.FindKeyOfType("applications", base::Value::Type::LIST);
+  if (!app_list_value || app_list_value->GetList().size() != 1) {
+    DVLOG(2) << "receiver_status does not contain exactly one app: "
+             << receiver_status;
+    return nullptr;
+  }
+
+  auto session = std::make_unique<CastSession>();
+
+  // Fill in mandatory Session fields.
+  const base::Value& app_value = app_list_value->GetList()[0];
+  if (!GetString(app_value, "sessionId", &session->session_id) ||
+      !GetString(app_value, "appId", &session->app_id) ||
+      !GetString(app_value, "transportId", &session->transport_id) ||
+      !GetString(app_value, "displayName", &session->display_name)) {
+    DVLOG(2) << "app_value missing mandatory fields: " << app_value;
+    return nullptr;
+  }
+
+  // Optional Session fields.
+  GetString(app_value, "statusText", &session->status);
+
+  base::Value receiver_value = CreateReceiver(sink, hash_token);
+  CopyValue(receiver_status, "volume", &receiver_value);
+  CopyValue(receiver_status, "isActiveInput", &receiver_value);
+
+  // Create |session->value|.
+  session->value = base::Value(base::Value::Type::DICTIONARY);
+  auto& session_value = session->value;
+  session_value.SetKey("sessionId", base::Value(session->session_id));
+  session_value.SetKey("appId", base::Value(session->app_id));
+  session_value.SetKey("transportId", base::Value(session->transport_id));
+  session_value.SetKey("receiver", std::move(receiver_value));
+
+  CopyValueWithDefault(app_value, "displayName", base::Value(""),
+                       &session_value);
+  CopyValueWithDefault(app_value, "senderApps", base::ListValue(),
+                       &session_value);
+  CopyValueWithDefault(app_value, "statusText", base::Value(), &session_value);
+  CopyValueWithDefault(app_value, "appImages", base::ListValue(),
+                       &session_value);
+  CopyValueWithDefault(app_value, "namespaces", base::ListValue(),
+                       &session_value);
+
+  const base::Value* namespaces_value =
+      app_value.FindKeyOfType("namespaces", base::Value::Type::LIST);
+  if (!namespaces_value || namespaces_value->GetList().empty()) {
+    // A session without namespaces is invalid, except for a multizone leader.
+    if (session->app_id != kMultizoneLeaderAppId)
+      return nullptr;
+  } else {
+    for (const auto& namespace_value : namespaces_value->GetList()) {
+      std::string message_namespace;
+      if (!namespace_value.is_dict() ||
+          !GetString(namespace_value, "name", &message_namespace))
+        return nullptr;
+
+      session->message_namespaces.insert(std::move(message_namespace));
+    }
+  }
+
+  session_value.SetKey("namespaces",
+                       namespaces_value ? namespaces_value->Clone()
+                                        : base::Value(base::Value::Type::LIST));
+
+  return session;
+}
+
+CastSession::CastSession() = default;
+CastSession::~CastSession() = default;
+
+// static
+std::string CastSession::GetRouteDescription(const CastSession& session) {
+  return !session.status.empty() ? session.status : session.display_name;
+}
+
+blink::mojom::PresentationConnectionMessagePtr CreateNewSessionMessage(
+    const CastSession& session,
+    const std::string& client_id) {
+  base::Value message(base::Value::Type::DICTIONARY);
+  message.SetKey("type", base::Value(CastInternalMessageTypeToString(
+                             CastInternalMessage::Type::kNewSession)));
+  message.SetKey("message", session.value.Clone());
+  message.SetKey("sequenceNumber", base::Value(-1));
+  message.SetKey("timeoutMillis", base::Value(0));
+  message.SetKey("clientId", base::Value(client_id));
+  return CreatePresentationConnectionMessage(message);
+}
+
+blink::mojom::PresentationConnectionMessagePtr CreateAppMessageAck(
+    const std::string& client_id,
+    int sequence_number) {
+  base::Value message(base::Value::Type::DICTIONARY);
+  message.SetKey("type", base::Value(CastInternalMessageTypeToString(
+                             CastInternalMessage::Type::kAppMessage)));
+  message.SetKey("message", base::Value());
+  message.SetKey("sequenceNumber", base::Value(sequence_number));
+  message.SetKey("timeoutMillis", base::Value(0));
+  message.SetKey("clientId", base::Value(client_id));
+  return CreatePresentationConnectionMessage(message);
+}
+
+blink::mojom::PresentationConnectionMessagePtr CreateAppMessage(
+    const std::string& session_id,
+    const std::string& client_id,
+    const cast_channel::CastMessage& cast_message) {
+  base::Value message(base::Value::Type::DICTIONARY);
+  message.SetKey("type", base::Value(CastInternalMessageTypeToString(
+                             CastInternalMessage::Type::kAppMessage)));
+  message.SetKey("message", CreateAppMessageBody(session_id, cast_message));
+  message.SetKey("sequenceNumber", base::Value(-1));
+  message.SetKey("timeoutMillis", base::Value(0));
+  message.SetKey("clientId", base::Value(client_id));
+  return CreatePresentationConnectionMessage(message);
+}
+
+}  // namespace media_router
diff --git a/chrome/browser/media/router/providers/cast/cast_internal_message_util.h b/chrome/browser/media/router/providers/cast/cast_internal_message_util.h
new file mode 100644
index 0000000..f1755f03
--- /dev/null
+++ b/chrome/browser/media/router/providers/cast/cast_internal_message_util.h
@@ -0,0 +1,123 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_MEDIA_ROUTER_PROVIDERS_CAST_CAST_INTERNAL_MESSAGE_UTIL_H_
+#define CHROME_BROWSER_MEDIA_ROUTER_PROVIDERS_CAST_CAST_INTERNAL_MESSAGE_UTIL_H_
+
+#include "base/containers/flat_set.h"
+#include "base/macros.h"
+#include "base/values.h"
+#include "third_party/blink/public/platform/modules/presentation/presentation.mojom.h"
+
+namespace cast_channel {
+class CastMessage;
+}
+
+namespace media_router {
+
+class MediaSinkInternal;
+
+// Represents a message sent or received by the Cast SDK via a
+// PresentationConnection.
+struct CastInternalMessage {
+  // TODO(crbug.com/809249): Add other types of messages.
+  enum class Type {
+    kClientConnect,   // Initial message sent by SDK client to connect to MRP.
+    kAppMessage,      // App messages to pass through between SDK client and the
+                      // receiver.
+    kReceiverAction,  // Message sent by MRP to inform SDK client of action.
+    kNewSession,      // Message sent by MRP to inform SDK client of new
+                      // session.
+    kOther            // All other types of messages which are not considered
+                      // part of communication with Cast SDK.
+  };
+
+  // Returns a CastInternalMessage for |message|, or nullptr is |message| is not
+  // a valid Cast internal message.
+  static std::unique_ptr<CastInternalMessage> From(base::Value message);
+
+  CastInternalMessage(Type type, const std::string& client_id);
+  ~CastInternalMessage();
+
+  Type type;
+  std::string client_id;
+  int sequence_number = -1;
+
+  // The following are set if |type| is |kAppMessage|.
+  std::string app_message_namespace;
+  std::string app_message_session_id;
+  base::Value app_message_body;
+
+  DISALLOW_COPY_AND_ASSIGN(CastInternalMessage);
+};
+
+// Represents a Cast session on a Cast device. Cast sessions are derived from
+// RECEIVER_STATUS messages sent by Cast devices.
+class CastSession {
+ public:
+  // Returns a CastSession from |receiver_status| message sent by |sink|, or
+  // nullptr if |receiver_status| is not a valid RECEIVER_STATUS message.
+  // |hash_token| is a per-profile value that is used to hash the sink ID.
+  static std::unique_ptr<CastSession> From(const MediaSinkInternal& sink,
+                                           const std::string& hash_token,
+                                           const base::Value& receiver_status);
+
+  // Returns a string that can be used as the description of the MediaRoute
+  // associated with this session.
+  static std::string GetRouteDescription(const CastSession& session);
+
+  CastSession();
+  ~CastSession();
+
+  // ID of the session.
+  std::string session_id;
+
+  // ID of the app in the session.
+  std::string app_id;
+
+  // ID used for communicating with the session over the Cast channel.
+  std::string transport_id;
+
+  // The set of accepted message namespaces. Must be non-empty, unless the
+  // session represents a multizone leader.
+  base::flat_set<std::string> message_namespaces;
+
+  // The human-readable name of the Cast application, for example, "YouTube".
+  // Mandatory.
+  std::string display_name;
+
+  // Descriptive text for the current application content, for example “My
+  // Wedding Slideshow”. May be empty.
+  std::string status;
+
+  // The dictionary representing this session, derived from |receiver_status|.
+  // For convenience, this is used for generating messages sent to the SDK that
+  // include the session value.
+  base::Value value;
+};
+
+// Utility methods for generating messages sent to the SDK.
+// |hash_token| is a per-profile value that is used to hash the sink ID.
+blink::mojom::PresentationConnectionMessagePtr CreateReceiverActionCastMessage(
+    const std::string& client_id,
+    const MediaSinkInternal& sink,
+    const std::string& hash_token);
+blink::mojom::PresentationConnectionMessagePtr CreateReceiverActionStopMessage(
+    const std::string& client_id,
+    const MediaSinkInternal& sink,
+    const std::string& hash_token);
+blink::mojom::PresentationConnectionMessagePtr CreateNewSessionMessage(
+    const CastSession& session,
+    const std::string& client_id);
+blink::mojom::PresentationConnectionMessagePtr CreateAppMessageAck(
+    const std::string& client_id,
+    int sequence_number);
+blink::mojom::PresentationConnectionMessagePtr CreateAppMessage(
+    const std::string& session_id,
+    const std::string& client_id,
+    const cast_channel::CastMessage& cast_message);
+
+}  // namespace media_router
+
+#endif  // CHROME_BROWSER_MEDIA_ROUTER_PROVIDERS_CAST_CAST_INTERNAL_MESSAGE_UTIL_H_
diff --git a/chrome/browser/media/router/providers/cast/cast_internal_message_util_unittest.cc b/chrome/browser/media/router/providers/cast/cast_internal_message_util_unittest.cc
new file mode 100644
index 0000000..ceed43cf
--- /dev/null
+++ b/chrome/browser/media/router/providers/cast/cast_internal_message_util_unittest.cc
@@ -0,0 +1,404 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/media/router/providers/cast/cast_internal_message_util.h"
+
+#include "base/json/json_reader.h"
+#include "chrome/browser/media/router/test/test_helper.h"
+#include "chrome/common/media_router/test/test_helper.h"
+#include "components/cast_channel/cast_test_util.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+namespace media_router {
+
+namespace {
+
+static constexpr char kReceiverIdToken[] = "token";
+
+std::unique_ptr<base::Value> ReceiverStatus() {
+  std::string receiver_status_str = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "displayName": "App display name",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "sessionId": "sessionId",
+        "statusText":"App status",
+        "transportId":"transportId"
+      }]
+  })";
+  return base::JSONReader::Read(receiver_status_str);
+}
+
+void ExpectNoCastSession(const MediaSinkInternal& sink,
+                         const std::string& receiver_status_str,
+                         const std::string& reason) {
+  auto receiver_status = base::JSONReader::Read(receiver_status_str);
+  ASSERT_TRUE(receiver_status);
+  auto session = CastSession::From(sink, kReceiverIdToken, *receiver_status);
+  EXPECT_FALSE(session) << "Shouldn't have created session because of "
+                        << reason;
+}
+
+void ExpectJSONMessagesEqual(const std::string& expected_message,
+                             const std::string& message) {
+  auto expected_message_value = base::JSONReader::Read(expected_message);
+  ASSERT_TRUE(expected_message_value);
+
+  auto message_value = base::JSONReader::Read(message);
+  ASSERT_TRUE(message_value);
+
+  EXPECT_EQ(*expected_message_value, *message_value);
+}
+
+void ExpectInvalidCastInternalMessage(const std::string& message_str,
+                                      const std::string& invalid_reason) {
+  auto message_value = base::JSONReader::Read(message_str);
+  ASSERT_TRUE(message_value);
+  EXPECT_FALSE(CastInternalMessage::From(std::move(*message_value)))
+      << "message expected to be invlaid: " << invalid_reason;
+}
+
+}  // namespace
+
+TEST(CastInternalMessageUtilTest, CastInternalMessageFromAppMessageString) {
+  std::string message_str = R"({
+    "type": "app_message",
+    "clientId": "12345",
+    "sequenceNumber": 999,
+    "message": {
+      "namespaceName": "urn:x-cast:com.google.foo",
+      "sessionId": "sessionId",
+      "message": { "foo": "bar" }
+    }
+  })";
+  auto message_value = base::JSONReader::Read(message_str);
+  ASSERT_TRUE(message_value);
+
+  auto message = CastInternalMessage::From(std::move(*message_value));
+  ASSERT_TRUE(message);
+  EXPECT_EQ(CastInternalMessage::Type::kAppMessage, message->type);
+  EXPECT_EQ("12345", message->client_id);
+  EXPECT_EQ(999, message->sequence_number);
+  EXPECT_EQ("urn:x-cast:com.google.foo", message->app_message_namespace);
+  EXPECT_EQ("sessionId", message->app_message_session_id);
+  base::Value message_body(base::Value::Type::DICTIONARY);
+  message_body.SetKey("foo", base::Value("bar"));
+  EXPECT_EQ(message_body, message->app_message_body);
+}
+
+TEST(CastInternalMessageUtilTest, CastInternalMessageFromClientConnectString) {
+  std::string message_str = R"({
+      "type": "client_connect",
+      "clientId": "12345",
+      "message": {}
+    })";
+  auto message_value = base::JSONReader::Read(message_str);
+  ASSERT_TRUE(message_value);
+
+  auto message = CastInternalMessage::From(std::move(*message_value));
+  ASSERT_TRUE(message);
+  EXPECT_EQ(CastInternalMessage::Type::kClientConnect, message->type);
+  EXPECT_EQ("12345", message->client_id);
+  EXPECT_EQ(-1, message->sequence_number);
+  EXPECT_TRUE(message->app_message_namespace.empty());
+  EXPECT_TRUE(message->app_message_session_id.empty());
+  EXPECT_EQ(base::Value(), message->app_message_body);
+}
+
+TEST(CastInternalMessageUtilTest, CastInternalMessageFromInvalidStrings) {
+  std::string unknown_type = R"({
+      "type": "some_unknown_type",
+      "clientId": "12345",
+      "message": {}
+    })";
+  ExpectInvalidCastInternalMessage(unknown_type, "unknown_type");
+
+  std::string missing_client_id = R"({
+      "type": "client_connect",
+      "message": {}
+    })";
+  ExpectInvalidCastInternalMessage(missing_client_id, "missing client ID");
+
+  std::string missing_message = R"({
+      "type": "client_connect",
+      "clientId": "12345"
+    })";
+  ExpectInvalidCastInternalMessage(missing_message, "missing message");
+
+  std::string app_message_missing_namespace = R"({
+    "type": "app_message",
+    "clientId": "12345",
+    "sequenceNumber": 999,
+    "message": {
+      "sessionId": "sessionId",
+      "message": { "foo": "bar" }
+    }
+  })";
+  ExpectInvalidCastInternalMessage(app_message_missing_namespace,
+                                   "missing namespace");
+
+  std::string app_message_missing_session_id = R"({
+    "type": "app_message",
+    "clientId": "12345",
+    "sequenceNumber": 999,
+    "message": {
+      "namespaceName": "urn:x-cast:com.google.foo",
+      "message": { "foo": "bar" }
+    }
+  })";
+  ExpectInvalidCastInternalMessage(app_message_missing_session_id,
+                                   "missing session ID");
+
+  std::string app_message_missing_message = R"({
+    "type": "app_message",
+    "clientId": "12345",
+    "sequenceNumber": 999,
+    "message": {
+      "namespaceName": "urn:x-cast:com.google.foo",
+      "sessionId": "sessionId"
+    }
+  })";
+  ExpectInvalidCastInternalMessage(app_message_missing_message,
+                                   "missing app message");
+}
+
+TEST(CastInternalMessageUtilTest, CastSessionFromReceiverStatusNoStatusText) {
+  MediaSinkInternal sink = CreateCastSink(1);
+  std::string receiver_status_str = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "displayName": "App display name",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "sessionId": "sessionId",
+        "transportId":"transportId"
+      }]
+  })";
+  auto receiver_status = base::JSONReader::Read(receiver_status_str);
+  ASSERT_TRUE(receiver_status);
+  auto session = CastSession::From(sink, kReceiverIdToken, *receiver_status);
+  ASSERT_TRUE(session);
+  EXPECT_EQ("sessionId", session->session_id);
+  EXPECT_EQ("ABCDEFGH", session->app_id);
+  EXPECT_EQ("transportId", session->transport_id);
+  base::flat_set<std::string> message_namespaces = {
+      "urn:x-cast:com.google.cast.media", "urn:x-cast:com.google.foo"};
+  EXPECT_EQ(message_namespaces, session->message_namespaces);
+  EXPECT_TRUE(session->value.is_dict());
+  EXPECT_EQ("App display name", CastSession::GetRouteDescription(*session));
+}
+
+TEST(CastInternalMessageUtilTest, CastSessionFromInvalidReceiverStatuses) {
+  MediaSinkInternal sink = CreateCastSink(1);
+  std::string missing_app_id = R"({
+      "applications": [{
+        "displayName": "App display name",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "sessionId": "sessionId",
+        "statusText":"App status",
+        "transportId":"transportId"
+      }]
+  })";
+  ExpectNoCastSession(sink, missing_app_id, "missing app id");
+
+  std::string missing_display_name = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "sessionId": "sessionId",
+        "statusText":"App status",
+        "transportId":"transportId"
+      }]
+  })";
+  ExpectNoCastSession(sink, missing_display_name, "missing display name");
+
+  std::string missing_namespaces = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "displayName": "App display name",
+        "namespaces": [],
+        "sessionId": "sessionId",
+        "statusText":"App status",
+        "transportId":"transportId"
+      }]
+  })";
+  ExpectNoCastSession(sink, missing_namespaces, "missing namespaces");
+
+  std::string missing_session_id = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "displayName": "App display name",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "statusText":"App status",
+        "transportId":"transportId"
+      }]
+  })";
+  ExpectNoCastSession(sink, missing_session_id, "missing session id");
+
+  std::string missing_transport_id = R"({
+      "applications": [{
+        "appId": "ABCDEFGH",
+        "displayName": "App display name",
+        "namespaces": [
+          {"name": "urn:x-cast:com.google.cast.media"},
+          {"name": "urn:x-cast:com.google.foo"}
+        ],
+        "sessionId": "sessionId",
+        "statusText":"App status"
+      }]
+  })";
+  ExpectNoCastSession(sink, missing_transport_id, "missing transport id");
+}
+
+TEST(CastInternalMessageUtilTest, CreateReceiverActionCastMessage) {
+  std::string client_id = "clientId";
+  MediaSinkInternal sink = CreateCastSink(1);
+  std::string expected_message = R"({
+     "clientId": "clientId",
+     "message": {
+        "action": "cast",
+        "receiver": {
+           "capabilities": [ "video_out", "audio_out" ],
+           "displayStatus": null,
+           "friendlyName": "friendly name 1",
+           "isActiveInput": null,
+           "label": "yYH_HCL9CKJFmvKJ9m3Une2cS8s",
+           "receiverType": "cast",
+           "volume": null
+        }
+     },
+     "sequenceNumber": -1,
+     "timeoutMillis": 0,
+     "type": "receiver_action"
+  })";
+
+  auto message =
+      CreateReceiverActionCastMessage(client_id, sink, kReceiverIdToken);
+  ExpectJSONMessagesEqual(expected_message, message->get_message());
+}
+
+TEST(CastInternalMessageUtilTest, CreateReceiverActionStopMessage) {
+  std::string client_id = "clientId";
+  MediaSinkInternal sink = CreateCastSink(1);
+  std::string expected_message = R"({
+     "clientId": "clientId",
+     "message": {
+        "action": "stop",
+        "receiver": {
+           "capabilities": [ "video_out", "audio_out" ],
+           "displayStatus": null,
+           "friendlyName": "friendly name 1",
+           "isActiveInput": null,
+           "label": "yYH_HCL9CKJFmvKJ9m3Une2cS8s",
+           "receiverType": "cast",
+           "volume": null
+        }
+     },
+     "sequenceNumber": -1,
+     "timeoutMillis": 0,
+     "type": "receiver_action"
+  })";
+
+  auto message =
+      CreateReceiverActionStopMessage(client_id, sink, kReceiverIdToken);
+  ExpectJSONMessagesEqual(expected_message, message->get_message());
+}
+
+TEST(CastInternalMessageUtilTest, CreateNewSessionMessage) {
+  MediaSinkInternal sink = CreateCastSink(1);
+  std::string client_id = "clientId";
+  auto receiver_status = ReceiverStatus();
+  ASSERT_TRUE(receiver_status);
+  auto session = CastSession::From(sink, kReceiverIdToken, *receiver_status);
+  ASSERT_TRUE(session);
+
+  std::string expected_message = R"({
+   "clientId": "clientId",
+   "message": {
+      "appId": "ABCDEFGH",
+      "appImages": [  ],
+      "displayName": "App display name",
+      "namespaces": [ {
+         "name": "urn:x-cast:com.google.cast.media"
+      }, {
+         "name": "urn:x-cast:com.google.foo"
+      } ],
+      "receiver": {
+         "capabilities": [ "video_out", "audio_out" ],
+         "displayStatus": null,
+         "friendlyName": "friendly name 1",
+         "isActiveInput": null,
+         "label": "yYH_HCL9CKJFmvKJ9m3Une2cS8s",
+         "receiverType": "cast",
+         "volume": null
+      },
+      "senderApps": [  ],
+      "sessionId": "sessionId",
+      "statusText": "App status",
+      "transportId": "transportId"
+   },
+   "sequenceNumber": -1,
+   "timeoutMillis": 0,
+   "type": "new_session"
+  })";
+
+  auto message = CreateNewSessionMessage(*session, client_id);
+  ExpectJSONMessagesEqual(expected_message, message->get_message());
+}
+
+TEST(CastInternalMessageUtilTest, CreateAppMessageAck) {
+  std::string client_id = "clientId";
+  int sequence_number = 12345;
+
+  std::string expected_message = R"({
+   "clientId": "clientId",
+   "message": null,
+   "sequenceNumber": 12345,
+   "timeoutMillis": 0,
+   "type": "app_message"
+  })";
+
+  auto message = CreateAppMessageAck(client_id, sequence_number);
+  ExpectJSONMessagesEqual(expected_message, message->get_message());
+}
+
+TEST(CastInternalMessageUtilTest, CreateAppMessage) {
+  std::string session_id = "sessionId";
+  std::string client_id = "clientId";
+  base::Value message_body(base::Value::Type::DICTIONARY);
+  message_body.SetKey("foo", base::Value("bar"));
+  cast_channel::CastMessage cast_message = cast_channel::CreateCastMessage(
+      "urn:x-cast:com.google.foo", message_body, "sourceId", "destinationId");
+
+  std::string expected_message = R"({
+   "clientId": "clientId",
+   "message": {
+      "message": "{\"foo\":\"bar\"}",
+      "namespaceName": "urn:x-cast:com.google.foo",
+      "sessionId": "sessionId"
+   },
+   "sequenceNumber": -1,
+   "timeoutMillis": 0,
+   "type": "app_message"
+  })";
+
+  auto message = CreateAppMessage(session_id, client_id, cast_message);
+  ExpectJSONMessagesEqual(expected_message, message->get_message());
+}
+
+}  // namespace media_router
diff --git a/chrome/browser/media/router/providers/cast/cast_media_route_provider_unittest.cc b/chrome/browser/media/router/providers/cast/cast_media_route_provider_unittest.cc
index b32a575..28ee791e 100644
--- a/chrome/browser/media/router/providers/cast/cast_media_route_provider_unittest.cc
+++ b/chrome/browser/media/router/providers/cast/cast_media_route_provider_unittest.cc
@@ -64,7 +64,7 @@
   EXPECT_CALL(app_discovery_service_, DoStartObservingMediaSinks(_)).Times(0);
   provider_->StartObservingMediaSinks(non_cast_source);
 
-  MediaSource::Id cast_source("cast:ABCDEFGH");
+  MediaSource::Id cast_source("cast:ABCDEFGH?clientId=123");
   EXPECT_CALL(app_discovery_service_, DoStartObservingMediaSinks(_));
   provider_->StartObservingMediaSinks(cast_source);
   EXPECT_FALSE(app_discovery_service_.callbacks().empty());
@@ -78,6 +78,7 @@
   media_sink_service_.AddOrUpdateSink(CreateCastSink(2));
   MediaSource::Id source_id(
       "cast:ABCDEFAB?capabilities=video_out,audio_out"
+      "&clientId=123"
       "&broadcastNamespace=namespace"
       "&broadcastMessage=message");
 
diff --git a/chrome/browser/media/webrtc/webrtc_event_log_manager_remote.cc b/chrome/browser/media/webrtc/webrtc_event_log_manager_remote.cc
index bcb86de..581135e 100644
--- a/chrome/browser/media/webrtc/webrtc_event_log_manager_remote.cc
+++ b/chrome/browser/media/webrtc/webrtc_event_log_manager_remote.cc
@@ -162,9 +162,17 @@
   // while destruction took place, thereby avoiding endless attempts to upload
   // the same file.
 
-  // |network_connection_tracker_| might already have posted a task back to us,
-  // but it will not run, because |task_runner_| has already been stopped.
-  network_connection_tracker_->RemoveNetworkConnectionObserver(this);
+  if (network_connection_tracker_) {
+    // * |network_connection_tracker_| might already have posted a task back
+    //   to us, but it will not run, because |task_runner_| has already been
+    //   stopped.
+    // * RemoveNetworkConnectionObserver() should generally be called on the
+    //   same thread as AddNetworkConnectionObserver(), but in this case it's
+    //   okay to remove on a separate thread, because this only happens during
+    //   Chrome shutdown, when no others tasks are running; there can be no
+    //   concurrently executing notification from the tracker.
+    network_connection_tracker_->RemoveNetworkConnectionObserver(this);
+  }
 }
 
 void WebRtcRemoteEventLogManager::SetNetworkConnectionTracker(
diff --git a/chrome/browser/metrics/subprocess_metrics_provider_unittest.cc b/chrome/browser/metrics/subprocess_metrics_provider_unittest.cc
index 7330132..9bb0117 100644
--- a/chrome/browser/metrics/subprocess_metrics_provider_unittest.cc
+++ b/chrome/browser/metrics/subprocess_metrics_provider_unittest.cc
@@ -6,6 +6,7 @@
 
 #include <memory>
 #include <string>
+#include <vector>
 
 #include "base/metrics/histogram.h"
 #include "base/metrics/histogram_flattener.h"
@@ -14,8 +15,12 @@
 #include "base/metrics/persistent_memory_allocator.h"
 #include "base/metrics/statistics_recorder.h"
 #include "content/public/test/test_browser_thread_bundle.h"
+#include "testing/gmock/include/gmock/gmock.h"
 #include "testing/gtest/include/gtest/gtest.h"
 
+using ::testing::UnorderedElementsAre;
+using ::testing::IsEmpty;
+
 namespace {
 
 const uint32_t TEST_MEMORY_SIZE = 64 << 10;  // 64 KiB
@@ -74,7 +79,7 @@
             std::string(), false));
   }
 
-  size_t GetSnapshotHistogramCount() {
+  std::vector<std::string> GetSnapshotHistogramNames() {
     // Merge the data from the allocator into the StatisticsRecorder.
     provider_.MergeHistogramDeltas();
 
@@ -85,7 +90,7 @@
     base::StatisticsRecorder::PrepareDeltas(true, base::Histogram::kNoFlags,
                                             base::Histogram::kNoFlags,
                                             &snapshot_manager);
-    return flattener.GetRecordedDeltaHistogramNames().size();
+    return flattener.GetRecordedDeltaHistogramNames();
   }
 
   void EnableRecording() { provider_.OnRecordingEnabled(); }
@@ -113,8 +118,7 @@
   DISALLOW_COPY_AND_ASSIGN(SubprocessMetricsProviderTest);
 };
 
-// Temporarily disabled until someone can troubleshoot http://crbug.com/863262
-TEST_F(SubprocessMetricsProviderTest, DISABLED_SnapshotMetrics) {
+TEST_F(SubprocessMetricsProviderTest, SnapshotMetrics) {
   base::HistogramBase* foo = base::Histogram::FactoryGet("foo", 1, 100, 10, 0);
   base::HistogramBase* bar = base::Histogram::FactoryGet("bar", 1, 100, 10, 0);
   base::HistogramBase* baz = base::Histogram::FactoryGet("baz", 1, 100, 10, 0);
@@ -130,25 +134,26 @@
                               CreateDuplicateAllocator(global_allocator.get()));
 
   // Recording should find the two histograms created in persistent memory.
-  EXPECT_EQ(2U, GetSnapshotHistogramCount());
+  EXPECT_THAT(GetSnapshotHistogramNames(), UnorderedElementsAre("foo", "bar"));
 
   // A second run should have nothing to produce.
-  EXPECT_EQ(0U, GetSnapshotHistogramCount());
+  EXPECT_THAT(GetSnapshotHistogramNames(), IsEmpty());
 
   // Create a new histogram and update existing ones. Should now report 3 items.
   baz->Add(1969);
   foo->Add(10);
   bar->Add(20);
-  EXPECT_EQ(3U, GetSnapshotHistogramCount());
+  EXPECT_THAT(GetSnapshotHistogramNames(),
+              UnorderedElementsAre("foo", "bar", "baz"));
 
   // Ensure that deregistering does a final merge of the data.
   foo->Add(10);
   bar->Add(20);
   DeregisterSubprocessAllocator(123);
-  EXPECT_EQ(2U, GetSnapshotHistogramCount());
+  EXPECT_THAT(GetSnapshotHistogramNames(), UnorderedElementsAre("foo", "bar"));
 
   // Further snapshots should be empty even if things have changed.
   foo->Add(10);
   bar->Add(20);
-  EXPECT_EQ(0U, GetSnapshotHistogramCount());
+  EXPECT_THAT(GetSnapshotHistogramNames(), IsEmpty());
 }
diff --git a/chrome/browser/password_manager/chrome_password_manager_client.cc b/chrome/browser/password_manager/chrome_password_manager_client.cc
index 86696fb..6f62c20 100644
--- a/chrome/browser/password_manager/chrome_password_manager_client.cc
+++ b/chrome/browser/password_manager/chrome_password_manager_client.cc
@@ -1084,7 +1084,10 @@
       PasswordAccessoryController::FromWebContents(web_contents());
   if (!accessory)
     return;  // No accessory needs change here.
-  accessory->RefreshSuggestionsForField(is_fillable, is_password_field);
+  accessory->RefreshSuggestionsForField(
+      password_manager_driver_bindings_.GetCurrentTargetFrame()
+          ->GetLastCommittedOrigin(),
+      is_fillable, is_password_field);
 #endif  // defined(OS_ANDROID)
 }
 
diff --git a/chrome/browser/password_manager/password_accessory_controller.cc b/chrome/browser/password_manager/password_accessory_controller.cc
index 6f5c332..11d7906 100644
--- a/chrome/browser/password_manager/password_accessory_controller.cc
+++ b/chrome/browser/password_manager/password_accessory_controller.cc
@@ -8,7 +8,6 @@
 
 #include "base/strings/utf_string_conversions.h"
 #include "chrome/browser/android/preferences/preferences_launcher.h"
-#include "chrome/browser/password_manager/password_accessory_view_interface.h"
 #include "chrome/browser/password_manager/password_generation_dialog_view_interface.h"
 #include "chrome/browser/ui/passwords/manage_passwords_view_utils.h"
 #include "chrome/grit/generated_resources.h"
@@ -22,8 +21,6 @@
 #include "content/public/browser/web_contents.h"
 #include "ui/base/l10n/l10n_util.h"
 
-#include "chrome/browser/android/preferences/preferences_launcher.h"
-
 using autofill::PasswordForm;
 using Item = PasswordAccessoryViewInterface::AccessoryItem;
 
@@ -210,10 +207,15 @@
 }
 
 void PasswordAccessoryController::RefreshSuggestionsForField(
+    const url::Origin& origin,
     bool is_fillable,
     bool is_password_field) {
   // TODO(crbug/853766): Record CTR metric.
-  SendViewItems(is_password_field);
+  view_->OnItemsAvailable(
+      CreateViewItems(origin,
+                      is_fillable ? origin_suggestions_[origin]
+                                  : std::vector<SuggestionElementData>(),
+                      is_password_field));
 }
 
 gfx::NativeView PasswordAccessoryController::container_view() const {
@@ -224,13 +226,11 @@
   return web_contents_->GetTopLevelNativeWindow();
 }
 
-void PasswordAccessoryController::SendViewItems(bool is_password_field) {
-  DCHECK(view_);
-  last_focused_field_was_for_passwords_ = is_password_field;
-  const url::Origin& origin =
-      web_contents_->GetFocusedFrame()->GetLastCommittedOrigin();
-  const std::vector<SuggestionElementData>& suggestions =
-      origin_suggestions_[origin];
+// static
+std::vector<Item> PasswordAccessoryController::CreateViewItems(
+    const url::Origin& origin,
+    const std::vector<SuggestionElementData>& suggestions,
+    bool is_password_field) {
   std::vector<Item> items;
   base::string16 passwords_title_str;
 
@@ -266,7 +266,5 @@
       IDS_PASSWORD_MANAGER_ACCESSORY_ALL_PASSWORDS_LINK);
   items.emplace_back(manage_passwords_title, manage_passwords_title, false,
                      Item::Type::OPTION);
-
-  // Notify the view about all just created elements.
-  view_->OnItemsAvailable(items);
+  return items;
 }
diff --git a/chrome/browser/password_manager/password_accessory_controller.h b/chrome/browser/password_manager/password_accessory_controller.h
index 9830285..2231d353 100644
--- a/chrome/browser/password_manager/password_accessory_controller.h
+++ b/chrome/browser/password_manager/password_accessory_controller.h
@@ -14,6 +14,7 @@
 #include "base/macros.h"
 #include "base/memory/weak_ptr.h"
 #include "base/strings/string16.h"
+#include "chrome/browser/password_manager/password_accessory_view_interface.h"
 #include "components/autofill/core/common/filling_status.h"
 #include "components/autofill/core/common/password_generation_util.h"
 #include "content/public/browser/web_contents_user_data.h"
@@ -28,7 +29,6 @@
 class PasswordManagerDriver;
 }  // namespace password_manager
 
-class PasswordAccessoryViewInterface;
 class PasswordGenerationDialogViewInterface;
 
 // The controller for the view located below the keyboard accessory.
@@ -91,8 +91,11 @@
   void OnFilledIntoFocusedField(autofill::FillingStatus status);
 
   // Makes sure, that all shown suggestions are appropriate for the currently
-  // focused field.
-  void RefreshSuggestionsForField(bool is_fillable, bool is_password_field);
+  // focused field and for fields that lost the focus. If a field lost focus,
+  // |is_fillable| will be false.
+  void RefreshSuggestionsForField(const url::Origin& origin,
+                                  bool is_fillable,
+                                  bool is_password_field);
 
   // The web page view containing the focused field.
   gfx::NativeView container_view() const;
@@ -129,10 +132,12 @@
       std::unique_ptr<PasswordAccessoryViewInterface> view,
       CreateDialogFactory create_dialog_callback);
 
-  // Creates the view items based on the |origin_suggestions_| and sends them to
-  // the view. If |is_password_field| is false, password suggestions won't be
-  // interactive.
-  void SendViewItems(bool is_password_field);
+  // Creates the view items based on the given |suggestions|.
+  // If |is_password_field| is false, password suggestions won't be interactive.
+  static std::vector<PasswordAccessoryViewInterface::AccessoryItem>
+  CreateViewItems(const url::Origin& origin,
+                  const std::vector<SuggestionElementData>& suggestions,
+                  bool is_password_field);
 
   // Contains the last set of credentials by origin.
   std::map<url::Origin, std::vector<SuggestionElementData>> origin_suggestions_;
diff --git a/chrome/browser/password_manager/password_accessory_controller_unittest.cc b/chrome/browser/password_manager/password_accessory_controller_unittest.cc
index 4d7ee08..c5603a5 100644
--- a/chrome/browser/password_manager/password_accessory_controller_unittest.cc
+++ b/chrome/browser/password_manager/password_accessory_controller_unittest.cc
@@ -245,7 +245,6 @@
         std::make_unique<StrictMock<MockPasswordAccessoryView>>(),
         mock_dialog_factory_.Get());
     NavigateAndCommit(GURL("https://example.com"));
-    FocusWebContentsOnMainFrame();
   }
 
   PasswordAccessoryController* controller() {
@@ -287,8 +286,10 @@
 
   controller()->SavePasswordsForOrigin({CreateEntry("Ben", "S3cur3").first},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 }
 
 TEST_F(PasswordAccessoryControllerTest, HintsToEmptyUserNames) {
@@ -304,8 +305,10 @@
 
   controller()->SavePasswordsForOrigin({CreateEntry("", "S3cur3").first},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 }
 
 TEST_F(PasswordAccessoryControllerTest, SortsAlphabeticalDuringTransform) {
@@ -338,8 +341,10 @@
       {CreateEntry("Ben", "S3cur3").first, CreateEntry("Zebra", "M3h").first,
        CreateEntry("Alf", "PWD").first, CreateEntry("Cat", "M1@u").first},
       url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 }
 
 TEST_F(PasswordAccessoryControllerTest, RepeatsSuggestionsForSameFrame) {
@@ -357,8 +362,10 @@
                                        url::Origin::Create(GURL(kExampleSite)));
 
   // Pretend that any input in the same frame was focused.
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_fillable=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_fillable=*/false);
 }
 
 TEST_F(PasswordAccessoryControllerTest, ProvidesEmptySuggestionsMessage) {
@@ -370,8 +377,10 @@
 
   controller()->SavePasswordsForOrigin({},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 }
 
 TEST_F(PasswordAccessoryControllerTest, RelaysAutomaticGenerationAvailable) {
@@ -439,8 +448,10 @@
   // This should result in the non-interactive suggestion expected above.
   controller()->SavePasswordsForOrigin({CreateEntry("Ben", "S3cur3").first},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 
   // Pretend that we focus a password field now: By triggering a refresh with
   // |is_password_field| set to true, all suggestions should become interactive.
@@ -452,8 +463,10 @@
                   MatchesItem(ASCIIToUTF16("S3cur3"), password_for_str("Ben"),
                               true, ItemType::SUGGESTION),
                   IsDivider(), MatchesOption(manage_passwords_str()))));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/true);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/true);
 }
 
 TEST_F(PasswordAccessoryControllerTest, CachesIsReplacedByNewPasswords) {
@@ -467,8 +480,10 @@
                   IsDivider(), MatchesOption(manage_passwords_str()))));
   controller()->SavePasswordsForOrigin({CreateEntry("Ben", "S3cur3").first},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
 
   EXPECT_CALL(*view(),
               OnItemsAvailable(ElementsAre(
@@ -480,6 +495,38 @@
                   IsDivider(), MatchesOption(manage_passwords_str()))));
   controller()->SavePasswordsForOrigin({CreateEntry("Alf", "M3lm4k").first},
                                        url::Origin::Create(GURL(kExampleSite)));
-  controller()->RefreshSuggestionsForField(/*is_fillable=*/true,
-                                           /*is_password_field=*/false);
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
+}
+
+TEST_F(PasswordAccessoryControllerTest, UnfillableFieldClearsSuggestions) {
+  EXPECT_CALL(*view(),
+              OnItemsAvailable(ElementsAre(
+                  MatchesLabel(passwords_title_str(kExampleDomain)),
+                  MatchesItem(ASCIIToUTF16("Ben"), ASCIIToUTF16("Ben"), false,
+                              ItemType::SUGGESTION),
+                  MatchesItem(ASCIIToUTF16("S3cur3"), password_for_str("Ben"),
+                              true, ItemType::NON_INTERACTIVE_SUGGESTION),
+                  IsDivider(), MatchesOption(manage_passwords_str()))));
+  // Set any, non-empty password list and pretend a username field was focused.
+  // This should result in the non-emtpy suggestions expected above.
+  controller()->SavePasswordsForOrigin({CreateEntry("Ben", "S3cur3").first},
+                                       url::Origin::Create(GURL(kExampleSite)));
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/true,
+      /*is_password_field=*/false);
+
+  // Pretend that the focus was lost or moved to an unfillable field. Now, only
+  // the empty state message should be sent.
+  EXPECT_CALL(*view(),
+              OnItemsAvailable(ElementsAre(
+                  MatchesLabel(passwords_empty_str(kExampleDomain)),
+                  IsDivider(), MatchesOption(manage_passwords_str()))));
+  controller()->RefreshSuggestionsForField(
+      url::Origin::Create(GURL(kExampleSite)),
+      /*is_fillable=*/false,
+      /*is_password_field=*/false);  // Unused.
 }
diff --git a/chrome/browser/picture_in_picture/picture_in_picture_window_controller_browsertest.cc b/chrome/browser/picture_in_picture/picture_in_picture_window_controller_browsertest.cc
index 8b46523..5ad7d13e 100644
--- a/chrome/browser/picture_in_picture/picture_in_picture_window_controller_browsertest.cc
+++ b/chrome/browser/picture_in_picture/picture_in_picture_window_controller_browsertest.cc
@@ -916,4 +916,27 @@
 
 #endif  // defined(OS_LINUX) && !defined(OS_CHROMEOS)
 
+// Tests that the Picture-in-Picture state is properly updated when the window
+// is closed at a system level.
+IN_PROC_BROWSER_TEST_F(PictureInPictureWindowControllerBrowserTest,
+                       CloseWindowNotifiesController) {
+  LoadTabAndEnterPictureInPicture(browser());
+
+  content::WebContents* active_web_contents =
+      browser()->tab_strip_model()->GetActiveWebContents();
+
+  OverlayWindowViews* overlay_window = static_cast<OverlayWindowViews*>(
+      window_controller()->GetWindowForTesting());
+  ASSERT_TRUE(overlay_window);
+  ASSERT_TRUE(overlay_window->IsVisible());
+
+  // Simulate closing from the system.
+  overlay_window->OnNativeWidgetDestroyed();
+
+  bool in_picture_in_picture = false;
+  ASSERT_TRUE(ExecuteScriptAndExtractBool(
+      active_web_contents, "isInPictureInPicture();", &in_picture_in_picture));
+  EXPECT_FALSE(in_picture_in_picture);
+}
+
 #endif  // !defined(OS_ANDROID)
diff --git a/chrome/browser/policy/configuration_policy_handler_list_factory.cc b/chrome/browser/policy/configuration_policy_handler_list_factory.cc
index 18f059e..2d0e643 100644
--- a/chrome/browser/policy/configuration_policy_handler_list_factory.cc
+++ b/chrome/browser/policy/configuration_policy_handler_list_factory.cc
@@ -794,11 +794,9 @@
 #endif
 
 #if !defined(OS_ANDROID)
-#if !defined(OS_CHROMEOS)
   { key::kRelaunchNotification,
     prefs::kRelaunchNotification,
     base::Value::Type::INTEGER },
-#endif  // !defined(OS_CHROMEOS)
   { key::kRelaunchNotificationPeriod,
     prefs::kRelaunchNotificationPeriod,
     base::Value::Type::INTEGER },
diff --git a/chrome/browser/prefs/browser_prefs.cc b/chrome/browser/prefs/browser_prefs.cc
index 083de60..29745cd 100644
--- a/chrome/browser/prefs/browser_prefs.cc
+++ b/chrome/browser/prefs/browser_prefs.cc
@@ -28,6 +28,7 @@
 #include "chrome/browser/media/media_device_id_salt.h"
 #include "chrome/browser/media/media_engagement_service.h"
 #include "chrome/browser/media/media_storage_id_salt.h"
+#include "chrome/browser/media/router/media_router_feature.h"
 #include "chrome/browser/media/webrtc/media_capture_devices_dispatcher.h"
 #include "chrome/browser/media/webrtc/media_stream_devices_controller.h"
 #include "chrome/browser/metrics/chrome_metrics_service_client.h"
@@ -651,6 +652,7 @@
   InstantService::RegisterProfilePrefs(registry);
   gcm::GCMChannelStatusSyncer::RegisterProfilePrefs(registry);
   gcm::RegisterProfilePrefs(registry);
+  media_router::RegisterProfilePrefs(registry);
   ntp_tiles::CustomLinksManagerImpl::RegisterProfilePrefs(registry);
   StartupBrowserCreator::RegisterProfilePrefs(registry);
 #endif
diff --git a/chrome/browser/profiles/gaia_info_update_service.cc b/chrome/browser/profiles/gaia_info_update_service.cc
index c15d673..ec2e2e2 100644
--- a/chrome/browser/profiles/gaia_info_update_service.cc
+++ b/chrome/browser/profiles/gaia_info_update_service.cc
@@ -16,7 +16,7 @@
 #include "chrome/browser/profiles/profile_manager.h"
 #include "chrome/browser/profiles/profile_metrics.h"
 #include "chrome/browser/profiles/profiles_state.h"
-#include "chrome/browser/signin/signin_manager_factory.h"
+#include "chrome/browser/signin/identity_manager_factory.h"
 #include "chrome/common/pref_names.h"
 #include "components/prefs/pref_service.h"
 #include "components/signin/core/browser/account_tracker_service.h"
@@ -39,9 +39,9 @@
 
 GAIAInfoUpdateService::GAIAInfoUpdateService(Profile* profile)
     : profile_(profile) {
-  SigninManagerBase* signin_manager =
-      SigninManagerFactory::GetForProfile(profile_);
-  signin_manager->AddObserver(this);
+  identity::IdentityManager* identity_manager =
+      IdentityManagerFactory::GetForProfile(profile_);
+  identity_manager->AddObserver(this);
 
   PrefService* prefs = profile_->GetPrefs();
   last_updated_ = base::Time::FromInternalValue(
@@ -55,9 +55,9 @@
 
 void GAIAInfoUpdateService::Update() {
   // The user must be logged in.
-  SigninManagerBase* signin_manager =
-      SigninManagerFactory::GetForProfile(profile_);
-  if (!signin_manager->IsAuthenticated())
+  identity::IdentityManager* identity_manager =
+      IdentityManagerFactory::GetForProfile(profile_);
+  if (!identity_manager->HasPrimaryAccount())
     return;
 
   if (profile_image_downloader_)
@@ -173,9 +173,9 @@
 void GAIAInfoUpdateService::Shutdown() {
   timer_.Stop();
   profile_image_downloader_.reset();
-  SigninManagerBase* signin_manager =
-      SigninManagerFactory::GetForProfile(profile_);
-  signin_manager->RemoveObserver(this);
+  identity::IdentityManager* identity_manager =
+      IdentityManagerFactory::GetForProfile(profile_);
+  identity_manager->RemoveObserver(this);
 
   // OK to reset |profile_| pointer here because GAIAInfoUpdateService will not
   // access it again.  This pointer is also used to implement the delegate for
@@ -208,12 +208,12 @@
   timer_.Start(FROM_HERE, delta, this, &GAIAInfoUpdateService::Update);
 }
 
-void GAIAInfoUpdateService::GoogleSigninSucceeded(const std::string& account_id,
-                                                  const std::string& username) {
-  OnUsernameChanged(username);
+void GAIAInfoUpdateService::OnPrimaryAccountSet(
+    const AccountInfo& primary_account_info) {
+  OnUsernameChanged(primary_account_info.gaia);
 }
 
-void GAIAInfoUpdateService::GoogleSignedOut(const std::string& account_id,
-                                            const std::string& username) {
+void GAIAInfoUpdateService::OnPrimaryAccountCleared(
+    const AccountInfo& previous_primary_account_info) {
   OnUsernameChanged(std::string());
 }
diff --git a/chrome/browser/profiles/gaia_info_update_service.h b/chrome/browser/profiles/gaia_info_update_service.h
index 5c84e89d..4128577 100644
--- a/chrome/browser/profiles/gaia_info_update_service.h
+++ b/chrome/browser/profiles/gaia_info_update_service.h
@@ -14,7 +14,7 @@
 #include "chrome/browser/profiles/profile_downloader.h"
 #include "chrome/browser/profiles/profile_downloader_delegate.h"
 #include "components/keyed_service/core/keyed_service.h"
-#include "components/signin/core/browser/signin_manager.h"
+#include "services/identity/public/cpp/identity_manager.h"
 
 class Profile;
 class ProfileDownloader;
@@ -23,7 +23,7 @@
 // The results are saved in the profile info cache.
 class GAIAInfoUpdateService : public KeyedService,
                               public ProfileDownloaderDelegate,
-                              public SigninManagerBase::Observer {
+                              public identity::IdentityManager::Observer {
  public:
   explicit GAIAInfoUpdateService(Profile* profile);
   ~GAIAInfoUpdateService() override;
@@ -54,11 +54,10 @@
   void OnUsernameChanged(const std::string& username);
   void ScheduleNextUpdate();
 
-  // Overridden from SigninManagerBase::Observer:
-  void GoogleSigninSucceeded(const std::string& account_id,
-                             const std::string& username) override;
-  void GoogleSignedOut(const std::string& account_id,
-                       const std::string& username) override;
+  // Overridden from identity::IdentityManager::Observer:
+  void OnPrimaryAccountSet(const AccountInfo& primary_account_info) override;
+  void OnPrimaryAccountCleared(
+      const AccountInfo& previous_primary_account_info) override;
 
   Profile* profile_;
   std::unique_ptr<ProfileDownloader> profile_image_downloader_;
diff --git a/chrome/browser/profiles/gaia_info_update_service_factory.cc b/chrome/browser/profiles/gaia_info_update_service_factory.cc
index 9edd14cd..b607b97 100644
--- a/chrome/browser/profiles/gaia_info_update_service_factory.cc
+++ b/chrome/browser/profiles/gaia_info_update_service_factory.cc
@@ -6,7 +6,7 @@
 
 #include "chrome/browser/profiles/gaia_info_update_service.h"
 #include "chrome/browser/profiles/profile.h"
-#include "chrome/browser/signin/signin_manager_factory.h"
+#include "chrome/browser/signin/identity_manager_factory.h"
 #include "chrome/common/pref_names.h"
 #include "components/keyed_service/content/browser_context_dependency_manager.h"
 #include "components/pref_registry/pref_registry_syncable.h"
@@ -15,7 +15,7 @@
     : BrowserContextKeyedServiceFactory(
         "GAIAInfoUpdateService",
         BrowserContextDependencyManager::GetInstance()) {
-  DependsOn(SigninManagerFactory::GetInstance());
+  DependsOn(IdentityManagerFactory::GetInstance());
 }
 
 GAIAInfoUpdateServiceFactory::~GAIAInfoUpdateServiceFactory() {}
diff --git a/chrome/browser/profiles/gaia_info_update_service_unittest.cc b/chrome/browser/profiles/gaia_info_update_service_unittest.cc
index c4bda264..ed21e6c5 100644
--- a/chrome/browser/profiles/gaia_info_update_service_unittest.cc
+++ b/chrome/browser/profiles/gaia_info_update_service_unittest.cc
@@ -19,6 +19,7 @@
 #include "chrome/browser/profiles/profiles_state.h"
 #include "chrome/browser/signin/account_tracker_service_factory.h"
 #include "chrome/browser/signin/chrome_signin_client_factory.h"
+#include "chrome/browser/signin/identity_manager_factory.h"
 #include "chrome/browser/signin/signin_manager_factory.h"
 #include "chrome/browser/signin/test_signin_client_builder.h"
 #include "chrome/common/pref_names.h"
@@ -29,6 +30,7 @@
 #include "components/signin/core/browser/account_tracker_service.h"
 #include "components/signin/core/browser/signin_pref_names.h"
 #include "components/sync_preferences/pref_service_syncable.h"
+#include "services/identity/public/cpp/identity_test_utils.h"
 #include "testing/gmock/include/gmock/gmock.h"
 #include "ui/gfx/image/image.h"
 #include "ui/gfx/image/image_unittest_util.h"
@@ -87,9 +89,9 @@
 
   Profile* CreateProfile(const std::string& name) {
     TestingProfile::TestingFactories testing_factories;
-    testing_factories.push_back(std::make_pair(
-        ChromeSigninClientFactory::GetInstance(),
-        signin::BuildTestSigninClient));
+    testing_factories.push_back(
+        std::make_pair(ChromeSigninClientFactory::GetInstance(),
+                       signin::BuildTestSigninClient));
     Profile* profile = testing_profile_manager_.CreateTestingProfile(
         name, std::unique_ptr<sync_preferences::PrefServiceSyncable>(),
         base::UTF8ToUTF16(name), 0, std::string(), testing_factories);
@@ -301,9 +303,9 @@
 #if !defined(OS_CHROMEOS)
 
 TEST_F(GAIAInfoUpdateServiceTest, LogOut) {
-  SigninManager* signin_manager =
-      SigninManagerFactory::GetForProfile(profile());
-  signin_manager->SetAuthenticatedAccountInfo("gaia_id", "pat@example.com");
+  identity::SetPrimaryAccount(SigninManagerFactory::GetForProfile(profile()),
+                              IdentityManagerFactory::GetForProfile(profile()),
+                              "pat@example.com");
   base::string16 gaia_name = base::UTF8ToUTF16("Pat Foo");
 
   ASSERT_EQ(1u, storage()->GetNumberOfProfiles());
@@ -319,8 +321,9 @@
   EXPECT_FALSE(service()->GetCachedPictureURL().empty());
 
   // Log out.
-  signin_manager->SignOut(signin_metrics::SIGNOUT_TEST,
-                          signin_metrics::SignoutDelete::IGNORE_METRIC);
+  identity::ClearPrimaryAccount(
+      SigninManagerFactory::GetForProfile(profile()),
+      IdentityManagerFactory::GetForProfile(profile()));
   // Verify that the GAIA name and picture, and picture URL are unset.
   EXPECT_TRUE(entry->GetGAIAName().empty());
   EXPECT_EQ(nullptr, entry->GetGAIAPicture());
@@ -330,11 +333,9 @@
 TEST_F(GAIAInfoUpdateServiceTest, LogIn) {
   // Log in.
   EXPECT_CALL(*service(), Update());
-  AccountTrackerServiceFactory::GetForProfile(profile())
-      ->SeedAccountInfo("gaia_id", "pat@example.com");
-  SigninManager* signin_manager =
-      SigninManagerFactory::GetForProfile(profile());
-  signin_manager->OnExternalSigninCompleted("pat@example.com");
+  identity::SetPrimaryAccount(SigninManagerFactory::GetForProfile(profile()),
+                              IdentityManagerFactory::GetForProfile(profile()),
+                              "pat@example.com");
 }
 
 #endif
diff --git a/chrome/browser/profiles/profile.cc b/chrome/browser/profiles/profile.cc
index e8f837f..5e93376 100644
--- a/chrome/browser/profiles/profile.cc
+++ b/chrome/browser/profiles/profile.cc
@@ -273,7 +273,12 @@
 
 bool Profile::IsSyncAllowed() {
   if (ProfileSyncServiceFactory::HasProfileSyncService(this)) {
-    return ProfileSyncServiceFactory::GetForProfile(this)->IsSyncAllowed();
+    browser_sync::ProfileSyncService* sync_service =
+        ProfileSyncServiceFactory::GetForProfile(this);
+    return !sync_service->HasDisableReason(
+               syncer::SyncService::DISABLE_REASON_PLATFORM_OVERRIDE) &&
+           !sync_service->HasDisableReason(
+               syncer::SyncService::DISABLE_REASON_ENTERPRISE_POLICY);
   }
 
   // No ProfileSyncService created yet - we don't want to create one, so just
diff --git a/chrome/browser/profiles/profile_browsertest.cc b/chrome/browser/profiles/profile_browsertest.cc
index 69270d03..deec95b 100644
--- a/chrome/browser/profiles/profile_browsertest.cc
+++ b/chrome/browser/profiles/profile_browsertest.cc
@@ -9,6 +9,8 @@
 #include <memory>
 
 #include "base/command_line.h"
+#include "base/files/file_path.h"
+#include "base/files/file_path_watcher.h"
 #include "base/files/file_util.h"
 #include "base/files/scoped_temp_dir.h"
 #include "base/json/json_reader.h"
@@ -20,6 +22,7 @@
 #include "base/sequenced_task_runner.h"
 #include "base/synchronization/waitable_event.h"
 #include "base/task_scheduler/task_scheduler.h"
+#include "base/test/bind_test_util.h"
 #include "base/test/scoped_feature_list.h"
 #include "base/threading/thread_restrictions.h"
 #include "base/values.h"
@@ -35,6 +38,7 @@
 #include "chrome/browser/ui/tabs/tab_strip_model.h"
 #include "chrome/common/chrome_constants.h"
 #include "chrome/common/chrome_features.h"
+#include "chrome/common/chrome_paths.h"
 #include "chrome/common/pref_names.h"
 #include "chrome/test/base/in_process_browser_test.h"
 #include "chrome/test/base/ui_test_utils.h"
@@ -846,3 +850,157 @@
       net::LOAD_ONLY_FROM_CACHE);
   url_fetcher_delegate3.WaitForCompletion();
 }
+
+namespace {
+
+// Watches for the destruction of the specified path (Which, in the tests that
+// use it, is typically a directory), and expects the parent directory not to be
+// deleted.
+//
+// This is used the the media cache deletion tests, so handles all the possible
+// orderings of events that could happen:
+//
+// * In PRE_* tests, the media cache could deleted before the test completes, by
+// the task posted on Profile / isolated app URLRequestContext creation.
+//
+// * In the followup test, the media cache could be deleted by the off-thread
+// delete media cache task before the FileDestructionWatcher starts watching for
+// deletion, or even before it's created.
+//
+// * In the followup test, the media cache could be deleted after the
+// FileDestructionWatcher starts watching.
+//
+// It also may be possible to get a notification of the media cache being
+// created from the the previous test, so this allows multiple watch events to
+// happen, before the path is actually deleted.
+//
+// The public methods are called on the UI thread, the private ones called on a
+// separate SequencedTaskRunner.
+class FileDestructionWatcher {
+ public:
+  explicit FileDestructionWatcher(const base::FilePath& watched_file_path)
+      : watched_file_path_(watched_file_path) {
+    DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+  }
+
+  void WaitForDestruction() {
+    DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+    DCHECK(!watcher_);
+    base::CreateSequencedTaskRunnerWithTraits({base::MayBlock()})
+        ->PostTask(FROM_HERE,
+                   base::BindOnce(&FileDestructionWatcher::StartWatchingPath,
+                                  base::Unretained(this)));
+    run_loop_.Run();
+    // The watcher should be destroyed before quitting the run loop, once the
+    // file has been destroyed.
+    DCHECK(!watcher_);
+
+    // Double check that the file was destroyed, and that the parent directory
+    // was not.
+    base::ScopedAllowBlockingForTesting allow_blocking;
+    EXPECT_FALSE(base::PathExists(watched_file_path_));
+    EXPECT_TRUE(base::PathExists(watched_file_path_.DirName()));
+  }
+
+ private:
+  void StartWatchingPath() {
+    DCHECK(!watcher_);
+    watcher_ = std::make_unique<base::FilePathWatcher>();
+    // Start watching before checking if the file exists, as the file could be
+    // destroyed between the existence check and when we start watching, if the
+    // order were reversed.
+    EXPECT_TRUE(watcher_->Watch(
+        watched_file_path_, false /* recursive */,
+        base::BindRepeating(&FileDestructionWatcher::OnPathChanged,
+                            base::Unretained(this))));
+    CheckIfPathExists();
+  }
+
+  void OnPathChanged(const base::FilePath& path, bool error) {
+    EXPECT_EQ(watched_file_path_, path);
+    EXPECT_FALSE(error);
+    CheckIfPathExists();
+  }
+
+  // Checks if the path exists, and if so, destroys the watcher and quits
+  // |run_loop_|.
+  void CheckIfPathExists() {
+    if (!base::PathExists(watched_file_path_)) {
+      watcher_.reset();
+      run_loop_.Quit();
+      return;
+    }
+  }
+
+  base::RunLoop run_loop_;
+  const base::FilePath watched_file_path_;
+
+  // Created and destroyed off of the UI thread, on the sequence used to watch
+  // for changes.
+  std::unique_ptr<base::FilePathWatcher> watcher_;
+
+  DISALLOW_COPY_AND_ASSIGN(FileDestructionWatcher);
+};
+
+}  // namespace
+
+// Create a media cache file, and make sure it's deleted by the time the next
+// test runs.
+IN_PROC_BROWSER_TEST_F(ProfileWithoutMediaCacheBrowserTest,
+                       PRE_DeleteMediaCache) {
+  base::FilePath media_cache_path =
+      browser()->profile()->GetPath().Append(chrome::kMediaCacheDirname);
+
+  base::ScopedAllowBlockingForTesting allow_blocking;
+  EXPECT_TRUE(base::CreateDirectory(media_cache_path));
+  std::string data = "foo";
+  base::WriteFile(media_cache_path.AppendASCII("foo"), data.c_str(),
+                  data.size());
+}
+
+IN_PROC_BROWSER_TEST_F(ProfileWithoutMediaCacheBrowserTest, DeleteMediaCache) {
+  base::FilePath media_cache_path =
+      browser()->profile()->GetPath().Append(chrome::kMediaCacheDirname);
+
+  base::ScopedAllowBlockingForTesting allow_blocking;
+
+  FileDestructionWatcher destruction_watcher(media_cache_path);
+  destruction_watcher.WaitForDestruction();
+}
+
+// Create a media cache file, and make sure it's deleted by initializing an
+// extension browser context.
+IN_PROC_BROWSER_TEST_F(ProfileWithoutMediaCacheBrowserTest,
+                       PRE_DeleteIsolatedAppMediaCache) {
+  scoped_refptr<const extensions::Extension> app =
+      BuildTestApp(browser()->profile());
+  content::StoragePartition* extension_partition =
+      content::BrowserContext::GetStoragePartitionForSite(
+          browser()->profile(),
+          extensions::Extension::GetBaseURLFromExtensionId(app->id()));
+
+  base::FilePath extension_media_cache_path =
+      extension_partition->GetPath().Append(chrome::kMediaCacheDirname);
+
+  base::ScopedAllowBlockingForTesting allow_blocking;
+  EXPECT_TRUE(base::CreateDirectory(extension_media_cache_path));
+  std::string data = "foo";
+  base::WriteFile(extension_media_cache_path.AppendASCII("foo"), data.c_str(),
+                  data.size());
+}
+
+IN_PROC_BROWSER_TEST_F(ProfileWithoutMediaCacheBrowserTest,
+                       DeleteIsolatedAppMediaCache) {
+  scoped_refptr<const extensions::Extension> app =
+      BuildTestApp(browser()->profile());
+  content::StoragePartition* extension_partition =
+      content::BrowserContext::GetStoragePartitionForSite(
+          browser()->profile(),
+          extensions::Extension::GetBaseURLFromExtensionId(app->id()));
+
+  base::FilePath extension_media_cache_path =
+      extension_partition->GetPath().Append(chrome::kMediaCacheDirname);
+
+  FileDestructionWatcher destruction_watcher(extension_media_cache_path);
+  destruction_watcher.WaitForDestruction();
+}
diff --git a/chrome/browser/profiles/profile_impl_io_data.cc b/chrome/browser/profiles/profile_impl_io_data.cc
index 6dd54b7..b8c9017 100644
--- a/chrome/browser/profiles/profile_impl_io_data.cc
+++ b/chrome/browser/profiles/profile_impl_io_data.cc
@@ -12,6 +12,8 @@
 #include "base/bind.h"
 #include "base/command_line.h"
 #include "base/feature_list.h"
+#include "base/files/file_path.h"
+#include "base/files/file_util.h"
 #include "base/logging.h"
 #include "base/macros.h"
 #include "base/metrics/field_trial.h"
@@ -19,6 +21,7 @@
 #include "base/stl_util.h"
 #include "base/strings/string_util.h"
 #include "base/task_scheduler/post_task.h"
+#include "base/task_scheduler/task_traits.h"
 #include "base/time/default_clock.h"
 #include "build/build_config.h"
 #include "chrome/browser/browser_process.h"
@@ -113,6 +116,19 @@
   return net::CACHE_BACKEND_DEFAULT;
 }
 
+void MaybeDeleteMediaCache(const base::FilePath& media_cache_path) {
+  if (!base::FeatureList::IsEnabled(features::kUseSameCacheForMedia) ||
+      media_cache_path.empty()) {
+    return;
+  }
+  base::PostTaskWithTraits(
+      FROM_HERE,
+      {base::TaskPriority::BACKGROUND, base::MayBlock(),
+       base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
+      base::BindOnce(base::IgnoreResult(&base::DeleteFile), media_cache_path,
+                     true /* recursive */));
+}
+
 }  // namespace
 
 using content::BrowserThread;
@@ -522,6 +538,8 @@
   InitializeExtensionsRequestContext(profile_params);
 #endif
 
+  MaybeDeleteMediaCache(lazy_params_->media_cache_path);
+
   // Create a media request context based on the main context, but using a
   // media cache.  It shares the same job factory as the main context.
   StoragePartitionDescriptor details(profile_path_, false);
@@ -555,6 +573,11 @@
         protocol_handler_interceptor,
     content::ProtocolHandlerMap* protocol_handlers,
     content::URLRequestInterceptorScopedVector request_interceptors) const {
+  if (!partition_descriptor.in_memory) {
+    MaybeDeleteMediaCache(
+        partition_descriptor.path.Append(chrome::kMediaCacheDirname));
+  }
+
   // Copy most state from the main context.
   AppRequestContext* context = new AppRequestContext();
   context->CopyFrom(main_context);
diff --git a/chrome/browser/resources/print_preview/new/destination_dialog.html b/chrome/browser/resources/print_preview/new/destination_dialog.html
index 02ea28f..eaf8b76 100644
--- a/chrome/browser/resources/print_preview/new/destination_dialog.html
+++ b/chrome/browser/resources/print_preview/new/destination_dialog.html
@@ -85,29 +85,35 @@
         width: 24px;
       }
 
-      :host #cloudprintPromo .close-button {
+      :host #cloudprintPromo button {
         -webkit-margin-start: 12px;
+        background-color: #f5f5f5;
         background-image: -webkit-image-set(
           url(chrome://theme/IDR_CLOSE_DIALOG) 1x,
           url(chrome://theme/IDR_CLOSE_DIALOG@2x) 2x);
         background-repeat: no-repeat;
         background-size: 14px;
-        height: 14px;
-        width: 14px;
+        border: none;
+        box-shadow: none;
+        min-height: 14px;
+        min-width: 14px;
+        outline: none;
+        padding: 0;
       }
 
-      :host #cloudprintPromo .close-button:hover {
-        background-image: -webkit-image-set(
-          url(chrome://theme/IDR_CLOSE_DIALOG_H) 1x,
-          url(chrome://theme/IDR_CLOSE_DIALOG_H@2x) 2x);
-      }
-
-      :host #cloudprintPromo .close-button:active {
+      :host #cloudprintPromo button:active {
         background-image: -webkit-image-set(
           url(chrome://theme/IDR_CLOSE_DIALOG_P) 1x,
           url(chrome://theme/IDR_CLOSE_DIALOG_P@2x) 2x);
       }
 
+      :host #cloudprintPromo button:focus,
+      :host #cloudprintPromo button:hover {
+        background-image: -webkit-image-set(
+          url(chrome://theme/IDR_CLOSE_DIALOG_H) 1x,
+          url(chrome://theme/IDR_CLOSE_DIALOG_H@2x) 2x);
+      }
+
       :host #invitationPromo {
         flex-direction: column;
       }
@@ -176,7 +182,7 @@
             hidden$="[[!showCloudPrintPromo]]">
           <img src="../images/cloud.png" class="icon" alt="">
           <div class="promo-text"></div>
-          <div class="close-button"></div>
+          <button on-click="onCloudPrintPromoDismissed_"></button>
         </div>
         <div class="promo" id="invitationPromo" hidden="[[!invitation_]]">
           <div inner-h-t-m-l="[[getInvitationText_(invitation_)]]"></div>
diff --git a/chrome/browser/resources/print_preview/new/destination_dialog.js b/chrome/browser/resources/print_preview/new/destination_dialog.js
index 3a526c5..4cf257b 100644
--- a/chrome/browser/resources/print_preview/new/destination_dialog.js
+++ b/chrome/browser/resources/print_preview/new/destination_dialog.js
@@ -100,9 +100,6 @@
   attached: function() {
     this.tracker_.add(
         assert(this.$$('.sign-in')), 'click', this.onSignInClick_.bind(this));
-    this.tracker_.add(
-        assert(this.$$('#cloudprintPromo > .close-button')), 'click',
-        this.onCloudPrintPromoDismissed_.bind(this));
   },
 
   /**
diff --git a/chrome/browser/resources/ukm/BUILD.gn b/chrome/browser/resources/ukm/BUILD.gn
deleted file mode 100644
index 4fe10b1..0000000
--- a/chrome/browser/resources/ukm/BUILD.gn
+++ /dev/null
@@ -1,18 +0,0 @@
-# Copyright 2018 The Chromium Authors. All rights reserved.
-# Use of this source code is governed by a BSD-style license that can be
-# found in the LICENSE file.
-
-import("//third_party/closure_compiler/compile_js.gni")
-
-js_type_check("closure_compile") {
-  deps = [
-    ":ukm_internals",
-  ]
-}
-
-js_library("ukm_internals") {
-  deps = [
-    "//ui/webui/resources/js:cr",
-    "//ui/webui/resources/js:util",
-  ]
-}
diff --git a/chrome/browser/resources/ukm/ukm_internals.js b/chrome/browser/resources/ukm/ukm_internals.js
deleted file mode 100644
index 9449c73..0000000
--- a/chrome/browser/resources/ukm/ukm_internals.js
+++ /dev/null
@@ -1,42 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-/**
- * Fetches data from the Ukm service and updates the DOM to display it as a
- * list.
- */
-function updateUkmData() {
-  cr.sendWithPromise('requestUkmData').then((ukmData) => {
-    $('state').innerText = ukmData.state ? 'True' : 'False';
-    $('clientid').innerText = ukmData.client_id;
-    $('sessionid').innerText = ukmData.session_id;
-
-    sourceDiv = $('sources');
-    for (const source of ukmData.sources) {
-      const sourceElement = document.createElement('h3');
-      if (source.url !== undefined)
-        sourceElement.innerText = `Id: ${source.id} Url: ${source.url}`;
-      else
-        sourceElement.innerText = `Id: ${source.id}`;
-      sourceDiv.appendChild(sourceElement);
-
-      for (const entry of source.entries) {
-        const entryElement = document.createElement('h4');
-        entryElement.innerText = `Entry: ${entry.name}`;
-        sourceDiv.appendChild(entryElement);
-
-        if (entry.metrics === undefined)
-          continue;
-        for (const metric of entry.metrics) {
-          const metricElement = document.createElement('h5');
-          metricElement.innerText =
-              `Metric: ${metric.name} Value: ${metric.value}`;
-          sourceDiv.appendChild(metricElement);
-        }
-      }
-    }
-  });
-}
-
-updateUkmData();
diff --git a/chrome/browser/search/ntp_icon_source.cc b/chrome/browser/search/ntp_icon_source.cc
index e013f4f..15f5e59 100644
--- a/chrome/browser/search/ntp_icon_source.cc
+++ b/chrome/browser/search/ntp_icon_source.cc
@@ -14,6 +14,7 @@
 #include "base/strings/string_number_conversions.h"
 #include "cc/paint/skia_paint_canvas.h"
 #include "chrome/browser/favicon/favicon_service_factory.h"
+#include "chrome/browser/history/top_sites_factory.h"
 #include "chrome/browser/profiles/profile.h"
 #include "chrome/browser/search/instant_io_context.h"
 #include "chrome/browser/search/suggestions/image_decoder_impl.h"
@@ -24,6 +25,7 @@
 #include "components/favicon/core/fallback_url_util.h"
 #include "components/favicon/core/favicon_service.h"
 #include "components/favicon_base/favicon_types.h"
+#include "components/history/core/browser/top_sites.h"
 #include "components/image_fetcher/core/image_fetcher_impl.h"
 #include "components/suggestions/proto/suggestions.pb.h"
 #include "components/suggestions/suggestions_service.h"
@@ -34,6 +36,7 @@
 #include "third_party/skia/include/core/SkColor.h"
 #include "third_party/skia/include/core/SkPaint.h"
 #include "ui/base/l10n/l10n_util.h"
+#include "ui/base/resource/resource_bundle.h"
 #include "ui/base/webui/web_ui_util.h"
 #include "ui/gfx/canvas.h"
 #include "ui/gfx/codec/png_codec.h"
@@ -255,18 +258,35 @@
   const ParsedNtpIconPath parsed = ParseNtpIconPath(path);
 
   if (parsed.url.is_valid()) {
+    int icon_size_in_pixels =
+        std::ceil(parsed.size_in_dip * parsed.device_scale_factor);
+    NtpIconRequest request(callback, parsed.url, icon_size_in_pixels,
+                           parsed.device_scale_factor);
+
+    // Check if the requested URL is part of the prepopulated pages (currently,
+    // only the Web Store).
+    scoped_refptr<history::TopSites> top_sites =
+        TopSitesFactory::GetForProfile(profile_);
+    if (top_sites) {
+      for (const auto& prepopulated_page : top_sites->GetPrepopulatedPages()) {
+        if (parsed.url == prepopulated_page.most_visited.url) {
+          gfx::Image& image =
+              ui::ResourceBundle::GetSharedInstance().GetImageNamed(
+                  prepopulated_page.favicon_id);
+          ReturnRenderedIconForRequest(request, image.AsBitmap());
+          return;
+        }
+      }
+    }
+
     // This will query for a local favicon. If not found, will take alternative
     // action in OnLocalFaviconAvailable.
     const bool fallback_to_host = true;
-    int icon_size_in_pixels =
-        std::ceil(parsed.size_in_dip * parsed.device_scale_factor);
     favicon_service->GetRawFaviconForPageURL(
         parsed.url, {favicon_base::IconType::kFavicon}, icon_size_in_pixels,
         fallback_to_host,
         base::Bind(&NtpIconSource::OnLocalFaviconAvailable,
-                   weak_ptr_factory_.GetWeakPtr(),
-                   NtpIconRequest(callback, parsed.url, icon_size_in_pixels,
-                                  parsed.device_scale_factor)),
+                   weak_ptr_factory_.GetWeakPtr(), request),
         &cancelable_task_tracker_);
   } else {
     callback.Run(nullptr);
diff --git a/chrome/browser/sync/chrome_sync_client.cc b/chrome/browser/sync/chrome_sync_client.cc
index 62fa3cb..60b26cfa 100644
--- a/chrome/browser/sync/chrome_sync_client.cc
+++ b/chrome/browser/sync/chrome_sync_client.cc
@@ -48,6 +48,7 @@
 #include "components/autofill/core/browser/webdata/autofill_profile_sync_bridge.h"
 #include "components/autofill/core/browser/webdata/autofill_profile_syncable_service.h"
 #include "components/autofill/core/browser/webdata/autofill_wallet_metadata_syncable_service.h"
+#include "components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h"
 #include "components/autofill/core/browser/webdata/autofill_wallet_syncable_service.h"
 #include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
 #include "components/autofill/core/common/autofill_features.h"
@@ -597,6 +598,17 @@
                  profile_web_data_service_.get())
           ->change_processor()
           ->GetControllerDelegateOnUIThread();
+    case syncer::AUTOFILL_WALLET_DATA: {
+      // TODO(feuunk): This doesn't allow switching which database to use at
+      // runtime. This should be fixed as part of the USS migration for
+      // payments.
+      auto service = account_web_data_service_ ? account_web_data_service_
+                                               : profile_web_data_service_;
+      return autofill::AutofillWalletSyncBridge::FromWebDataService(
+                 service.get())
+          ->change_processor()
+          ->GetControllerDelegateOnUIThread();
+    }
 #if defined(OS_CHROMEOS)
     case syncer::PRINTERS:
       return chromeos::SyncedPrintersManagerFactory::GetForBrowserContext(
diff --git a/chrome/browser/themes/increased_contrast_theme_supplier.cc b/chrome/browser/themes/increased_contrast_theme_supplier.cc
index 700cbe0a..b403cab6 100644
--- a/chrome/browser/themes/increased_contrast_theme_supplier.cc
+++ b/chrome/browser/themes/increased_contrast_theme_supplier.cc
@@ -22,11 +22,10 @@
       *color = SK_ColorWHITE;
       return true;
     case ThemeProperties::COLOR_FRAME_INACTIVE:
+    case ThemeProperties::COLOR_FRAME_INCOGNITO_INACTIVE:
       *color = SK_ColorGRAY;
       return true;
     case ThemeProperties::COLOR_FRAME:
-      *color = SK_ColorDKGRAY;
-      return true;
     case ThemeProperties::COLOR_FRAME_INCOGNITO:
       *color = SK_ColorDKGRAY;
       return true;
diff --git a/chrome/browser/ui/BUILD.gn b/chrome/browser/ui/BUILD.gn
index 2e5deaba4..82df345 100644
--- a/chrome/browser/ui/BUILD.gn
+++ b/chrome/browser/ui/BUILD.gn
@@ -425,14 +425,6 @@
       "cocoa/one_click_signin_dialog_controller.mm",
       "cocoa/one_click_signin_view_controller.h",
       "cocoa/one_click_signin_view_controller.mm",
-      "cocoa/page_info/page_info_bubble_controller.h",
-      "cocoa/page_info/page_info_bubble_controller.mm",
-      "cocoa/page_info/page_info_utils_cocoa.h",
-      "cocoa/page_info/page_info_utils_cocoa.mm",
-      "cocoa/page_info/permission_selector_button.h",
-      "cocoa/page_info/permission_selector_button.mm",
-      "cocoa/page_info/split_block_button.h",
-      "cocoa/page_info/split_block_button.mm",
       "cocoa/password_reuse_warning_dialog_cocoa.h",
       "cocoa/password_reuse_warning_dialog_cocoa.mm",
       "cocoa/password_reuse_warning_view_controller.h",
@@ -551,8 +543,6 @@
       "cocoa/touchbar/web_textfield_touch_bar_controller.mm",
       "cocoa/translate/translate_bubble_bridge_views.h",
       "cocoa/translate/translate_bubble_bridge_views.mm",
-      "cocoa/translate/translate_bubble_controller.h",
-      "cocoa/translate/translate_bubble_controller.mm",
       "cocoa/url_drop_target.h",
       "cocoa/url_drop_target.mm",
       "cocoa/vertical_gradient_view.h",
@@ -2622,8 +2612,6 @@
         "cocoa/importer/import_lock_dialog_cocoa.mm",
         "cocoa/login_handler_cocoa.h",
         "cocoa/login_handler_cocoa.mm",
-        "cocoa/page_info/page_info_bubble_controller.h",
-        "cocoa/page_info/page_info_bubble_controller.mm",
         "cocoa/password_reuse_warning_dialog_cocoa.h",
         "cocoa/password_reuse_warning_dialog_cocoa.mm",
         "cocoa/password_reuse_warning_view_controller.h",
@@ -3046,6 +3034,12 @@
       "views/permission_bubble/chooser_bubble_ui.h",
       "views/permission_bubble/permission_prompt_impl.cc",
       "views/permission_bubble/permission_prompt_impl.h",
+      "views/relaunch_notification/relaunch_notification_controller.cc",
+      "views/relaunch_notification/relaunch_notification_controller.h",
+      "views/relaunch_notification/relaunch_recommended_bubble_view.cc",
+      "views/relaunch_notification/relaunch_recommended_bubble_view.h",
+      "views/relaunch_notification/relaunch_required_dialog_view.cc",
+      "views/relaunch_notification/relaunch_required_dialog_view.h",
       "views/safe_browsing/password_reuse_modal_warning_dialog.cc",
       "views/safe_browsing/password_reuse_modal_warning_dialog.h",
       "views/session_crashed_bubble_view.cc",
@@ -3498,22 +3492,17 @@
 
     if (!is_chromeos) {
       sources += [
-        "views/relaunch_notification/relaunch_notification_controller.cc",
-        "views/relaunch_notification/relaunch_notification_controller.h",
-        "views/relaunch_notification/relaunch_recommended_bubble_view.cc",
-        "views/relaunch_notification/relaunch_recommended_bubble_view.h",
-        "views/relaunch_notification/relaunch_required_dialog_view.cc",
-        "views/relaunch_notification/relaunch_required_dialog_view.h",
         "views/screen_capture_notification_ui_views.cc",
       ]
-      if (is_mac) {
-        sources += [
-          "views/policy/enterprise_startup_dialog_mac_util.h",
-          "views/policy/enterprise_startup_dialog_mac_util.mm",
-          "views/relaunch_notification/get_app_menu_anchor_point.h",
-          "views/relaunch_notification/get_app_menu_anchor_point.mm",
-        ]
-      }
+    }
+
+    if (is_mac) {
+      sources += [
+        "views/policy/enterprise_startup_dialog_mac_util.h",
+        "views/policy/enterprise_startup_dialog_mac_util.mm",
+        "views/relaunch_notification/get_app_menu_anchor_point.h",
+        "views/relaunch_notification/get_app_menu_anchor_point.mm",
+      ]
     }
 
     if (is_chrome_branded) {
diff --git a/chrome/browser/ui/browser_ui_prefs.cc b/chrome/browser/ui/browser_ui_prefs.cc
index a6bf7f2..edb6476 100644
--- a/chrome/browser/ui/browser_ui_prefs.cc
+++ b/chrome/browser/ui/browser_ui_prefs.cc
@@ -39,9 +39,7 @@
   registry->RegisterBooleanPref(prefs::kAllowFileSelectionDialogs, true);
 
 #if !defined(OS_ANDROID)
-#if !defined(OS_CHROMEOS)
   registry->RegisterIntegerPref(prefs::kRelaunchNotification, 0);
-#endif  // !defined(OS_CHROMEOS)
   registry->RegisterIntegerPref(
       prefs::kRelaunchNotificationPeriod,
       base::saturated_cast<int>(
diff --git a/chrome/browser/ui/cocoa/app_menu/app_menu_controller.mm b/chrome/browser/ui/cocoa/app_menu/app_menu_controller.mm
index c50f112..e5ed7c24 100644
--- a/chrome/browser/ui/cocoa/app_menu/app_menu_controller.mm
+++ b/chrome/browser/ui/cocoa/app_menu/app_menu_controller.mm
@@ -367,6 +367,9 @@
     return;
 
   // Remove and re-add menu item so menu gets the correct size.
+  // |removeItemAtIndex:| can trigger an eager dealloc, so retain it
+  // in the meantime.
+  base::scoped_nsobject<NSMenuItem> menuItem([browserActionsMenuItem_ retain]);
   NSInteger index = [[self menu] indexOfItem:browserActionsMenuItem_];
   [[self menu] removeItemAtIndex:index];
   [[self menu] insertItem:browserActionsMenuItem_ atIndex:index];
diff --git a/chrome/browser/ui/cocoa/browser_window_controller.h b/chrome/browser/ui/cocoa/browser_window_controller.h
index 1662cbf..5bf172eb 100644
--- a/chrome/browser/ui/cocoa/browser_window_controller.h
+++ b/chrome/browser/ui/cocoa/browser_window_controller.h
@@ -57,7 +57,6 @@
 @class TabStripControllerCocoa;
 @class TabStripView;
 @class ToolbarController;
-@class TranslateBubbleController;
 
 namespace content {
 class WebContents;
@@ -109,8 +108,6 @@
   BOOL initializing_;  // YES while we are currently in initWithBrowser:
   BOOL ownsBrowser_;  // Only ever NO when testing
 
-  TranslateBubbleController* translateBubbleController_;  // Weak.
-
   // The total amount by which we've grown the window up or down (to display a
   // bookmark bar and/or download shelf), respectively; reset to 0 when moved
   // away from the bottom/top or resized (or zoomed).
diff --git a/chrome/browser/ui/cocoa/browser_window_controller.mm b/chrome/browser/ui/cocoa/browser_window_controller.mm
index 6739d09..534764a 100644
--- a/chrome/browser/ui/cocoa/browser_window_controller.mm
+++ b/chrome/browser/ui/cocoa/browser_window_controller.mm
@@ -79,7 +79,6 @@
 #import "chrome/browser/ui/cocoa/toolbar/toolbar_controller.h"
 #import "chrome/browser/ui/cocoa/touchbar/browser_window_touch_bar_controller.h"
 #include "chrome/browser/ui/cocoa/translate/translate_bubble_bridge_views.h"
-#import "chrome/browser/ui/cocoa/translate/translate_bubble_controller.h"
 #include "chrome/browser/ui/exclusive_access/fullscreen_controller.h"
 #include "chrome/browser/ui/location_bar/location_bar.h"
 #include "chrome/browser/ui/tabs/tab_strip_model.h"
@@ -1599,54 +1598,8 @@
                                      step:(translate::TranslateStep)step
                                 errorType:(translate::TranslateErrors::Type)
                                 errorType {
-  if (chrome::ShowAllDialogsWithViewsToolkit()) {
-    ShowTranslateBubbleViews([self window], [self locationBarBridge], contents,
-                             step, errorType, true);
-    return;
-  }
-  // TODO(hajimehoshi): The similar logic exists at TranslateBubbleView::
-  // ShowBubble. This should be unified.
-  if (translateBubbleController_) {
-    // When the user reads the advanced setting panel, the bubble should not be
-    // changed because they are focusing on the bubble.
-    if (translateBubbleController_.webContents == contents &&
-        translateBubbleController_.model->GetViewState() ==
-        TranslateBubbleModel::VIEW_STATE_ADVANCED) {
-      return;
-    }
-    if (step != translate::TRANSLATE_STEP_TRANSLATE_ERROR) {
-      TranslateBubbleModel::ViewState viewState =
-          TranslateBubbleModelImpl::TranslateStepToViewState(step);
-      [translateBubbleController_ switchView:viewState];
-    } else {
-      [translateBubbleController_ switchToErrorView:errorType];
-    }
-    return;
-  }
-
-  std::string sourceLanguage;
-  std::string targetLanguage;
-  ChromeTranslateClient::GetTranslateLanguages(
-      contents, &sourceLanguage, &targetLanguage);
-
-  std::unique_ptr<translate::TranslateUIDelegate> uiDelegate(
-      new translate::TranslateUIDelegate(
-          ChromeTranslateClient::GetManagerFromWebContents(contents)
-              ->GetWeakPtr(),
-          sourceLanguage, targetLanguage));
-  std::unique_ptr<TranslateBubbleModel> model(
-      new TranslateBubbleModelImpl(step, std::move(uiDelegate)));
-  translateBubbleController_ =
-      [[TranslateBubbleController alloc] initWithParentWindow:self
-                                                        model:std::move(model)
-                                                  webContents:contents];
-  [translateBubbleController_ showWindow:nil];
-
-  NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
-  [center addObserver:self
-             selector:@selector(translateBubbleWindowWillClose:)
-                 name:NSWindowWillCloseNotification
-               object:[translateBubbleController_ window]];
+  ShowTranslateBubbleViews([self window], [self locationBarBridge], contents,
+                           step, errorType, true);
 }
 
 - (void)dismissPermissionBubble {
@@ -1655,17 +1608,6 @@
     delegate->Closing();
 }
 
-// Nil out the weak translate bubble controller reference.
-- (void)translateBubbleWindowWillClose:(NSNotification*)notification {
-  DCHECK_EQ([notification object], [translateBubbleController_ window]);
-
-  NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
-  [center removeObserver:self
-                    name:NSWindowWillCloseNotification
-                  object:[translateBubbleController_ window]];
-  translateBubbleController_ = nil;
-}
-
 // If the browser is in incognito mode or has multi-profiles, install the image
 // view to decorate the window at the upper right. Use the same base y
 // coordinate as the tab strip.
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.h b/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.h
deleted file mode 100644
index 3abc77d..0000000
--- a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.h
+++ /dev/null
@@ -1,160 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_BUBBLE_CONTROLLER_H_
-#define CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_BUBBLE_CONTROLLER_H_
-
-#import <Cocoa/Cocoa.h>
-
-#include <memory>
-
-#include "base/mac/scoped_nsobject.h"
-#include "base/macros.h"
-#import "chrome/browser/ui/cocoa/base_bubble_controller.h"
-#include "chrome/browser/ui/page_info/page_info_ui.h"
-#include "content/public/browser/web_contents_observer.h"
-
-class LocationBarDecoration;
-class PageInfoUIBridge;
-@class InspectLinkView;
-
-namespace content {
-class WebContents;
-}
-
-namespace net {
-class X509Certificate;
-}
-
-// This NSWindowController subclass manages the InfoBubbleWindow and view that
-// are displayed when the user clicks the omnibox security indicator icon.
-@interface PageInfoBubbleController : BaseBubbleController {
- @private
-  content::WebContents* webContents_;
-
-  base::scoped_nsobject<NSView> contentView_;
-
-  // The main content view for the Permissions tab.
-  NSView* securitySectionView_;
-
-  // Displays the short security summary for the page
-  // (private/not private/etc.).
-  NSTextField* securitySummaryField_;
-
-  // Displays a longer explanation of the page's security state, and how the
-  // user should treat it.
-  NSTextField* securityDetailsField_;
-
-  // The link button for opening a Chrome Help Center page explaining connection
-  // security.
-  NSButton* connectionHelpButton_;
-
-  // URL of the page for which the bubble is shown.
-  GURL url_;
-
-  // Displays a paragraph to accompany the reset decisions button, explaining
-  // that the user has made a decision to trust an invalid security certificate
-  // for the current site.
-  // This field only shows when there is an acrive certificate exception.
-  NSTextField* resetDecisionsField_;
-
-  // The link button for revoking certificate decisions.
-  // This link only shows when there is an active certificate exception.
-  NSButton* resetDecisionsButton_;
-
-  // The server certificate from the identity info. This should always be
-  // non-null on a cryptographic connection, and null otherwise.
-  scoped_refptr<net::X509Certificate> certificate_;
-
-  // Separator line.
-  NSView* separatorAfterSecuritySection_;
-
-  // Container for the site settings section.
-  NSView* siteSettingsSectionView_;
-
-  // Container for certificate info in the site settings section.
-  InspectLinkView* certificateView_;
-
-  // Container for cookies info in the site settings section.
-  InspectLinkView* cookiesView_;
-
-  // Container for permission info in the site settings section.
-  NSView* permissionsView_;
-
-  // The link button for showing site settings.
-  NSButton* siteSettingsButton_;
-
-  // The UI translates user actions to specific events and forwards them to the
-  // |presenter_|. The |presenter_| handles these events and updates the UI.
-  std::unique_ptr<PageInfo> presenter_;
-
-  // Bridge which implements the PageInfoUI interface and forwards
-  // methods on to this class.
-  std::unique_ptr<PageInfoUIBridge> bridge_;
-
-  // The omnibox icon the bubble is anchored to. The icon is set as active
-  // when the bubble is opened, and inactive when the bubble is closed.
-  // Usually we would override OmniboxDecorationBubbleController but the page
-  // info icon has a race condition where it might switch between
-  // LocationIconDecoration and SecurityStateBubbleDecoration.
-  LocationBarDecoration* decoration_;  // Weak.
-
-  // The button for changing password decisions.
-  // This button only shows when there is an password reuse event.
-  NSButton* changePasswordButton_;
-
-  // The button for whitelisting password reuse decisions.
-  // This button only shows when there is an password reuse event.
-  NSButton* whitelistPasswordReuseButton_;
-}
-
-// Designated initializer. The controller will release itself when the bubble
-// is closed. |parentWindow| cannot be nil. |webContents| may be nil for
-// testing purposes.
-- (id)initWithParentWindow:(NSWindow*)parentWindow
-          pageInfoUIBridge:(PageInfoUIBridge*)bridge
-               webContents:(content::WebContents*)webContents
-                       url:(const GURL&)url;
-
-// Return the default width of the window. It may be wider to fit the content.
-// This may be overriden by a subclass for testing purposes.
-- (CGFloat)defaultWindowWidth;
-
-@end
-
-// Provides a bridge between the PageInfoUI C++ interface and the Cocoa
-// implementation in PageInfoBubbleController.
-class PageInfoUIBridge : public content::WebContentsObserver,
-                         public PageInfoUI {
- public:
-  explicit PageInfoUIBridge(content::WebContents* web_contents);
-  ~PageInfoUIBridge() override;
-
-  void set_bubble_controller(PageInfoBubbleController* bubble_controller);
-
-  // WebContentsObserver implementation.
-  void RenderFrameDeleted(content::RenderFrameHost* render_frame_host) override;
-
-  // PageInfoUI implementations.
-  void SetCookieInfo(const CookieInfoList& cookie_info_list) override;
-  void SetPermissionInfo(const PermissionInfoList& permission_info_list,
-                         ChosenObjectInfoList chosen_object_info_list) override;
-  void SetIdentityInfo(const IdentityInfo& identity_info) override;
-
- protected:
-  // WebContentsObserver implementation.
-  void DidFinishNavigation(
-      content::NavigationHandle* navigation_handle) override;
-
- private:
-  // The WebContents the bubble UI is attached to.
-  content::WebContents* web_contents_;
-
-  // The Cocoa controller for the bubble UI.
-  PageInfoBubbleController* bubble_controller_;
-
-  DISALLOW_COPY_AND_ASSIGN(PageInfoUIBridge);
-};
-
-#endif  // CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_BUBBLE_CONTROLLER_H_
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.mm b/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.mm
deleted file mode 100644
index 87f5d3f..0000000
--- a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.mm
+++ /dev/null
@@ -1,1540 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.h"
-
-#import <AppKit/AppKit.h>
-
-#include <cmath>
-
-#include "base/bind.h"
-#include "base/i18n/rtl.h"
-#include "base/mac/foundation_util.h"
-#include "base/mac/mac_util.h"
-#include "base/strings/sys_string_conversions.h"
-#import "chrome/browser/certificate_viewer.h"
-#include "chrome/browser/infobars/infobar_service.h"
-#include "chrome/browser/profiles/profile.h"
-#include "chrome/browser/ui/browser_finder.h"
-#include "chrome/browser/ui/browser_window.h"
-#include "chrome/browser/ui/cocoa/browser_dialogs_views_mac.h"
-#import "chrome/browser/ui/cocoa/browser_window_controller.h"
-#include "chrome/browser/ui/cocoa/bubble_anchor_helper.h"
-#import "chrome/browser/ui/cocoa/info_bubble_view.h"
-#import "chrome/browser/ui/cocoa/info_bubble_window.h"
-#include "chrome/browser/ui/cocoa/key_equivalent_constants.h"
-#import "chrome/browser/ui/cocoa/location_bar/location_bar_view_mac.h"
-#import "chrome/browser/ui/cocoa/location_bar/page_info_bubble_decoration.h"
-#import "chrome/browser/ui/cocoa/page_info/permission_selector_button.h"
-#include "chrome/browser/ui/page_info/page_info_dialog.h"
-#include "chrome/browser/ui/page_info/permission_menu_model.h"
-#import "chrome/browser/ui/tab_dialogs.h"
-#include "chrome/common/url_constants.h"
-#include "chrome/grit/theme_resources.h"
-#include "components/content_settings/core/browser/content_settings_registry.h"
-#include "components/strings/grit/components_chromium_strings.h"
-#include "components/strings/grit/components_strings.h"
-#include "content/public/browser/navigation_handle.h"
-#include "content/public/browser/page_navigator.h"
-#include "content/public/browser/ssl_host_state_delegate.h"
-#include "content/public/browser/web_contents.h"
-#include "content/public/common/url_constants.h"
-#include "extensions/common/constants.h"
-#include "skia/ext/skia_utils_mac.h"
-#import "third_party/google_toolbox_for_mac/src/AppKit/GTMUILocalizerAndLayoutTweaker.h"
-#import "ui/base/cocoa/a11y_util.h"
-#include "ui/base/cocoa/cocoa_base_utils.h"
-#import "ui/base/cocoa/controls/button_utils.h"
-#import "ui/base/cocoa/controls/hyperlink_button_cell.h"
-#import "ui/base/cocoa/flipped_view.h"
-#import "ui/base/cocoa/hover_image_button.h"
-#include "ui/base/l10n/l10n_util.h"
-#include "ui/base/resource/resource_bundle.h"
-#include "ui/gfx/color_palette.h"
-#include "ui/gfx/image/image_skia_util_mac.h"
-#import "ui/gfx/mac/coordinate_conversion.h"
-#include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h"
-#include "ui/resources/grit/ui_resources.h"
-
-using ChosenObjectInfoPtr = std::unique_ptr<PageInfoUI::ChosenObjectInfo>;
-using ChosenObjectDeleteCallback =
-    base::Callback<void(const PageInfoUI::ChosenObjectInfo&)>;
-
-namespace {
-
-// General ---------------------------------------------------------------------
-
-// The default width of the window, in view coordinates. It may be larger to
-// fit the content.
-constexpr CGFloat kDefaultWindowWidth = 320;
-
-// Padding around each section
-constexpr CGFloat kSectionVerticalPadding = 20;
-constexpr CGFloat kSectionHorizontalPadding = 16;
-
-// Links are buttons with invisible padding, so we need to move them back to
-// align with other text.
-constexpr CGFloat kLinkButtonXAdjustment = 1;
-
-// Built-in margin for NSButton to take into account.
-constexpr CGFloat kNSButtonBuiltinMargin = 4;
-
-// Security Section ------------------------------------------------------------
-
-// Spacing between security summary, security details, and cert decisions text.
-constexpr CGFloat kSecurityParagraphSpacing = 12;
-
-// Site Settings Section -------------------------------------------------------
-
-// Square size of the permission images.
-constexpr CGFloat kPermissionImageSize = 16;
-
-// Spacing between a permission image and the text.
-constexpr CGFloat kPermissionImageSpacing = 6;
-
-// Minimum distance between the label and its corresponding menu.
-constexpr CGFloat kMinSeparationBetweenLabelAndMenu = 16;
-
-// Square size of the permission delete button image.
-constexpr CGFloat kPermissionDeleteImageSize = 16;
-
-// The spacing between individual permissions.
-constexpr CGFloat kPermissionsVerticalSpacing = 16;
-
-// Spacing to add after a permission label, either directly on top of
-// kPermissionsVerticalSpacing, or before additional text (e.g. "X in use" for
-// cookies).
-constexpr CGFloat kPermissionLabelBottomPadding = 4;
-
-// Amount to lower each permission icon to align the icon baseline with the
-// label text.
-constexpr CGFloat kPermissionIconYAdjustment = 1;
-
-// Amount to lower each permission popup button to make its text align with the
-// permission label.
-constexpr CGFloat kPermissionPopupButtonYAdjustment = 3;
-
-// Internal Page Bubble --------------------------------------------------------
-
-// Padding between the window frame and content for the internal page bubble.
-constexpr CGFloat kInternalPageFramePadding = 10;
-
-// Spacing between the image and text for internal pages.
-constexpr CGFloat kInternalPageImageSpacing = 10;
-
-// -----------------------------------------------------------------------------
-
-// A unique tag given to chosen object views (e.g. to show a site has access to
-// a USB/Bluetooth device) in order to repopulate them on permissions updates.
-// This number must not be the same as any permission in ContentSettingsType.
-constexpr int kChosenObjectTag = CONTENT_SETTINGS_NUM_TYPES;
-
-// NOTE: This assumes that there will never be more than one page info
-// bubble shown, and that the one that is shown is associated with the current
-// window. This matches the behaviour in Views: see PageInfoBubbleView.
-PageInfoBubbleController* g_page_info_bubble = nullptr;
-
-// Takes in the parent window, which should be a BrowserWindow, and gets the
-// proper anchor point for the bubble. The returned point is in screen
-// coordinates.
-NSPoint AnchorPointForWindow(NSWindow* parent) {
-  Browser* browser = chrome::FindBrowserWithWindow(parent);
-  DCHECK(browser);
-  return GetPageInfoAnchorPointForBrowser(browser);
-}
-
-NSImage* GetNSImageFromImageSkia(const gfx::ImageSkia& image) {
-  return NSImageFromImageSkiaWithColorSpace(image,
-                                            base::mac::GetSRGBColorSpace());
-}
-
-SkColor GetRelatedTextColor() {
-  return skia::NSDeviceColorToSkColor(
-      [[NSColor textColor] colorUsingColorSpaceName:NSDeviceRGBColorSpace]);
-}
-
-}  // namespace
-
-// The |InspectLinkView| objects are used to show the Cookie and Certificate
-// status and a link to inspect the underlying data.
-@interface InspectLinkView : FlippedView
-@end
-
-@implementation InspectLinkView {
-  NSButton* actionLink_;
-}
-
-- (id)initWithFrame:(NSRect)frame {
-  if (self = [super initWithFrame:frame]) {
-    [self setAutoresizingMask:NSViewWidthSizable];
-  }
-  return self;
-}
-
-- (void)setActionLink:(NSButton*)actionLink {
-  actionLink_ = actionLink;
-}
-
-- (void)setLinkText:(NSString*)linkText {
-  [actionLink_ setTitle:linkText];
-  [GTMUILocalizerAndLayoutTweaker sizeToFitView:actionLink_];
-}
-
-- (void)setLinkToolTip:(NSString*)linkToolTip {
-  [actionLink_ setToolTip:linkToolTip];
-}
-
-- (void)setLinkTarget:(NSObject*)target withAction:(SEL)action {
-  [actionLink_ setTarget:target];
-  [actionLink_ setAction:action];
-}
-@end
-
-@interface ChosenObjectDeleteButton : HoverImageButton {
- @private
-  ChosenObjectInfoPtr objectInfo_;
-  ChosenObjectDeleteCallback callback_;
-}
-
-// Designated initializer. Takes ownership of |objectInfo|.
-- (instancetype)initWithChosenObject:(ChosenObjectInfoPtr)objectInfo
-                             atPoint:(NSPoint)point
-                        withCallback:(ChosenObjectDeleteCallback)callback;
-
-// Action when the button is clicked.
-- (void)deleteClicked:(id)sender;
-
-@end
-
-@implementation ChosenObjectDeleteButton
-
-- (instancetype)initWithChosenObject:(ChosenObjectInfoPtr)objectInfo
-                             atPoint:(NSPoint)point
-                        withCallback:(ChosenObjectDeleteCallback)callback {
-  NSRect frame = NSMakeRect(point.x, point.y, kPermissionDeleteImageSize,
-                            kPermissionDeleteImageSize);
-  if (self = [super initWithFrame:frame]) {
-    ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
-    [self setDefaultImage:rb.GetNativeImageNamed(IDR_CLOSE_2).ToNSImage()];
-    [self setHoverImage:rb.GetNativeImageNamed(IDR_CLOSE_2_H).ToNSImage()];
-    [self setPressedImage:rb.GetNativeImageNamed(IDR_CLOSE_2_P).ToNSImage()];
-    [self setBordered:NO];
-    [self setToolTip:l10n_util::GetNSString(
-                         objectInfo->ui_info.delete_tooltip_string_id)];
-    [self setTarget:self];
-    [self setAction:@selector(deleteClicked:)];
-    objectInfo_ = std::move(objectInfo);
-    callback_ = callback;
-  }
-  return self;
-}
-
-- (void)deleteClicked:(id)sender {
-  callback_.Run(*objectInfo_);
-}
-
-@end
-
-@implementation PageInfoBubbleController
-
-+ (PageInfoBubbleController*)getPageInfoBubbleForTest {
-  return g_page_info_bubble;
-}
-
-- (CGFloat)defaultWindowWidth {
-  return kDefaultWindowWidth;
-}
-
-bool IsInternalURL(const GURL& url) {
-  return url.SchemeIs(content::kChromeUIScheme) ||
-         url.SchemeIs(content::kChromeDevToolsScheme) ||
-         url.SchemeIs(extensions::kExtensionScheme) ||
-         url.SchemeIs(content::kViewSourceScheme);
-}
-
-- (id)initWithParentWindow:(NSWindow*)parentWindow
-          pageInfoUIBridge:(PageInfoUIBridge*)bridge
-               webContents:(content::WebContents*)webContents
-                       url:(const GURL&)url {
-  DCHECK(parentWindow);
-
-  webContents_ = webContents;
-  url_ = url;
-
-  // Use an arbitrary height; it will be changed in performLayout.
-  NSRect contentRect = NSMakeRect(0, 0, [self defaultWindowWidth], 1);
-  // Create an empty window into which content is placed.
-  base::scoped_nsobject<InfoBubbleWindow> window([[InfoBubbleWindow alloc]
-      initWithContentRect:contentRect
-                styleMask:NSBorderlessWindowMask
-                  backing:NSBackingStoreBuffered
-                    defer:NO]);
-
-  if ((self = [super initWithWindow:window.get()
-                       parentWindow:parentWindow
-                         anchoredAt:NSZeroPoint])) {
-    [[self bubble] setArrowLocation:info_bubble::kTopLeading];
-
-    // Create the container view that uses flipped coordinates.
-    NSRect contentFrame = NSMakeRect(0, 0, [self defaultWindowWidth], 300);
-    contentView_.reset([[FlippedView alloc] initWithFrame:contentFrame]);
-
-    // Replace the window's content.
-    [[[self window] contentView]
-        setSubviews:[NSArray arrayWithObject:contentView_.get()]];
-
-    if (IsInternalURL(url_)) {
-      [self initializeContentsForInternalPage:url_];
-    } else {
-      [self initializeContents];
-    }
-
-    bridge_.reset(bridge);
-    bridge_->set_bubble_controller(self);
-  }
-  return self;
-}
-
-- (void)showWindow:(id)sender {
-  BrowserWindowController* controller = [BrowserWindowController
-      browserWindowControllerForWindow:[self parentWindow]];
-  LocationBarViewMac* locationBar = [controller locationBarBridge];
-  if (locationBar) {
-    decoration_ = locationBar->page_info_decoration();
-    decoration_->SetActive(true);
-  }
-
-  [super showWindow:sender];
-}
-
-- (void)close {
-  if (decoration_) {
-    decoration_->SetActive(false);
-    decoration_ = nullptr;
-  }
-
-  [super close];
-}
-
-- (Profile*)profile {
-  return Profile::FromBrowserContext(webContents_->GetBrowserContext());
-}
-
-- (void)windowWillClose:(NSNotification*)notification {
-  if (presenter_.get())
-    presenter_->OnUIClosing();
-  presenter_.reset();
-  [super windowWillClose:notification];
-}
-
-- (void)setPresenter:(PageInfo*)presenter {
-  presenter_.reset(presenter);
-}
-
-// Create the subviews for the bubble for internal Chrome pages.
-- (void)initializeContentsForInternalPage:(const GURL&)url {
-  ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
-
-  int text = IDS_PAGE_INFO_INTERNAL_PAGE;
-  int icon = IDR_PRODUCT_LOGO_16;
-  if (url.SchemeIs(extensions::kExtensionScheme)) {
-    text = IDS_PAGE_INFO_EXTENSION_PAGE;
-    icon = IDR_PLUGINS_FAVICON;
-  } else if (url.SchemeIs(content::kViewSourceScheme)) {
-    text = IDS_PAGE_INFO_VIEW_SOURCE_PAGE;
-    // view-source scheme uses the same icon as chrome:// pages.
-    icon = IDR_PRODUCT_LOGO_16;
-  } else if (!url.SchemeIs(content::kChromeUIScheme) &&
-             !url.SchemeIs(content::kChromeDevToolsScheme)) {
-    NOTREACHED();
-  }
-
-  NSPoint controlOrigin =
-      NSMakePoint(kInternalPageFramePadding,
-                  kInternalPageFramePadding + info_bubble::kBubbleArrowHeight);
-  NSImage* productLogoImage = rb.GetNativeImageNamed(icon).ToNSImage();
-  NSImageView* imageView = [self addImageWithSize:[productLogoImage size]
-                                           toView:contentView_
-                                          atPoint:controlOrigin];
-  [imageView setImage:productLogoImage];
-
-  NSRect imageFrame = [imageView frame];
-  controlOrigin.x += NSWidth(imageFrame) + kInternalPageImageSpacing;
-  NSTextField* textField = [self addText:l10n_util::GetStringUTF16(text)
-                                withSize:[NSFont smallSystemFontSize]
-                                    bold:NO
-                                  toView:contentView_
-                                 atPoint:controlOrigin];
-  // Center the image vertically with the text. Previously this code centered
-  // the text vertically while holding the image in place. That produced correct
-  // results when the image, at 26x26, was taller than (or just slightly
-  // shorter) than the text, but produced incorrect results once the icon
-  // shrank to 16x16. The icon should now always be shorter than the text.
-  // See crbug.com/572044 .
-  NSRect textFrame = [textField frame];
-  imageFrame.origin.y += (NSHeight(textFrame) - NSHeight(imageFrame)) / 2;
-  [imageView setFrame:imageFrame];
-
-  // Adjust the contentView to fit everything.
-  CGFloat maxY = std::max(NSMaxY(imageFrame), NSMaxY(textFrame));
-  [contentView_ setFrame:NSMakeRect(0, 0, [self defaultWindowWidth],
-                                    maxY + kInternalPageFramePadding)];
-
-  [self sizeAndPositionWindow];
-}
-
-// Create the subviews for the page info bubble.
-- (void)initializeContents {
-  securitySectionView_ = [self addSecuritySectionToView:contentView_];
-  separatorAfterSecuritySection_ = [self addSeparatorToView:contentView_];
-  siteSettingsSectionView_ = [self addSiteSettingsSectionToView:contentView_];
-
-  [self performLayout];
-}
-
-// Create and return a subview for the security section and add it to the given
-// |superview|. |superview| retains the new view.
-- (NSView*)addSecuritySectionToView:(NSView*)superview {
-  base::scoped_nsobject<NSView> securitySectionView(
-      [[FlippedView alloc] initWithFrame:[superview frame]]);
-  [superview addSubview:securitySectionView];
-
-  // Create a controlOrigin to place the text fields. The y value doesn't
-  // matter, because the correct value is calculated in -performLayout.
-  NSPoint controlOrigin = NSMakePoint(kSectionHorizontalPadding, 0);
-
-  // Create a text field for the security summary (private/not private/etc.).
-  securitySummaryField_ = [self addText:base::string16()
-                               withSize:[NSFont systemFontSize]
-                                   bold:NO
-                                 toView:securitySectionView
-                                atPoint:controlOrigin];
-
-  securityDetailsField_ = [self addText:base::string16()
-                               withSize:[NSFont smallSystemFontSize]
-                                   bold:NO
-                                 toView:securitySectionView
-                                atPoint:controlOrigin];
-
-  // These will be created only if necessary.
-  resetDecisionsField_ = nil;
-  resetDecisionsButton_ = nil;
-  changePasswordButton_ = nil;
-  whitelistPasswordReuseButton_ = nil;
-
-  NSString* connectionHelpButtonText = l10n_util::GetNSString(IDS_LEARN_MORE);
-  connectionHelpButton_ = [self addLinkButtonWithText:connectionHelpButtonText
-                                               toView:securitySectionView];
-  [connectionHelpButton_ setTarget:self];
-  [connectionHelpButton_ setAction:@selector(openConnectionHelp:)];
-
-  if (base::i18n::IsRTL()) {
-    securitySummaryField_.alignment = NSRightTextAlignment;
-    securityDetailsField_.alignment = NSRightTextAlignment;
-  }
-
-  return securitySectionView.get();
-}
-
-// Create and return a subview for the site settings and add it to the given
-// |superview|. |superview| retains the new view.
-- (NSView*)addSiteSettingsSectionToView:(NSView*)superview {
-  base::scoped_nsobject<NSView> siteSettingsSectionView(
-      [[FlippedView alloc] initWithFrame:[superview frame]]);
-  [superview addSubview:siteSettingsSectionView];
-
-  permissionsView_ =
-      [[[FlippedView alloc] initWithFrame:[superview frame]] autorelease];
-  [siteSettingsSectionView addSubview:permissionsView_];
-
-  // The certificate section is created on demand.
-  certificateView_ = nil;
-
-  // Initialize the two containers that hold the controls. The initial frames
-  // are arbitrary, and will be adjusted after the controls are laid out.
-  PageInfoUI::PermissionInfo info;
-  info.type = CONTENT_SETTINGS_TYPE_COOKIES;
-  info.setting = CONTENT_SETTING_ALLOW;
-  cookiesView_ = [self
-      addInspectLinkToView:siteSettingsSectionView
-               sectionIcon:GetNSImageFromImageSkia(
-                               PageInfoUI::GetPermissionIcon(
-                                   info, GetRelatedTextColor()))
-              sectionTitle:l10n_util::GetStringUTF16(IDS_PAGE_INFO_COOKIES)
-                  linkText:l10n_util::GetPluralNSStringF(
-                               IDS_PAGE_INFO_NUM_COOKIES, 0)];
-  [cookiesView_ setLinkTarget:self
-                   withAction:@selector(showCookiesAndSiteData:)];
-
-  // Create the link button to view site settings. Its position will be set in
-  // performLayout.
-  NSString* siteSettingsButtonText =
-      l10n_util::GetNSString(IDS_PAGE_INFO_SITE_SETTINGS_LINK);
-  siteSettingsButton_ = [self addButtonWithText:siteSettingsButtonText
-                                         toView:siteSettingsSectionView];
-  [GTMUILocalizerAndLayoutTweaker sizeToFitView:siteSettingsButton_];
-
-  [siteSettingsButton_ setTarget:self];
-  [siteSettingsButton_ setAction:@selector(showSiteSettingsData:)];
-
-  return siteSettingsSectionView.get();
-}
-
-- (InspectLinkView*)addInspectLinkToView:(NSView*)superview
-                             sectionIcon:(NSImage*)imageIcon
-                            sectionTitle:(const base::string16&)titleText
-                                linkText:(NSString*)linkText {
-  // Create the subview.
-  base::scoped_nsobject<InspectLinkView> newView(
-      [[InspectLinkView alloc] initWithFrame:[superview frame]]);
-  [superview addSubview:newView];
-
-  bool isRTL = base::i18n::IsRTL();
-  NSPoint controlOrigin = NSMakePoint(kSectionHorizontalPadding, 0);
-
-  CGFloat viewWidth = NSWidth([newView frame]);
-
-  // Reset X for the icon.
-  if (isRTL) {
-    controlOrigin.x =
-        viewWidth - kPermissionImageSize - kSectionHorizontalPadding;
-  }
-
-  NSImageView* imageView = [self addImageWithSize:[imageIcon size]
-                                           toView:newView
-                                          atPoint:controlOrigin];
-  [imageView setImage:imageIcon];
-
-  NSButton* actionLink = [self addLinkButtonWithText:linkText toView:newView];
-  [newView setActionLink:actionLink];
-
-  if (isRTL) {
-    controlOrigin.x -= kPermissionImageSpacing;
-    NSTextField* sectionTitle = [self addText:titleText
-                                     withSize:[NSFont systemFontSize]
-                                         bold:NO
-                                       toView:newView
-                                      atPoint:controlOrigin];
-    [sectionTitle sizeToFit];
-
-    NSPoint sectionTitleOrigin = [sectionTitle frame].origin;
-    sectionTitleOrigin.x -= NSWidth([sectionTitle frame]);
-    [sectionTitle setFrameOrigin:sectionTitleOrigin];
-
-    // Align the icon with the text.
-    [self alignPermissionIcon:imageView withTextField:sectionTitle];
-
-    controlOrigin.y +=
-        NSHeight([sectionTitle frame]) + kPermissionLabelBottomPadding;
-    controlOrigin.x -= NSWidth([actionLink frame]) - kLinkButtonXAdjustment;
-    [actionLink setFrameOrigin:controlOrigin];
-  } else {
-    controlOrigin.x += kPermissionImageSize + kPermissionImageSpacing;
-    NSTextField* sectionTitle = [self addText:titleText
-                                     withSize:[NSFont systemFontSize]
-                                         bold:NO
-                                       toView:newView
-                                      atPoint:controlOrigin];
-    [sectionTitle sizeToFit];
-
-    // Align the icon with the text.
-    [self alignPermissionIcon:imageView withTextField:sectionTitle];
-
-    controlOrigin.y +=
-        NSHeight([sectionTitle frame]) + kPermissionLabelBottomPadding;
-    controlOrigin.x -= kLinkButtonXAdjustment;
-    [actionLink setFrameOrigin:controlOrigin];
-  }
-
-  controlOrigin.y += NSHeight([actionLink frame]);
-  [newView setFrameSize:NSMakeSize(NSWidth([newView frame]), controlOrigin.y)];
-
-  return newView.get();
-}
-
-// Handler for the link button below the list of cookies.
-- (void)showCookiesAndSiteData:(id)sender {
-  DCHECK(webContents_);
-  DCHECK(presenter_);
-  presenter_->RecordPageInfoAction(PageInfo::PAGE_INFO_COOKIES_DIALOG_OPENED);
-  TabDialogs::FromWebContents(webContents_)->ShowCollectedCookies();
-}
-
-// Handler for the site settings button below the list of permissions.
-- (void)showSiteSettingsData:(id)sender {
-  DCHECK(webContents_);
-  DCHECK(presenter_);
-  presenter_->OpenSiteSettingsView();
-}
-
-- (void)openConnectionHelp:(id)sender {
-  DCHECK(webContents_);
-  DCHECK(presenter_);
-  presenter_->RecordPageInfoAction(PageInfo::PAGE_INFO_CONNECTION_HELP_OPENED);
-  webContents_->OpenURL(content::OpenURLParams(
-      GURL(chrome::kPageInfoHelpCenterURL), content::Referrer(),
-      WindowOpenDisposition::NEW_FOREGROUND_TAB, ui::PAGE_TRANSITION_LINK,
-      false));
-}
-
-// Handler for the link button to show certificate information.
-- (void)showCertificateInfo:(id)sender {
-  DCHECK(certificate_.get());
-  DCHECK(presenter_);
-  presenter_->RecordPageInfoAction(
-      PageInfo::PAGE_INFO_CERTIFICATE_DIALOG_OPENED);
-  ShowCertificateViewer(webContents_, [self parentWindow], certificate_.get());
-}
-
-// Handler for the link button to revoke user certificate decisions.
-- (void)resetCertificateDecisions:(id)sender {
-  DCHECK(resetDecisionsButton_);
-  presenter_->OnRevokeSSLErrorBypassButtonPressed();
-  [self close];
-}
-
-// Handler for the button to change password decisions.
-- (void)changePasswordDecisions:(id)sender {
-  DCHECK(changePasswordButton_);
-  presenter_->OnChangePasswordButtonPressed(webContents_);
-  [self close];
-}
-
-// Handler for the button to whitelist password reuse decisions.
-- (void)whitelistPasswordReuseDecisions:(id)sender {
-  DCHECK(whitelistPasswordReuseButton_);
-  presenter_->OnWhitelistPasswordReuseButtonPressed(webContents_);
-  [self close];
-}
-
-- (CGFloat)layoutViewAtRTLStart:(NSView*)view withYPosition:(CGFloat)yPos {
-  CGFloat xPos;
-  if (base::i18n::IsRTL()) {
-    xPos = kDefaultWindowWidth - kSectionHorizontalPadding -
-           NSWidth([view frame]) + kNSButtonBuiltinMargin;
-  } else {
-    xPos = kSectionHorizontalPadding - kNSButtonBuiltinMargin;
-  }
-  [view setFrameOrigin:NSMakePoint(xPos, yPos - kNSButtonBuiltinMargin)];
-  return yPos + NSHeight([view frame]) - kNSButtonBuiltinMargin;
-}
-
-// Set the Y position of |view| to the given position, and return the position
-// of its bottom edge.
-- (CGFloat)setYPositionOfView:(NSView*)view to:(CGFloat)position {
-  NSRect frame = [view frame];
-  frame.origin.y = position;
-  [view setFrame:frame];
-  return position + NSHeight(frame);
-}
-
-- (void)setWidthOfView:(NSView*)view to:(CGFloat)width {
-  [view setFrameSize:NSMakeSize(width, NSHeight([view frame]))];
-}
-
-- (void)setHeightOfView:(NSView*)view to:(CGFloat)height {
-  [view setFrameSize:NSMakeSize(NSWidth([view frame]), height)];
-}
-
-// Layout all of the controls in the window. This should be called whenever
-// the content has changed.
-- (void)performLayout {
-  // Skip layout if the bubble is closing.
-  InfoBubbleWindow* bubbleWindow =
-      base::mac::ObjCCastStrict<InfoBubbleWindow>([self window]);
-  if ([bubbleWindow isClosing])
-    return;
-
-  // Make the content at least as wide as the permissions view.
-  CGFloat contentWidth =
-      std::max([self defaultWindowWidth], NSWidth([permissionsView_ frame]));
-
-  // Set the width of the content view now, so that all the text fields will
-  // be sized to fit before their heights and vertical positions are adjusted.
-  [self setWidthOfView:contentView_ to:contentWidth];
-  [self setWidthOfView:securitySectionView_ to:contentWidth];
-  [self setWidthOfView:siteSettingsSectionView_ to:contentWidth];
-
-  CGFloat yPos = 0;
-
-  [self layoutSecuritySection];
-  yPos = [self setYPositionOfView:securitySectionView_ to:yPos];
-
-  yPos = [self setYPositionOfView:separatorAfterSecuritySection_ to:yPos];
-
-  [self layoutSiteSettingsSection];
-  yPos = [self setYPositionOfView:siteSettingsSectionView_ to:yPos];
-
-  [contentView_ setFrame:NSMakeRect(0, 0, NSWidth([contentView_ frame]), yPos)];
-
-  [self sizeAndPositionWindow];
-}
-
-- (void)layoutSecuritySection {
-  // Margins are handled by the caller.
-  CGFloat yPos = 0;
-
-  [self sizeTextFieldHeightToFit:securitySummaryField_];
-  yPos = [self setYPositionOfView:securitySummaryField_
-                               to:yPos + kSectionVerticalPadding];
-
-  [self sizeTextFieldHeightToFit:securityDetailsField_];
-  yPos = [self setYPositionOfView:securityDetailsField_
-                               to:yPos + kSecurityParagraphSpacing];
-
-  // A common anchor point for link elements
-  CGFloat linkY = kSectionHorizontalPadding - kLinkButtonXAdjustment;
-
-  NSPoint helpOrigin = NSMakePoint(linkY, yPos);
-  if (base::i18n::IsRTL()) {
-    helpOrigin.x = NSWidth([contentView_ frame]) - helpOrigin.x -
-                   NSWidth(connectionHelpButton_.frame);
-  }
-  [connectionHelpButton_ setFrameOrigin:helpOrigin];
-  yPos = NSMaxY([connectionHelpButton_ frame]);
-
-  if (resetDecisionsButton_) {
-    DCHECK(resetDecisionsField_);
-    yPos = [self setYPositionOfView:resetDecisionsField_
-                                 to:yPos + kSecurityParagraphSpacing];
-
-    NSPoint resetOrigin = NSMakePoint(linkY, yPos);
-    if (base::i18n::IsRTL()) {
-      resetOrigin.x = NSWidth([contentView_ frame]) - resetOrigin.x -
-                      NSWidth(resetDecisionsButton_.frame);
-    }
-    [resetDecisionsButton_ setFrameOrigin:resetOrigin];
-    yPos = NSMaxY([resetDecisionsButton_ frame]);
-  }
-
-  if (changePasswordButton_) {
-    NSPoint changePasswordButtonOrigin;
-    NSPoint whitelistReuseButtonOrigin;
-    CGFloat viewWidth = NSWidth([contentView_ frame]);
-    CGFloat changePasswordButtonWidth = NSWidth([changePasswordButton_ frame]);
-    CGFloat whitelistReuseButtonWidth =
-        NSWidth([whitelistPasswordReuseButton_ frame]);
-    CGFloat horizontalPadding =
-        kSectionHorizontalPadding - kNSButtonBuiltinMargin;
-    bool canFitInOneLine = changePasswordButtonWidth +
-                               whitelistReuseButtonWidth +
-                               2 * horizontalPadding <=
-                           viewWidth;
-    bool isRTL = base::i18n::IsRTL();
-    // Buttons are left-aligned for LTR languages, and are right aligned for
-    // RTL languages. Button order follows OSX convention.
-    if (canFitInOneLine) {
-      whitelistReuseButtonOrigin.y = changePasswordButtonOrigin.y =
-          yPos + kSecurityParagraphSpacing;
-      if (isRTL) {
-        whitelistReuseButtonOrigin.x =
-            viewWidth - whitelistReuseButtonWidth - horizontalPadding;
-        changePasswordButtonOrigin.x =
-            whitelistReuseButtonOrigin.x - changePasswordButtonWidth;
-      } else {
-        whitelistReuseButtonOrigin.x = horizontalPadding;
-        changePasswordButtonOrigin.x =
-            whitelistReuseButtonOrigin.x + whitelistReuseButtonWidth;
-      }
-    } else {
-      // If these buttons cannot fit in one line, stack them vertically.
-      CGFloat buttonWidth = viewWidth - 2 * horizontalPadding;
-      whitelistReuseButtonOrigin.x = horizontalPadding;
-      whitelistReuseButtonOrigin.y = yPos + kSecurityParagraphSpacing;
-      [whitelistPasswordReuseButton_
-          setFrameSize:NSMakeSize(
-                           buttonWidth,
-                           NSHeight([whitelistPasswordReuseButton_ frame]))];
-      changePasswordButtonOrigin.x = horizontalPadding;
-      changePasswordButtonOrigin.y =
-          yPos + kSecurityParagraphSpacing +
-          NSHeight([whitelistPasswordReuseButton_ frame]);
-      [changePasswordButton_
-          setFrameSize:NSMakeSize(buttonWidth,
-                                  NSHeight([changePasswordButton_ frame]))];
-    }
-    [changePasswordButton_ setFrameOrigin:changePasswordButtonOrigin];
-    [whitelistPasswordReuseButton_ setFrameOrigin:whitelistReuseButtonOrigin];
-    yPos = NSMaxY([changePasswordButton_ frame]) - kNSButtonBuiltinMargin;
-  }
-
-  // Resize the height based on contents.
-  [self setHeightOfView:securitySectionView_ to:yPos + kSectionVerticalPadding];
-}
-
-- (void)layoutSiteSettingsSection {
-  // Margins are handled by the caller.
-  CGFloat yPos = 0;
-
-  yPos = [self setYPositionOfView:permissionsView_ to:yPos] +
-         kPermissionsVerticalSpacing;
-
-  if (certificateView_) {
-    yPos = [self setYPositionOfView:certificateView_ to:yPos] +
-           kPermissionsVerticalSpacing;
-  }
-
-  yPos =
-      [self setYPositionOfView:cookiesView_ to:yPos] + kSectionVerticalPadding;
-
-  yPos = [self layoutViewAtRTLStart:siteSettingsButton_ withYPosition:yPos] +
-         kSectionVerticalPadding;
-
-  // Resize the height based on contents.
-  [self setHeightOfView:siteSettingsSectionView_ to:yPos];
-}
-
-// Adjust the size of the window to match the size of the content, and position
-// the bubble anchor appropriately.
-- (void)sizeAndPositionWindow {
-  NSRect windowFrame = [contentView_ frame];
-  windowFrame.size =
-      [[[self window] contentView] convertSize:windowFrame.size toView:nil];
-  // Adjust the origin by the difference in height.
-  windowFrame.origin = [[self window] frame].origin;
-  windowFrame.origin.y -=
-      NSHeight(windowFrame) - NSHeight([[self window] frame]);
-
-  // Resize the window. Only animate if the window is visible, otherwise it
-  // could be "growing" while it's opening, looking awkward.
-  [[self window] setFrame:windowFrame
-                  display:YES
-                  animate:[[self window] isVisible]];
-
-  // Adjust the anchor for the bubble.
-  [self setAnchorPoint:AnchorPointForWindow([self parentWindow])];
-}
-
-// Sets properties on the given |field| to act as the title or description
-// labels in the bubble.
-- (void)configureTextFieldAsLabel:(NSTextField*)textField {
-  [textField setEditable:NO];
-  [textField setSelectable:YES];
-  [textField setDrawsBackground:NO];
-  [textField setBezeled:NO];
-}
-
-// Adjust the height of the given text field to match its text.
-- (void)sizeTextFieldHeightToFit:(NSTextField*)textField {
-  NSRect frame = [textField frame];
-  frame.size.height +=
-      [GTMUILocalizerAndLayoutTweaker sizeToFitFixedWidthTextField:textField];
-  [textField setFrame:frame];
-}
-
-// Create a new text field and add it to the given array of subviews.
-// The array will retain a reference to the object.
-- (NSTextField*)addText:(const base::string16&)text
-               withSize:(CGFloat)fontSize
-                   bold:(BOOL)bold
-                 toView:(NSView*)view
-                atPoint:(NSPoint)point {
-  // Size the text to take up the full available width, with some padding.
-  // The height is arbitrary as it will be adjusted later.
-  CGFloat width = NSWidth([view frame]) - point.x - kSectionHorizontalPadding;
-  NSRect frame = NSMakeRect(point.x, point.y, width, 100);
-  base::scoped_nsobject<NSTextField> textField(
-      [[NSTextField alloc] initWithFrame:frame]);
-  [self configureTextFieldAsLabel:textField.get()];
-  [textField setStringValue:base::SysUTF16ToNSString(text)];
-  NSFont* font = bold ? [NSFont boldSystemFontOfSize:fontSize]
-                      : [NSFont systemFontOfSize:fontSize];
-  [textField setFont:font];
-  [self sizeTextFieldHeightToFit:textField];
-  [textField setAutoresizingMask:NSViewWidthSizable];
-  [view addSubview:textField.get()];
-  return textField.get();
-}
-
-// Add an image as a subview of the given view, placed at a pre-determined x
-// position and the given y position. The image is not in the accessibility
-// order, since the image is always accompanied by text in this bubble. Return
-// the new NSImageView.
-- (NSImageView*)addImageWithSize:(NSSize)size
-                          toView:(NSView*)view
-                         atPoint:(NSPoint)point {
-  NSRect frame = NSMakeRect(point.x, point.y, size.width, size.height);
-  base::scoped_nsobject<NSImageView> imageView(
-      [[NSImageView alloc] initWithFrame:frame]);
-  ui::a11y_util::HideImageFromAccessibilityOrder(imageView);
-  [imageView setImageFrameStyle:NSImageFrameNone];
-  [view addSubview:imageView.get()];
-  return imageView.get();
-}
-
-// Add a separator as a subview of the given view. Return the new view.
-- (NSView*)addSeparatorToView:(NSView*)view {
-  // Use an arbitrary position; it will be adjusted in performLayout.
-  NSBox* spacer = [self
-      horizontalSeparatorWithFrame:NSMakeRect(0, 0, NSWidth([view frame]), 0)];
-  [view addSubview:spacer];
-  return spacer;
-}
-
-// Add a link button with the given text to |view|.
-- (NSButton*)addLinkButtonWithText:(NSString*)text toView:(NSView*)view {
-  // Frame size is arbitrary; it will be adjusted by the layout tweaker.
-  NSRect frame = NSMakeRect(kSectionHorizontalPadding, 0, 100, 10);
-  base::scoped_nsobject<NSButton> button(
-      [[NSButton alloc] initWithFrame:frame]);
-  base::scoped_nsobject<HyperlinkButtonCell> cell(
-      [[HyperlinkButtonCell alloc] initTextCell:text]);
-  [cell setControlSize:NSSmallControlSize];
-  [button setCell:cell.get()];
-  [button setButtonType:NSMomentaryPushInButton];
-  [button setBezelStyle:NSRegularSquareBezelStyle];
-  [view addSubview:button.get()];
-
-  [GTMUILocalizerAndLayoutTweaker sizeToFitView:button.get()];
-  return button.get();
-}
-
-// Create and return a button with the specified text and add it to the given
-// |view|. |view| retains the new button.
-- (NSButton*)addButtonWithText:(NSString*)text toView:(NSView*)view {
-  NSRect containerFrame = [view frame];
-  // Frame size is arbitrary; it will be adjusted by the layout tweaker.
-  NSRect frame = NSMakeRect(kSectionHorizontalPadding, 0, 100, 10);
-  base::scoped_nsobject<NSButton> button(
-      [[NSButton alloc] initWithFrame:frame]);
-
-  // Determine the largest possible size for this button. The size is the width
-  // of the connection section minus the padding on both sides minus the
-  // connection image size and spacing.
-  CGFloat maxTitleWidth =
-      containerFrame.size.width - kSectionHorizontalPadding * 2;
-
-  base::scoped_nsobject<NSButtonCell> cell(
-      [[NSButtonCell alloc] initTextCell:text]);
-  [button setCell:cell.get()];
-  [GTMUILocalizerAndLayoutTweaker sizeToFitView:button.get()];
-
-  // Ensure the containing view is large enough to contain the button with its
-  // widest possible title.
-  NSRect buttonFrame = [button frame];
-  buttonFrame.size.width = maxTitleWidth;
-
-  [button setFrame:buttonFrame];
-  [button setButtonType:NSMomentaryPushInButton];
-  [button setBezelStyle:NSRegularSquareBezelStyle];
-  [view addSubview:button.get()];
-
-  return button.get();
-}
-
-// Set the content of the identity and identity status fields, and add the
-// Certificate view or password reuse buttons if applicable.
-- (void)setIdentityInfo:(const PageInfoUI::IdentityInfo&)identityInfo {
-  std::unique_ptr<PageInfoUI::SecurityDescription> security_description =
-      identityInfo.GetSecurityDescription();
-  [securitySummaryField_
-      setStringValue:base::SysUTF16ToNSString(security_description->summary)];
-
-  [securityDetailsField_
-      setStringValue:base::SysUTF16ToNSString(security_description->details)];
-
-  certificate_ = identityInfo.certificate;
-
-  if (certificate_) {
-    if (identityInfo.show_ssl_decision_revoke_button) {
-      resetDecisionsField_ =
-          [self addText:base::string16()
-               withSize:[NSFont smallSystemFontSize]
-                   bold:NO
-                 toView:securitySectionView_
-                atPoint:NSMakePoint(kSectionHorizontalPadding, 0)];
-      [resetDecisionsField_
-          setStringValue:l10n_util::GetNSString(
-                             IDS_PAGE_INFO_INVALID_CERTIFICATE_DESCRIPTION)];
-      [self sizeTextFieldHeightToFit:resetDecisionsField_];
-
-      resetDecisionsButton_ = [self
-          addLinkButtonWithText:
-              l10n_util::GetNSString(
-                  IDS_PAGE_INFO_RESET_INVALID_CERTIFICATE_DECISIONS_BUTTON)
-                         toView:securitySectionView_];
-      [resetDecisionsButton_ setTarget:self];
-      [resetDecisionsButton_ setAction:@selector(resetCertificateDecisions:)];
-
-      if (base::i18n::IsRTL()) {
-        resetDecisionsField_.alignment = NSRightTextAlignment;
-      }
-    }
-
-    // Show information about the page's certificate.
-    bool isValid =
-        (identityInfo.identity_status != PageInfo::SITE_IDENTITY_STATUS_ERROR);
-    NSString* linkText = l10n_util::GetNSString(
-        isValid ? IDS_PAGE_INFO_CERTIFICATE_VALID_LINK
-                : IDS_PAGE_INFO_CERTIFICATE_INVALID_LINK);
-
-    certificateView_ =
-        [self addInspectLinkToView:siteSettingsSectionView_
-                       sectionIcon:NSImageFromImageSkia(
-                                       PageInfoUI::GetCertificateIcon(
-                                           GetRelatedTextColor()))
-                      sectionTitle:l10n_util::GetStringUTF16(
-                                       IDS_PAGE_INFO_CERTIFICATE)
-                          linkText:linkText];
-    if (isValid) {
-      [certificateView_
-          setLinkToolTip:l10n_util::GetNSStringF(
-                             IDS_PAGE_INFO_CERTIFICATE_VALID_LINK_TOOLTIP,
-                             base::UTF8ToUTF16(
-                                 certificate_->issuer().GetDisplayName()))];
-    } else {
-      [certificateView_
-          setLinkToolTip:l10n_util::GetNSString(
-                             IDS_PAGE_INFO_CERTIFICATE_INVALID_LINK_TOOLTIP)];
-    }
-
-    [certificateView_ setLinkTarget:self
-                         withAction:@selector(showCertificateInfo:)];
-  }
-  if (identityInfo.show_change_password_buttons) {
-    whitelistPasswordReuseButton_ = [ButtonUtils
-        buttonWithTitle:l10n_util::GetNSString(
-                            IDS_PAGE_INFO_WHITELIST_PASSWORD_REUSE_BUTTON)
-                 action:@selector(whitelistPasswordReuseDecisions:)
-                 target:self];
-    [whitelistPasswordReuseButton_ sizeToFit];
-    [whitelistPasswordReuseButton_ setKeyEquivalent:kKeyEquivalentEscape];
-    [securitySectionView_ addSubview:whitelistPasswordReuseButton_];
-    changePasswordButton_ =
-        [ButtonUtils buttonWithTitle:l10n_util::GetNSString(
-                                         IDS_PAGE_INFO_CHANGE_PASSWORD_BUTTON)
-                              action:@selector(changePasswordDecisions:)
-                              target:self];
-    [changePasswordButton_ sizeToFit];
-    [changePasswordButton_ setKeyEquivalent:kKeyEquivalentReturn];
-    [securitySectionView_ addSubview:changePasswordButton_];
-  }
-
-  [self performLayout];
-}
-
-// Add a pop-up button for |permissionInfo| to the given view.
-- (NSPopUpButton*)addPopUpButtonForPermission:
-                      (const PageInfoUI::PermissionInfo&)permissionInfo
-                                       toView:(NSView*)view
-                                      atPoint:(NSPoint)point {
-  GURL url = webContents_ ? webContents_->GetURL() : GURL();
-  __block PageInfoBubbleController* weakSelf = self;
-  PermissionMenuModel::ChangeCallback callback = base::BindRepeating(
-      base::RetainBlock(^(const PageInfoUI::PermissionInfo& permission) {
-        [weakSelf onPermissionChanged:permission.type to:permission.setting];
-      }));
-  base::scoped_nsobject<PermissionSelectorButton> button(
-      [[PermissionSelectorButton alloc] initWithPermissionInfo:permissionInfo
-                                                        forURL:url
-                                                  withCallback:callback
-                                                       profile:[self profile]]);
-
-  // Determine the largest possible size for this button.
-  CGFloat maxTitleWidth =
-      [button maxTitleWidthForContentSettingsType:permissionInfo.type
-                               withDefaultSetting:permissionInfo.default_setting
-                                          profile:[self profile]];
-
-  // Ensure the containing view is large enough to contain the button with its
-  // widest possible title.
-  NSRect containerFrame = [view frame];
-  containerFrame.size.width =
-      std::max(NSWidth(containerFrame),
-               point.x + maxTitleWidth + kSectionHorizontalPadding);
-  [view setFrame:containerFrame];
-  [view addSubview:button.get()];
-
-  // Tag the button with the permission type so it can be updated later.
-  [button setTag:permissionInfo.type];
-  return button.get();
-}
-
-// Add a delete button for |objectInfo| to the given view.
-- (NSButton*)addDeleteButtonForChosenObject:(ChosenObjectInfoPtr)objectInfo
-                                     toView:(NSView*)view
-                                    atPoint:(NSPoint)point {
-  __block PageInfoBubbleController* weakSelf = self;
-  auto callback = base::BindRepeating(
-      base::RetainBlock(^(const PageInfoUI::ChosenObjectInfo& objectInfo) {
-        [weakSelf onChosenObjectDeleted:objectInfo];
-      }));
-  base::scoped_nsobject<ChosenObjectDeleteButton> button(
-      [[ChosenObjectDeleteButton alloc]
-          initWithChosenObject:std::move(objectInfo)
-                       atPoint:point
-                  withCallback:callback]);
-
-  // Ensure the containing view is large enough to contain the button.
-  NSRect containerFrame = [view frame];
-  containerFrame.size.width =
-      std::max(NSWidth(containerFrame), point.x + kPermissionDeleteImageSize +
-                                            kSectionHorizontalPadding);
-  [view setFrame:containerFrame];
-  [view addSubview:button.get()];
-  [button setTag:kChosenObjectTag];
-  return button.get();
-}
-
-// Called when the user changes the setting of a permission.
-- (void)onPermissionChanged:(ContentSettingsType)permissionType
-                         to:(ContentSetting)newSetting {
-  if (presenter_)
-    presenter_->OnSitePermissionChanged(permissionType, newSetting);
-}
-
-// Called when the user revokes permission for a previously chosen object.
-- (void)onChosenObjectDeleted:(const PageInfoUI::ChosenObjectInfo&)info {
-  if (presenter_)
-    presenter_->OnSiteChosenObjectDeleted(info.ui_info, *info.object);
-}
-
-// Adds a new row to the UI listing the permissions. Returns the NSPoint of the
-// last UI element added (either the permission button, in LTR, or the text
-// label, in RTL).
-- (NSPoint)addPermission:(const PageInfoUI::PermissionInfo&)permissionInfo
-                  toView:(NSView*)view
-                 atPoint:(NSPoint)point {
-  base::string16 labelText =
-      PageInfoUI::PermissionTypeToUIString(permissionInfo.type);
-  bool isRTL = base::i18n::IsRTL();
-  base::scoped_nsobject<NSImage> image(
-      [GetNSImageFromImageSkia(PageInfoUI::GetPermissionIcon(
-          permissionInfo, GetRelatedTextColor())) retain]);
-
-  NSPoint position;
-  NSImageView* imageView;
-  NSPopUpButton* button;
-  NSTextField* label;
-
-  CGFloat viewWidth = NSWidth([view frame]);
-
-  if (isRTL) {
-    point.x = NSWidth([view frame]) - kPermissionImageSize -
-              kSectionHorizontalPadding;
-    imageView = [self addImageWithSize:[image size] toView:view atPoint:point];
-    [imageView setImage:image];
-    point.x -= kPermissionImageSpacing;
-
-    label = [self addText:labelText
-                 withSize:[NSFont systemFontSize]
-                     bold:NO
-                   toView:view
-                  atPoint:point];
-    [label sizeToFit];
-    point.x -= NSWidth([label frame]);
-    [label setFrameOrigin:point];
-
-    position =
-        NSMakePoint(point.x, point.y + kPermissionPopupButtonYAdjustment);
-    button = [self addPopUpButtonForPermission:permissionInfo
-                                        toView:view
-                                       atPoint:position];
-    position.x -= NSWidth([button frame]);
-    [button setFrameOrigin:position];
-  } else {
-    imageView = [self addImageWithSize:[image size] toView:view atPoint:point];
-    [imageView setImage:image];
-    point.x += kPermissionImageSize + kPermissionImageSpacing;
-
-    label = [self addText:labelText
-                 withSize:[NSFont systemFontSize]
-                     bold:NO
-                   toView:view
-                  atPoint:point];
-    [label sizeToFit];
-
-    position = NSMakePoint(NSMaxX([label frame]),
-                           point.y + kPermissionPopupButtonYAdjustment);
-
-    button = [self addPopUpButtonForPermission:permissionInfo
-                                        toView:view
-                                       atPoint:position];
-  }
-  [label setToolTip:base::SysUTF16ToNSString(labelText)];
-
-  [view setFrameSize:NSMakeSize(viewWidth, NSHeight([view frame]))];
-
-  // Adjust the vertical position of the button so that its title text is
-  // aligned with the label. Assumes that the text is the same size in both.
-  // Also adjust the horizontal position to remove excess space due to the
-  // invisible bezel.
-  NSRect titleRect = [[button cell] titleRectForBounds:[button bounds]];
-  if (isRTL) {
-    position.x = kSectionHorizontalPadding;
-  } else {
-    position.x = kDefaultWindowWidth - kSectionHorizontalPadding -
-                 [button frame].size.width;
-  }
-  position.y -= titleRect.origin.y;
-  [button setFrameOrigin:position];
-
-  // Truncate the label if it's too wide.
-  // This is a workaround for https://crbug.com/654268 until MacViews ships.
-  NSRect labelFrame = [label frame];
-  if (isRTL) {
-    CGFloat maxLabelWidth = NSMaxX(labelFrame) - NSMaxX([button frame]) -
-                            kMinSeparationBetweenLabelAndMenu;
-    if (NSWidth(labelFrame) > maxLabelWidth) {
-      labelFrame.origin.x = NSMaxX(labelFrame) - maxLabelWidth;
-      labelFrame.size.width = maxLabelWidth;
-      [label setFrame:labelFrame];
-      [[label cell] setLineBreakMode:NSLineBreakByTruncatingTail];
-    }
-  } else {
-    CGFloat maxLabelWidth = NSMinX([button frame]) - NSMinX(labelFrame) -
-                            kMinSeparationBetweenLabelAndMenu;
-    if (NSWidth(labelFrame) > maxLabelWidth) {
-      labelFrame.size.width = maxLabelWidth;
-      [label setFrame:labelFrame];
-      [[label cell] setLineBreakMode:NSLineBreakByTruncatingTail];
-    }
-  }
-
-  // Align the icon with the text.
-  [self alignPermissionIcon:imageView withTextField:label];
-
-  // Permissions specified by policy or an extension cannot be changed.
-  if (permissionInfo.source == content_settings::SETTING_SOURCE_EXTENSION ||
-      permissionInfo.source == content_settings::SETTING_SOURCE_POLICY) {
-    [button setEnabled:NO];
-  }
-
-  // Update |point| to match the y of the bottomost UI element added (|button|).
-  NSRect buttonFrame = [button frame];
-  point.y = NSMaxY(labelFrame) + kPermissionLabelBottomPadding;
-
-  // Show the reason for the permission decision in a new row if it did not come
-  // from the user.
-  base::string16 reason = PageInfoUI::PermissionDecisionReasonToUIString(
-      [self profile], permissionInfo, url_);
-  if (!reason.empty()) {
-    // Do this even in RTL to make sure -addText sets the right width for the
-    // permission decision reason label.
-    point.x = kSectionHorizontalPadding + kPermissionImageSize +
-              kPermissionImageSpacing;
-
-    label = [self addText:reason
-                 withSize:[NSFont smallSystemFontSize]
-                     bold:NO
-                   toView:view
-                  atPoint:point];
-    if (isRTL) {
-      [label setAlignment:NSRightTextAlignment];
-      // Shift the reason left to align the permission label and the permission
-      // decision reason's right edges.
-      point.x -= (kPermissionImageSize + kPermissionImageSpacing);
-      [label setFrameOrigin:point];
-    }
-
-    label.textColor =
-        skia::SkColorToSRGBNSColor(PageInfoUI::GetSecondaryTextColor());
-    point.y += NSHeight(label.frame);
-  }
-
-  return NSMakePoint(NSMaxX(buttonFrame), point.y);
-}
-
-// Adds a new row to the UI listing the permissions. Returns the NSPoint of the
-// last UI element added (either the permission button, in LTR, or the text
-// label, in RTL).
-- (NSPoint)addChosenObject:(ChosenObjectInfoPtr)objectInfo
-                    toView:(NSView*)view
-                   atPoint:(NSPoint)point {
-  base::string16 labelText = l10n_util::GetStringFUTF16(
-      objectInfo->ui_info.label_string_id,
-      PageInfoUI::ChosenObjectToUIString(*objectInfo));
-  bool isRTL = base::i18n::IsRTL();
-  base::scoped_nsobject<NSImage> image(
-      [GetNSImageFromImageSkia(PageInfoUI::GetChosenObjectIcon(
-          *objectInfo, false, GetRelatedTextColor())) retain]);
-
-  NSPoint position;
-  NSImageView* imageView;
-  NSButton* button;
-  NSTextField* label;
-
-  CGFloat viewWidth = NSWidth([view frame]);
-
-  if (isRTL) {
-    point.x = NSWidth([view frame]) - kPermissionImageSize -
-              kPermissionImageSpacing - kSectionHorizontalPadding;
-    imageView = [self addImageWithSize:[image size] toView:view atPoint:point];
-    [imageView setImage:image];
-    point.x -= kPermissionImageSpacing;
-
-    label = [self addText:labelText
-                 withSize:[NSFont systemFontSize]
-                     bold:NO
-                   toView:view
-                  atPoint:point];
-    [label sizeToFit];
-    point.x -= NSWidth([label frame]);
-    [label setFrameOrigin:point];
-
-    position = NSMakePoint(point.x, point.y);
-    button = [self addDeleteButtonForChosenObject:std::move(objectInfo)
-                                           toView:view
-                                          atPoint:position];
-    position.x -= NSWidth([button frame]);
-    [button setFrameOrigin:position];
-  } else {
-    imageView = [self addImageWithSize:[image size] toView:view atPoint:point];
-    [imageView setImage:image];
-    point.x += kPermissionImageSize + kPermissionImageSpacing;
-
-    label = [self addText:labelText
-                 withSize:[NSFont systemFontSize]
-                     bold:NO
-                   toView:view
-                  atPoint:point];
-    [label sizeToFit];
-
-    position = NSMakePoint(NSMaxX([label frame]), point.y);
-    button = [self addDeleteButtonForChosenObject:std::move(objectInfo)
-                                           toView:view
-                                          atPoint:position];
-  }
-  [imageView setTag:kChosenObjectTag];
-  [label setTag:kChosenObjectTag];
-
-  [view setFrameSize:NSMakeSize(viewWidth, NSHeight([view frame]))];
-
-  // Adjust the vertical position of the button so that its title text is
-  // aligned with the label. Assumes that the text is the same size in both.
-  // Also adjust the horizontal position to remove excess space due to the
-  // invisible bezel.
-  NSRect titleRect = [[button cell] titleRectForBounds:[button bounds]];
-  position.y -= titleRect.origin.y;
-  [button setFrameOrigin:position];
-
-  // Align the icon with the text.
-  [self alignPermissionIcon:imageView withTextField:label];
-
-  NSRect buttonFrame = [button frame];
-  return NSMakePoint(NSMaxX(buttonFrame), NSMaxY(buttonFrame));
-}
-
-// Align an image with a text field by vertically centering the image on
-// the cap height of the first line of text.
-- (void)alignPermissionIcon:(NSImageView*)imageView
-              withTextField:(NSTextField*)textField {
-  NSRect frame = [imageView frame];
-  frame.origin.y += kPermissionIconYAdjustment;
-  [imageView setFrame:frame];
-}
-
-- (void)setCookieInfo:(const CookieInfoList&)cookieInfoList {
-  // |cookieInfoList| should only ever have 2 items: first- and third-party
-  // cookies.
-  DCHECK_EQ(cookieInfoList.size(), 2u);
-
-  int totalAllowed = 0;
-  for (const auto& i : cookieInfoList) {
-    totalAllowed += i.allowed;
-  }
-
-  [cookiesView_ setLinkText:l10n_util::GetPluralNSStringF(
-                                IDS_PAGE_INFO_NUM_COOKIES, totalAllowed)];
-  [self performLayout];
-}
-
-- (void)setPermissionInfo:(const PermissionInfoList&)permissionInfoList
-         andChosenObjects:(ChosenObjectInfoList)chosenObjectInfoList {
-  NSPoint controlOrigin = NSMakePoint(kSectionHorizontalPadding, 0);
-
-  // If |permissionsView_| is already populated, just handle updates to
-  // permissions made by the user here. This will avoid removing/adding new
-  // views (and thus breaking the responder chain), and also avoid immediately
-  // removing permissions set back to the default, which could be user error.
-  // Note that "update" means setPermissionInfo will only change the title of
-  // the permission |PermissionSelectorButton|. This is OK because it is not
-  // possible for the following to occur without closing the Page Info bubble:
-  //   - a permission gets changed away from the factory default (and thus needs
-  //     to be shown in Page Info)
-  //   - a permission's source changes (and becomes disabled / needs to show a
-  //     reason).
-  if ([permissionsView_ subviews].count != 0) {
-    NSView* view = nil;
-    // Remove all the chosen object views. They will be repopulated if the site
-    // still has access to them.
-    while ((view = [permissionsView_ viewWithTag:kChosenObjectTag]))
-      [view removeFromSuperview];
-
-    for (view in [permissionsView_ subviews]) {
-      // Skip views that don't need to be modified (default tags are -1 or 0).
-      if ([view tag] <= 0)
-        continue;
-
-      ContentSettingsType permissionType =
-          static_cast<ContentSettingsType>([view tag]);
-
-      PermissionSelectorButton* button =
-          base::mac::ObjCCastStrict<PermissionSelectorButton>(view);
-      const int yOrigin = [button frame].origin.y;
-      // Permissions set back to the factory default setting will disappear from
-      // |permissionInfoList|, so use |updated| to keep track of whether
-      // |button| has been updated with its new permission value yet.
-      bool updated = false;
-      for (const auto& permission : permissionInfoList) {
-        if (permissionType != permission.type)
-          continue;
-
-        updated = true;
-        [button setButtonTitle:permission profile:[self profile]];
-        break;
-      }
-
-      if (!updated) {
-        // Permissions that are no longer in |permissionInfoList| have been set
-        // back to factory default settings.
-        PageInfoUI::PermissionInfo default_info;
-        default_info.type = permissionType;
-        default_info.setting = CONTENT_SETTING_DEFAULT;
-        default_info.default_setting =
-            content_settings::ContentSettingsRegistry::GetInstance()
-                ->Get(permissionType)
-                ->GetInitialDefaultSetting();
-        default_info.source = content_settings::SETTING_SOURCE_USER;
-        default_info.is_incognito = [self profile]->IsOffTheRecord();
-        [button setButtonTitle:default_info profile:[self profile]];
-      }
-
-      // Updating the text might have changed the width of the
-      // |PermissionSelectorRow|, so reposition here.
-      if (base::i18n::IsRTL()) {
-        [button setFrameOrigin:NSMakePoint(kSectionHorizontalPadding, yOrigin)];
-      } else {
-        [button setFrameOrigin:NSMakePoint(NSWidth([permissionsView_ frame]) -
-                                               kSectionHorizontalPadding -
-                                               NSWidth([button frame]),
-                                           yOrigin)];
-      }
-    }
-    if ([[permissionsView_ subviews] count] != 0) {
-      controlOrigin.y =
-          NSMaxY([[[permissionsView_ subviews] lastObject] frame]);
-    }
-  } else {
-    // Creates permissions views (if any) for the first time.
-    for (const auto& permission : permissionInfoList) {
-      controlOrigin.y += kPermissionsVerticalSpacing;
-      NSPoint rowBottomRight = [self addPermission:permission
-                                            toView:permissionsView_
-                                           atPoint:controlOrigin];
-      controlOrigin.y = rowBottomRight.y;
-    }
-  }
-
-  for (auto& object : chosenObjectInfoList) {
-    controlOrigin.y += kPermissionsVerticalSpacing;
-    NSPoint rowBottomRight = [self addChosenObject:std::move(object)
-                                            toView:permissionsView_
-                                           atPoint:controlOrigin];
-    controlOrigin.y = rowBottomRight.y;
-  }
-
-  // |permissionsView_| was updated here, so make sure keyboard access still
-  // works by updating the responder chain.
-  [[self window] recalculateKeyViewLoop];
-
-  [permissionsView_ setFrameSize:NSMakeSize(NSWidth([permissionsView_ frame]),
-                                            controlOrigin.y)];
-  [self performLayout];
-}
-
-@end
-
-PageInfoUIBridge::PageInfoUIBridge(content::WebContents* web_contents)
-    : content::WebContentsObserver(web_contents),
-      web_contents_(web_contents),
-      bubble_controller_(nil) {
-  DCHECK(!g_page_info_bubble);
-}
-
-PageInfoUIBridge::~PageInfoUIBridge() {
-  DCHECK(g_page_info_bubble);
-  g_page_info_bubble = nullptr;
-}
-
-void PageInfoUIBridge::set_bubble_controller(
-    PageInfoBubbleController* controller) {
-  bubble_controller_ = controller;
-  g_page_info_bubble = controller;
-}
-
-void PageInfoUIBridge::SetIdentityInfo(
-    const PageInfoUI::IdentityInfo& identity_info) {
-  [bubble_controller_ setIdentityInfo:identity_info];
-}
-
-void PageInfoUIBridge::RenderFrameDeleted(
-    content::RenderFrameHost* render_frame_host) {
-  if (render_frame_host == web_contents_->GetMainFrame()) {
-    [bubble_controller_ close];
-  }
-}
-
-void PageInfoUIBridge::SetCookieInfo(const CookieInfoList& cookie_info_list) {
-  [bubble_controller_ setCookieInfo:cookie_info_list];
-}
-
-void PageInfoUIBridge::SetPermissionInfo(
-    const PermissionInfoList& permission_info_list,
-    ChosenObjectInfoList chosen_object_info_list) {
-  [bubble_controller_ setPermissionInfo:permission_info_list
-                       andChosenObjects:std::move(chosen_object_info_list)];
-}
-
-void PageInfoUIBridge::DidFinishNavigation(
-    content::NavigationHandle* navigation_handle) {
-  if (!navigation_handle->IsInMainFrame() ||
-      !navigation_handle->HasCommitted()) {
-    return;
-  }
-  // If the browser navigates to another page, close the bubble.
-  [bubble_controller_ close];
-}
-
-void ShowPageInfoDialogImpl(Browser* browser,
-                            content::WebContents* web_contents,
-                            const GURL& virtual_url,
-                            const security_state::SecurityInfo& security_info,
-                            bubble_anchor_util::Anchor anchor) {
-  if (chrome::ShowAllDialogsWithViewsToolkit()) {
-    chrome::ShowPageInfoBubbleViews(browser, web_contents, virtual_url,
-                                    security_info, anchor);
-    return;
-  }
-
-  // Don't show the bubble if it's already being shown. Since this method is
-  // called each time the location icon is clicked, each click toggles the
-  // bubble in and out.
-  if (g_page_info_bubble)
-    return;
-
-  // Create the bridge. This will be owned by the bubble controller.
-  PageInfoUIBridge* bridge = new PageInfoUIBridge(web_contents);
-  NSWindow* parent = browser->window()->GetNativeWindow();
-
-  // Create the bubble controller. It will dealloc itself when it closes,
-  // resetting |g_page_info_bubble|.
-  PageInfoBubbleController* bubble_controller =
-      [[PageInfoBubbleController alloc] initWithParentWindow:parent
-                                            pageInfoUIBridge:bridge
-                                                 webContents:web_contents
-                                                         url:virtual_url];
-
-  if (!IsInternalURL(virtual_url)) {
-    // Initialize the presenter, which holds the model and controls the UI.
-    // This is also owned by the bubble controller.
-    PageInfo* presenter =
-        new PageInfo(bridge, browser->profile(),
-                     TabSpecificContentSettings::FromWebContents(web_contents),
-                     web_contents, virtual_url, security_info);
-    [bubble_controller setPresenter:presenter];
-  }
-
-  [bubble_controller showWindow:nil];
-}
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller_unittest.mm b/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller_unittest.mm
deleted file mode 100644
index ef88c91..0000000
--- a/chrome/browser/ui/cocoa/page_info/page_info_bubble_controller_unittest.mm
+++ /dev/null
@@ -1,369 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/page_info/page_info_bubble_controller.h"
-
-#include <stddef.h>
-
-#include "base/i18n/rtl.h"
-#include "base/macros.h"
-#include "base/strings/utf_string_conversions.h"
-#include "chrome/browser/ui/browser_window.h"
-#include "chrome/browser/ui/cocoa/browser_window_controller.h"
-#include "chrome/browser/ui/cocoa/location_bar/location_bar_view_mac.h"
-#import "chrome/browser/ui/cocoa/location_bar/page_info_bubble_decoration.h"
-#import "chrome/browser/ui/cocoa/test/cocoa_profile_test.h"
-#include "content/public/test/test_web_contents_factory.h"
-#include "net/test/test_certificate_data.h"
-#include "testing/gtest_mac.h"
-
-@interface PageInfoBubbleController (ExposedForTesting)
-- (NSView*)permissionsView;
-- (NSButton*)resetDecisionsButton;
-- (NSButton*)connectionHelpButton;
-@end
-
-@implementation PageInfoBubbleController (ExposedForTesting)
-- (NSView*)permissionsView {
-  return permissionsView_;
-}
-- (NSButton*)resetDecisionsButton {
-  return resetDecisionsButton_;
-}
-- (NSButton*)connectionHelpButton {
-  return connectionHelpButton_;
-}
-- (NSButton*)changePasswordButton {
-  return changePasswordButton_;
-}
-- (NSButton*)whitelistPasswordReuseButton {
-  return whitelistPasswordReuseButton_;
-}
-@end
-
-@interface PageInfoBubbleControllerForTesting : PageInfoBubbleController {
- @private
-  CGFloat defaultWindowWidth_;
-}
-@end
-
-@implementation PageInfoBubbleControllerForTesting
-- (void)setDefaultWindowWidth:(CGFloat)width {
-  defaultWindowWidth_ = width;
-}
-- (CGFloat)defaultWindowWidth {
-  // If |defaultWindowWidth_| is 0, use the superclass implementation.
-  return defaultWindowWidth_ ? defaultWindowWidth_ : [super defaultWindowWidth];
-}
-@end
-
-namespace {
-
-// Indices of the menu items in the permission menu.
-enum PermissionMenuIndices {
-  kMenuIndexContentSettingAllow = 0,
-  kMenuIndexContentSettingBlock,
-  kMenuIndexContentSettingDefault
-};
-
-const ContentSettingsType kTestPermissionTypes[] = {
-    CONTENT_SETTINGS_TYPE_IMAGES,
-    CONTENT_SETTINGS_TYPE_MEDIASTREAM_CAMERA,
-    CONTENT_SETTINGS_TYPE_JAVASCRIPT,
-    CONTENT_SETTINGS_TYPE_PLUGINS,
-    CONTENT_SETTINGS_TYPE_POPUPS,
-    CONTENT_SETTINGS_TYPE_GEOLOCATION,
-    CONTENT_SETTINGS_TYPE_NOTIFICATIONS,
-    CONTENT_SETTINGS_TYPE_MEDIASTREAM_MIC};
-
-const ContentSetting kTestSettings[] = {
-    CONTENT_SETTING_DEFAULT, CONTENT_SETTING_DEFAULT, CONTENT_SETTING_DEFAULT,
-    CONTENT_SETTING_ALLOW,   CONTENT_SETTING_BLOCK,   CONTENT_SETTING_ALLOW,
-    CONTENT_SETTING_BLOCK,   CONTENT_SETTING_BLOCK};
-
-const ContentSetting kTestDefaultSettings[] = {
-    CONTENT_SETTING_ALLOW, CONTENT_SETTING_BLOCK, CONTENT_SETTING_ASK};
-
-const content_settings::SettingSource kTestSettingSources[] = {
-    content_settings::SETTING_SOURCE_USER,
-    content_settings::SETTING_SOURCE_USER,
-    content_settings::SETTING_SOURCE_USER,
-    content_settings::SETTING_SOURCE_USER,
-    content_settings::SETTING_SOURCE_USER,
-    content_settings::SETTING_SOURCE_POLICY,
-    content_settings::SETTING_SOURCE_POLICY,
-    content_settings::SETTING_SOURCE_EXTENSION};
-
-class PageInfoBubbleControllerTest : public CocoaProfileTest {
- public:
-  PageInfoBubbleControllerTest() { controller_ = nil; }
-
-  void TearDown() override {
-    [controller_ close];
-    CocoaProfileTest::TearDown();
-  }
-
- protected:
-  PageInfoUIBridge* bridge_;  // Weak, owned by controller.
-
-  enum MatchType { TEXT_EQUAL = 0, TEXT_NOT_EQUAL };
-
-  // Creates a new page info bubble, with the given default width.
-  // If |default_width| is 0, the *default* default width will be used.
-  void CreateBubbleWithWidth(CGFloat default_width) {
-    bridge_ = new PageInfoUIBridge(nullptr);
-
-    // The controller cleans up after itself when the window closes.
-    controller_ = [PageInfoBubbleControllerForTesting alloc];
-    [controller_ setDefaultWindowWidth:default_width];
-    [controller_ initWithParentWindow:browser()->window()->GetNativeWindow()
-                     pageInfoUIBridge:bridge_
-                          webContents:web_contents_factory_.CreateWebContents(
-                                          browser()->profile())
-                                  url:GURL("https://www.google.com")];
-    window_ = [controller_ window];
-    [controller_ showWindow:nil];
-  }
-
-  void CreateBubble() { CreateBubbleWithWidth(0.0); }
-
-  // Return a pointer to the first NSTextField found that either matches, or
-  // doesn't match, the given text.
-  NSTextField* FindTextField(MatchType match_type, NSString* text) {
-    // The window's only immediate child is an invisible view that has a flipped
-    // coordinate origin. It is into this that all views get placed.
-    NSArray* window_subviews = [[window_ contentView] subviews];
-    EXPECT_EQ(1U, [window_subviews count]);
-    NSArray* bubble_subviews = [[window_subviews lastObject] subviews];
-    NSArray* security_section_subviews =
-        [[bubble_subviews firstObject] subviews];
-
-    /**
-     *Expect 3 views:
-     * - the identity
-     * - identity status
-     * - security details link
-     */
-    EXPECT_EQ(3U, [security_section_subviews count]);
-
-    bool desired_result = match_type == TEXT_EQUAL;
-    for (NSView* view in security_section_subviews) {
-      if ([view isKindOfClass:[NSTextField class]]) {
-        NSTextField* text_field = static_cast<NSTextField*>(view);
-        if ([[text_field stringValue] isEqual:text] == desired_result)
-          return text_field;
-      }
-    }
-    return nil;
-  }
-
-  NSMutableArray* FindAllSubviewsOfClass(NSView* parent_view, Class a_class) {
-    NSMutableArray* views = [NSMutableArray array];
-    for (NSView* view in [parent_view subviews]) {
-      if ([view isKindOfClass:a_class])
-        [views addObject:view];
-    }
-    return views;
-  }
-
-  // Sets up the dialog with some test permission settings.
-  void SetTestPermissions() {
-    // Create a list of 5 different permissions, corresponding to all the
-    // possible settings:
-    // - [allow, block, ask] by default
-    // - [block, allow] * by user
-    PermissionInfoList permission_info_list;
-    PageInfoUI::PermissionInfo info;
-    for (size_t i = 0; i < arraysize(kTestPermissionTypes); ++i) {
-      info.type = kTestPermissionTypes[i];
-      info.setting = kTestSettings[i];
-      if (info.setting == CONTENT_SETTING_DEFAULT)
-        info.default_setting = kTestDefaultSettings[i];
-      info.source = kTestSettingSources[i];
-      info.is_incognito = false;
-      permission_info_list.push_back(info);
-    }
-    ChosenObjectInfoList chosen_object_info_list;
-    bridge_->SetPermissionInfo(permission_info_list,
-                               std::move(chosen_object_info_list));
-  }
-
-  int NumSettingsNotSetByUser() const {
-    int num_non_user_settings = 0;
-    for (size_t i = 0; i < arraysize(kTestSettingSources); ++i) {
-      num_non_user_settings +=
-          (kTestSettingSources[i] != content_settings::SETTING_SOURCE_USER) ? 1
-                                                                            : 0;
-    }
-    return num_non_user_settings;
-  }
-
-  content::TestWebContentsFactory web_contents_factory_;
-
-  PageInfoBubbleControllerForTesting* controller_;  // Weak, owns self.
-  NSWindow* window_;  // Weak, owned by controller.
-};
-
-TEST_F(PageInfoBubbleControllerTest, ConnectionHelpButton) {
-  PageInfoUI::IdentityInfo info;
-  info.site_identity = std::string("example.com");
-  info.identity_status = PageInfo::SITE_IDENTITY_STATUS_UNKNOWN;
-
-  CreateBubble();
-
-  bridge_->SetIdentityInfo(const_cast<PageInfoUI::IdentityInfo&>(info));
-
-  EXPECT_EQ([[controller_ connectionHelpButton] action],
-            @selector(openConnectionHelp:));
-}
-
-TEST_F(PageInfoBubbleControllerTest, ResetDecisionsButton) {
-  PageInfoUI::IdentityInfo info;
-  info.site_identity = std::string("example.com");
-  info.identity_status = PageInfo::SITE_IDENTITY_STATUS_UNKNOWN;
-
-  CreateBubble();
-
-  // Set identity info, specifying that the button should not be shown.
-  info.show_ssl_decision_revoke_button = false;
-  bridge_->SetIdentityInfo(const_cast<PageInfoUI::IdentityInfo&>(info));
-  EXPECT_EQ([controller_ resetDecisionsButton], nil);
-
-  // Set identity info, specifying that the button should be shown.
-  info.certificate = net::X509Certificate::CreateFromBytes(
-      reinterpret_cast<const char*>(google_der), sizeof(google_der));
-  ASSERT_TRUE(info.certificate);
-  info.show_ssl_decision_revoke_button = true;
-  bridge_->SetIdentityInfo(const_cast<PageInfoUI::IdentityInfo&>(info));
-  EXPECT_NE([controller_ resetDecisionsButton], nil);
-
-  // Check that clicking the button calls the right selector.
-  EXPECT_EQ([[controller_ resetDecisionsButton] action],
-            @selector(resetCertificateDecisions:));
-
-  // Since the bubble is only created once per identity, we only need to check
-  // the button is *added* when needed. So we don't check that it's removed
-  // when we set an identity with `show_ssl_decision_revoke_button == false`
-  // again.
-}
-
-TEST_F(PageInfoBubbleControllerTest, SetPermissionInfo) {
-  CreateBubble();
-  SetTestPermissions();
-
-  // There should be three subviews per permission.
-  NSArray* subviews = [[controller_ permissionsView] subviews];
-  EXPECT_EQ(arraysize(kTestPermissionTypes) * 3,
-            [subviews count] - NumSettingsNotSetByUser());
-
-  // Ensure that there is a label for each permission.
-  NSMutableArray* permission_labels = [NSMutableArray array];
-  for (NSView* view in subviews) {
-    if ([view isKindOfClass:[NSTextField class]])
-      [permission_labels
-          addObject:[static_cast<NSTextField*>(view) stringValue]];
-  }
-  EXPECT_EQ(arraysize(kTestPermissionTypes),
-            [permission_labels count] - NumSettingsNotSetByUser());
-
-  // Ensure that the button labels are distinct, and look for the correct
-  // number of disabled buttons.
-  int disabled_count = 0;
-  NSMutableSet* button_labels = [NSMutableSet set];
-  for (NSView* view in subviews) {
-    if ([view isKindOfClass:[NSPopUpButton class]]) {
-      NSPopUpButton* button = static_cast<NSPopUpButton*>(view);
-      [button_labels addObject:[[button selectedCell] title]];
-
-      if (![button isEnabled])
-        ++disabled_count;
-    }
-  }
-  EXPECT_EQ(5UL, [button_labels count]);
-
-  // Permissions with a setting source of SETTING_SOURCE_POLICY or
-  // SETTING_SOURCE_EXTENSION should have their buttons disabled.
-  EXPECT_EQ(NumSettingsNotSetByUser(), disabled_count);
-}
-
-TEST_F(PageInfoBubbleControllerTest, WindowWidth) {
-  const CGFloat kBigEnoughBubbleWidth = 310;
-  // Creating a window that should fit everything.
-  CreateBubbleWithWidth(kBigEnoughBubbleWidth);
-  SetTestPermissions();
-
-  CGFloat window_width = NSWidth([[controller_ window] frame]);
-
-  // Check the window was made bigger to fit the content.
-  EXPECT_EQ(kBigEnoughBubbleWidth, window_width);
-
-  // Check that the window is wider than the right edge of all the permission
-  // popup buttons (LTR locales) or wider than the left edge (RTL locales).
-  bool is_rtl = base::i18n::IsRTL();
-  for (NSView* view in [[controller_ permissionsView] subviews]) {
-    if (is_rtl) {
-      if ([view isKindOfClass:[NSPopUpButton class]]) {
-        NSPopUpButton* button = static_cast<NSPopUpButton*>(view);
-        EXPECT_GT(NSMinX([button frame]), 0);
-      }
-      if ([view isKindOfClass:[NSImageView class]]) {
-        NSImageView* icon = static_cast<NSImageView*>(view);
-        EXPECT_LT(NSMaxX([icon frame]), window_width);
-      }
-    } else {
-      if ([view isKindOfClass:[NSImageView class]]) {
-        NSImageView* icon = static_cast<NSImageView*>(view);
-        EXPECT_GT(NSMinX([icon frame]), 0);
-      }
-      if ([view isKindOfClass:[NSPopUpButton class]]) {
-        NSPopUpButton* button = static_cast<NSPopUpButton*>(view);
-        EXPECT_LT(NSMaxX([button frame]), window_width);
-      }
-    }
-  }
-}
-
-// Tests the page icon decoration's active state.
-TEST_F(PageInfoBubbleControllerTest, PageIconDecorationActiveState) {
-  NSWindow* window = browser()->window()->GetNativeWindow();
-  BrowserWindowController* controller =
-      [BrowserWindowController browserWindowControllerForWindow:window];
-  LocationBarDecoration* decoration =
-      [controller locationBarBridge]->page_info_decoration();
-
-  CreateBubble();
-  EXPECT_TRUE([[controller_ window] isVisible]);
-  EXPECT_TRUE(decoration->active());
-
-  [controller_ close];
-  EXPECT_FALSE(decoration->active());
-}
-
-TEST_F(PageInfoBubbleControllerTest, PasswordReuseButtons) {
-  PageInfoUI::IdentityInfo info;
-  info.site_identity = std::string("example.com");
-  info.identity_status = PageInfo::SITE_IDENTITY_STATUS_UNKNOWN;
-
-  CreateBubble();
-
-  // Set identity info, specifying that buttons should not be shown.
-  info.show_change_password_buttons = false;
-  bridge_->SetIdentityInfo(const_cast<PageInfoUI::IdentityInfo&>(info));
-  EXPECT_EQ([controller_ changePasswordButton], nil);
-  EXPECT_EQ([controller_ whitelistPasswordReuseButton], nil);
-
-  // Set identity info, specifying that buttons should be shown.
-  info.show_change_password_buttons = true;
-  bridge_->SetIdentityInfo(const_cast<PageInfoUI::IdentityInfo&>(info));
-  EXPECT_NE([controller_ changePasswordButton], nil);
-  EXPECT_NE([controller_ whitelistPasswordReuseButton], nil);
-
-  // Check that clicking the button calls the right selector.
-  EXPECT_EQ([[controller_ changePasswordButton] action],
-            @selector(changePasswordDecisions:));
-  EXPECT_EQ([[controller_ whitelistPasswordReuseButton] action],
-            @selector(whitelistPasswordReuseDecisions:));
-}
-
-}  // namespace
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_bubble_views_mac_browsertest.mm b/chrome/browser/ui/cocoa/page_info/page_info_bubble_views_mac_browsertest.mm
index 83921dac..212a2ae 100644
--- a/chrome/browser/ui/cocoa/page_info/page_info_bubble_views_mac_browsertest.mm
+++ b/chrome/browser/ui/cocoa/page_info/page_info_bubble_views_mac_browsertest.mm
@@ -66,19 +66,13 @@
       access_manager->fullscreen_controller();
 
   fullscreen_controller->ToggleBrowserFullscreenMode();
-  if (ui::MaterialDesignController::IsSecondaryUiMaterial()) {
-    views::BubbleDialogDelegateView* page_info =
-        PageInfoBubbleView::GetPageInfoBubble();
-    EXPECT_TRUE(page_info);
-    views::Widget* page_info_bubble = page_info->GetWidget();
-    EXPECT_TRUE(page_info_bubble);
-    EXPECT_EQ(GetParam().bubble_type, PageInfoBubbleView::GetShownBubbleType());
-    EXPECT_TRUE(page_info_bubble->IsVisible());
-  } else {
-    EXPECT_TRUE([PageInfoBubbleController getPageInfoBubbleForTest]);
-    // In Cocoa, the crash occurs when the bubble tries to re-layout.
-    [[PageInfoBubbleController getPageInfoBubbleForTest] performLayout];
-  }
+  views::BubbleDialogDelegateView* page_info =
+      PageInfoBubbleView::GetPageInfoBubble();
+  EXPECT_TRUE(page_info);
+  views::Widget* page_info_bubble = page_info->GetWidget();
+  EXPECT_TRUE(page_info_bubble);
+  EXPECT_EQ(GetParam().bubble_type, PageInfoBubbleView::GetShownBubbleType());
+  EXPECT_TRUE(page_info_bubble->IsVisible());
 
   // There should be no crash here from re-anchoring the Page Info bubble while
   // transitioning into full screen.
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.h b/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.h
deleted file mode 100644
index 4b09b80..0000000
--- a/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.h
+++ /dev/null
@@ -1,12 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_UTILS_COCOA_H_
-#define CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_UTILS_COCOA_H_
-
-#import <Cocoa/Cocoa.h>
-
-NSSize SizeForPageInfoButtonTitle(NSPopUpButton* button, NSString* title);
-
-#endif  // CHROME_BROWSER_UI_COCOA_PAGE_INFO_PAGE_INFO_UTILS_COCOA_H_
diff --git a/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.mm b/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.mm
deleted file mode 100644
index 1dbf9567..0000000
--- a/chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.mm
+++ /dev/null
@@ -1,23 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.h"
-
-namespace {
-// The amount of horizontal space between the button's title and its arrow icon.
-const CGFloat kButtonTitleRightPadding = 4.0f;
-}
-
-// Determine the size of a popup button with the given title.
-NSSize SizeForPageInfoButtonTitle(NSPopUpButton* button, NSString* title) {
-  NSDictionary* textAttributes =
-      [[button attributedTitle] attributesAtIndex:0 effectiveRange:NULL];
-  NSSize titleSize = [title sizeWithAttributes:textAttributes];
-
-  NSRect frame = [button frame];
-  NSRect titleRect = [[button cell] titleRectForBounds:frame];
-  CGFloat width = titleSize.width + NSWidth(frame) - NSWidth(titleRect);
-
-  return NSMakeSize(width + kButtonTitleRightPadding, NSHeight(frame));
-}
diff --git a/chrome/browser/ui/cocoa/page_info/permission_selector_button.h b/chrome/browser/ui/cocoa/page_info/permission_selector_button.h
deleted file mode 100644
index a613e0a..0000000
--- a/chrome/browser/ui/cocoa/page_info/permission_selector_button.h
+++ /dev/null
@@ -1,43 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_UI_COCOA_PAGE_INFO_PERMISSION_SELECTOR_BUTTON_H_
-#define CHROME_BROWSER_UI_COCOA_PAGE_INFO_PERMISSION_SELECTOR_BUTTON_H_
-
-#import <Cocoa/Cocoa.h>
-
-#include <memory>
-
-#include "base/mac/scoped_nsobject.h"
-#include "chrome/browser/ui/page_info/permission_menu_model.h"
-#include "components/content_settings/core/common/content_settings.h"
-
-@class MenuControllerCocoa;
-
-class Profile;
-
-@interface PermissionSelectorButton : NSPopUpButton {
- @private
-  std::unique_ptr<PermissionMenuModel> menuModel_;
-  base::scoped_nsobject<MenuControllerCocoa> menuController_;
-}
-
-// Designated initializer.
-- (id)initWithPermissionInfo:(const PageInfoUI::PermissionInfo&)permissionInfo
-                      forURL:(const GURL&)url
-                withCallback:(PermissionMenuModel::ChangeCallback)callback
-                     profile:(Profile*)profile;
-
-// Returns the largest possible size given all of the items in the menu.
-- (CGFloat)maxTitleWidthForContentSettingsType:(ContentSettingsType)type
-                            withDefaultSetting:(ContentSetting)defaultSetting
-                                       profile:(Profile*)profile;
-
-// Updates the title of the NSPopUpButton and resizes it to fit the new text.
-- (void)setButtonTitle:(const PageInfoUI::PermissionInfo&)permissionInfo
-               profile:(Profile*)profile;
-
-@end
-
-#endif  // CHROME_BROWSER_UI_COCOA_PAGE_INFO_PERMISSION_SELECTOR_BUTTON_H_
diff --git a/chrome/browser/ui/cocoa/page_info/permission_selector_button.mm b/chrome/browser/ui/cocoa/page_info/permission_selector_button.mm
deleted file mode 100644
index 0ec3193..0000000
--- a/chrome/browser/ui/cocoa/page_info/permission_selector_button.mm
+++ /dev/null
@@ -1,81 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/page_info/permission_selector_button.h"
-
-#include "base/strings/sys_string_conversions.h"
-#include "chrome/browser/ui/cocoa/page_info/page_info_utils_cocoa.h"
-#include "chrome/browser/ui/page_info/page_info_ui.h"
-#import "ui/base/cocoa/menu_controller.h"
-
-@implementation PermissionSelectorButton
-
-- (id)initWithPermissionInfo:(const PageInfoUI::PermissionInfo&)permissionInfo
-                      forURL:(const GURL&)url
-                withCallback:(PermissionMenuModel::ChangeCallback)callback
-                     profile:(Profile*)profile {
-  if (self = [super initWithFrame:NSMakeRect(0, 0, 1, 1) pullsDown:NO]) {
-    [self setFont:[NSFont systemFontOfSize:[NSFont smallSystemFontSize]]];
-    [self setBordered:NO];
-    [[self cell] setControlSize:NSSmallControlSize];
-
-    menuModel_.reset(
-        new PermissionMenuModel(profile, url, permissionInfo, callback));
-
-    menuController_.reset([[MenuControllerCocoa alloc]
-                 initWithModel:menuModel_.get()
-        useWithPopUpButtonCell:NO]);
-    [self setMenu:[menuController_ menu]];
-    [self selectItemWithTag:permissionInfo.setting];
-
-    [self setButtonTitle:permissionInfo profile:profile];
-
-    NSString* description = base::SysUTF16ToNSString(
-        PageInfoUI::PermissionTypeToUIString(permissionInfo.type));
-    [[self cell]
-        accessibilitySetOverrideValue:description
-                         forAttribute:NSAccessibilityDescriptionAttribute];
-  }
-  return self;
-}
-
-- (CGFloat)maxTitleWidthForContentSettingsType:(ContentSettingsType)type
-                            withDefaultSetting:(ContentSetting)defaultSetting
-                                       profile:(Profile*)profile {
-  // Determine the largest possible size for this button.
-  CGFloat maxTitleWidth = 0;
-  for (NSMenuItem* item in [self itemArray]) {
-    NSString* title =
-        base::SysUTF16ToNSString(PageInfoUI::PermissionActionToUIString(
-            profile, type, static_cast<ContentSetting>([item tag]),
-            defaultSetting, content_settings::SETTING_SOURCE_USER));
-    NSSize size = SizeForPageInfoButtonTitle(self, title);
-    maxTitleWidth = std::max(maxTitleWidth, size.width);
-  }
-  return maxTitleWidth;
-}
-
-// Accessor function for testing only.
-- (NSMenu*)permissionMenu {
-  return [menuController_ menu];
-}
-
-- (void)setButtonTitle:(const PageInfoUI::PermissionInfo&)permissionInfo
-               profile:(Profile*)profile {
-  // Set the button title.
-  base::scoped_nsobject<NSMenuItem> titleItem([[NSMenuItem alloc] init]);
-  base::string16 buttonTitle = PageInfoUI::PermissionActionToUIString(
-      profile, permissionInfo.type, permissionInfo.setting,
-      permissionInfo.default_setting, permissionInfo.source);
-  [titleItem setTitle:base::SysUTF16ToNSString(buttonTitle)];
-  [[self cell] setUsesItemFromMenu:NO];
-  [[self cell] setMenuItem:titleItem];
-  // Although the frame is reset, below, this sizes the cell properly.
-  [self sizeToFit];
-
-  // Size the button to just fit the visible title - not all of its items.
-  [self setFrameSize:SizeForPageInfoButtonTitle(self, [self title])];
-}
-
-@end
diff --git a/chrome/browser/ui/cocoa/page_info/permission_selector_button_unittest.mm b/chrome/browser/ui/cocoa/page_info/permission_selector_button_unittest.mm
deleted file mode 100644
index 63188c3..0000000
--- a/chrome/browser/ui/cocoa/page_info/permission_selector_button_unittest.mm
+++ /dev/null
@@ -1,74 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/page_info/permission_selector_button.h"
-
-#include "base/mac/scoped_nsobject.h"
-#include "base/test/scoped_feature_list.h"
-#import "chrome/browser/ui/cocoa/test/cocoa_test_helper.h"
-#include "chrome/browser/ui/page_info/page_info_ui.h"
-#include "chrome/test/base/testing_profile.h"
-#include "content/public/test/test_browser_thread_bundle.h"
-#include "ui/base/ui_base_features.h"
-
-@interface PermissionSelectorButton (Testing)
-- (NSMenu*)permissionMenu;
-@end
-
-namespace {
-
-const ContentSettingsType kTestPermissionType =
-    CONTENT_SETTINGS_TYPE_MEDIASTREAM_MIC;
-
-class PermissionSelectorButtonTest : public CocoaTest {
- public:
-  PermissionSelectorButtonTest() {
-    // This file only tests Cocoa UI and can be deleted when kSecondaryUiMd is
-    // default.
-    scoped_feature_list_.InitAndDisableFeature(features::kSecondaryUiMd);
-
-    got_callback_ = false;
-    PageInfoUI::PermissionInfo test_info;
-    test_info.type = kTestPermissionType;
-    test_info.setting = CONTENT_SETTING_BLOCK;
-    test_info.source = content_settings::SETTING_SOURCE_USER;
-    test_info.is_incognito = false;
-    GURL test_url("http://www.google.com");
-    PermissionMenuModel::ChangeCallback callback = base::Bind(
-        &PermissionSelectorButtonTest::Callback, base::Unretained(this));
-    view_.reset([[PermissionSelectorButton alloc]
-        initWithPermissionInfo:test_info
-                        forURL:test_url
-                  withCallback:callback
-                       profile:&profile_]);
-    [[test_window() contentView] addSubview:view_];
-  }
-
-  void Callback(const PageInfoUI::PermissionInfo& permission) {
-    EXPECT_TRUE(permission.type == kTestPermissionType);
-    got_callback_ = true;
-  }
-
-  content::TestBrowserThreadBundle thread_bundle_;
-  TestingProfile profile_;
-
-  bool got_callback_;
-  base::scoped_nsobject<PermissionSelectorButton> view_;
-
- private:
-  base::test::ScopedFeatureList scoped_feature_list_;
-
-  DISALLOW_COPY_AND_ASSIGN(PermissionSelectorButtonTest);
-};
-
-TEST_VIEW(PermissionSelectorButtonTest, view_);
-
-TEST_F(PermissionSelectorButtonTest, Callback) {
-  NSMenu* menu = [view_ permissionMenu];
-  NSMenuItem* item = [menu itemAtIndex:0];
-  [[item target] performSelector:[item action] withObject:item];
-  EXPECT_TRUE(got_callback_);
-}
-
-}  // namespace
diff --git a/chrome/browser/ui/cocoa/page_info/split_block_button.h b/chrome/browser/ui/cocoa/page_info/split_block_button.h
deleted file mode 100644
index 689ae1a..0000000
--- a/chrome/browser/ui/cocoa/page_info/split_block_button.h
+++ /dev/null
@@ -1,35 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_UI_COCOA_PAGE_INFO_SPLIT_BLOCK_BUTTON_H_
-#define CHROME_BROWSER_UI_COCOA_PAGE_INFO_SPLIT_BLOCK_BUTTON_H_
-
-#import <Cocoa/Cocoa.h>
-
-#import "chrome/browser/ui/cocoa/constrained_window/constrained_window_button.h"
-#include "ui/base/models/simple_menu_model.h"
-
-@class SplitButtonPopUpCell;
-@class SplitButtonTitleCell;
-
-// Block ('deny') button for the permissions bubble.  Subclassed from
-// ConstrainedWindowButton, so that it shares styling, but contains two cells
-// instead of just one.  The left cell behaves as a normal button, and when
-// clicked, calls the button's |action| on its |target|.  The right cell behaves
-// as a NSPopUpButtonCell, and implements a single-item menu.
-@interface SplitBlockButton : ConstrainedWindowButton {
- @private
-  base::scoped_nsobject<SplitButtonTitleCell> leftCell_;
-  base::scoped_nsobject<SplitButtonPopUpCell> rightCell_;
-
-  ui::ScopedCrTrackingArea leftTrackingArea_;
-  ui::ScopedCrTrackingArea rightTrackingArea_;
-}
-
-// Designated initializer.
-- (id)initWithMenuDelegate:(ui::SimpleMenuModel::Delegate*)menuDelegate;
-
-@end
-
-#endif  // CHROME_BROWSER_UI_COCOA_PAGE_INFO_SPLIT_BLOCK_BUTTON_H_
diff --git a/chrome/browser/ui/cocoa/page_info/split_block_button.mm b/chrome/browser/ui/cocoa/page_info/split_block_button.mm
deleted file mode 100644
index de71ab0..0000000
--- a/chrome/browser/ui/cocoa/page_info/split_block_button.mm
+++ /dev/null
@@ -1,339 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/page_info/split_block_button.h"
-
-#include <cmath>
-
-#include "base/logging.h"
-#include "base/mac/scoped_nsobject.h"
-#include "chrome/grit/generated_resources.h"
-#include "skia/ext/skia_utils_mac.h"
-#import "ui/base/cocoa/menu_controller.h"
-#include "ui/base/l10n/l10n_util_mac.h"
-#include "ui/base/models/simple_menu_model.h"
-
-namespace {
-
-enum MouseLocation {
-  kInsideLeftCell,
-  kInsideRightCell,
-  kNotInside,
-};
-
-enum CornerType {
-  kRounded,
-  kAngled,
-};
-
-NSBezierPath* PathWithCornerStyles(NSRect frame,
-                                   CornerType leftCornerStyle,
-                                   CornerType rightCornerStyle) {
-  base::scoped_nsobject<NSBezierPath> path([[NSBezierPath bezierPath] retain]);
-  const CGFloat x0 = NSMinX(frame);
-  const CGFloat x1 = NSMaxX(frame);
-  const CGFloat y0 = NSMinY(frame);
-  const CGFloat y1 = NSMaxY(frame);
-  const CGFloat radius = 2;
-
-  // Start at the center bottom.  Draw left and up, including both left corners.
-  [path moveToPoint:NSMakePoint(std::floor((x1 - x0) * .5), y0)];
-  if (leftCornerStyle == kAngled) {
-    [path lineToPoint:NSMakePoint(x0, y0)];
-    [path lineToPoint:NSMakePoint(x0, y1)];
-  } else {
-    [path appendBezierPathWithArcFromPoint:NSMakePoint(x0, y0)
-                                   toPoint:NSMakePoint(x0, y0 + radius)
-                                    radius:radius];
-    [path appendBezierPathWithArcFromPoint:NSMakePoint(x0, y1)
-                                   toPoint:NSMakePoint(x0 + radius, y1)
-                                    radius:radius];
-  }
-  // Draw through the upper right-hand and lower-right-hand corners.
-  if (rightCornerStyle == kAngled) {
-    [path lineToPoint:NSMakePoint(x1, y1)];
-    [path lineToPoint:NSMakePoint(x1, y0)];
-  } else {
-    [path appendBezierPathWithArcFromPoint:NSMakePoint(x1, y1)
-                                   toPoint:NSMakePoint(x1, y1 - radius)
-                                    radius:radius];
-    [path appendBezierPathWithArcFromPoint:NSMakePoint(x1, y0)
-                                   toPoint:NSMakePoint(x1 - radius, y0)
-                                    radius:radius];
-  }
-  return path.autorelease();
-}
-
-void DrawBezel(id<ConstrainedWindowButtonDrawableCell> cell,
-               CornerType leftCorners,
-               CornerType rightCorners,
-               NSRect frame,
-               NSView* view) {
-  if ([cell isMouseInside]) {
-    base::scoped_nsobject<NSBezierPath> path(
-        [PathWithCornerStyles(frame, leftCorners, rightCorners) retain]);
-    [ConstrainedWindowButton DrawBackgroundAndShadowForPath:path
-                                                   withCell:cell
-                                                     inView:view];
-    [ConstrainedWindowButton DrawInnerHighlightForPath:path
-                                              withCell:cell
-                                                inView:view];
-  }
-}
-
-}  // namespace
-
-// A button cell used by SplitBlockButton, containing the title.
-@interface SplitButtonTitleCell : ConstrainedWindowButtonCell
-- (NSRect)rect;
-@end
-
-// A button cell used by SplitBlockButton, containing the popup menu.
-@interface SplitButtonPopUpCell
-    : NSPopUpButtonCell<ConstrainedWindowButtonDrawableCell> {
- @private
-  BOOL isMouseInside_;
-  base::scoped_nsobject<MenuControllerCocoa> menuController_;
-  std::unique_ptr<ui::SimpleMenuModel> menuModel_;
-}
-
-// Designated initializer.
-- (id)initWithMenuDelegate:(ui::SimpleMenuModel::Delegate*)menuDelegate;
-
-- (NSRect)rect;
-
-@end
-
-@implementation SplitBlockButton
-
-- (id)initWithMenuDelegate:(ui::SimpleMenuModel::Delegate*)menuDelegate {
-  if (self = [super initWithFrame:NSZeroRect]) {
-    leftCell_.reset([[SplitButtonTitleCell alloc] init]);
-    rightCell_.reset(
-        [[SplitButtonPopUpCell alloc] initWithMenuDelegate:menuDelegate]);
-    [leftCell_ setTitle:l10n_util::GetNSString(IDS_PERMISSION_DENY)];
-    [leftCell_ setEnabled:YES];
-    [rightCell_ setEnabled:YES];
-  }
-  return self;
-}
-
-+ (Class)cellClass {
-  return nil;
-}
-
-- (NSString*)title {
-  return [leftCell_ title];
-}
-
-- (void)setAction:(SEL)action {
-  [leftCell_ setAction:action];
-}
-
-- (void)setTarget:(id)target {
-  [leftCell_ setTarget:target];
-}
-
-- (void)drawRect:(NSRect)rect {
-  // Copy the base class:  inset to leave room for the shadow.
-  --rect.size.height;
-
-  // This function assumes that |rect| is always the same as [self frame].
-  // If that changes, the drawing functions will need to be adjusted.
-  const CGFloat radius = 2;
-  NSBezierPath* path = [NSBezierPath bezierPathWithRoundedRect:rect
-                                                       xRadius:radius
-                                                       yRadius:radius];
-  [ConstrainedWindowButton DrawBackgroundAndShadowForPath:path
-                                                 withCell:nil
-                                                   inView:self];
-
-  // Use intersection rects for the cell drawing, to ensure the height
-  // adjustment is honored.
-  [leftCell_ setControlView:self];
-  [leftCell_ drawWithFrame:NSIntersectionRect(rect, [self leftCellRect])
-                    inView:self];
-
-  [rightCell_ setControlView:self];
-  [rightCell_ drawWithFrame:NSIntersectionRect(rect, [self rightCellRect])
-                     inView:self];
-
-  // Draw the border.
-  path = [NSBezierPath bezierPathWithRoundedRect:NSInsetRect(rect, 0.5, 0.5)
-                                         xRadius:radius
-                                         yRadius:radius];
-  [ConstrainedWindowButton DrawBorderForPath:path withCell:nil inView:self];
-}
-
-- (void)updateTrackingAreas {
-  [self updateTrackingArea:&leftTrackingArea_
-                   forCell:leftCell_
-                  withRect:[self leftCellRect]];
-
-  [self updateTrackingArea:&rightTrackingArea_
-                   forCell:rightCell_
-                  withRect:[self rightCellRect]];
-}
-
-- (void)updateTrackingArea:(ui::ScopedCrTrackingArea*)trackingArea
-                   forCell:(id<ConstrainedWindowButtonDrawableCell>)cell
-                  withRect:(NSRect)rect {
-  DCHECK(trackingArea);
-  NSTrackingAreaOptions options =
-      NSTrackingMouseEnteredAndExited | NSTrackingActiveInActiveApp;
-  [self removeTrackingArea:trackingArea->get()];
-  trackingArea->reset([[CrTrackingArea alloc] initWithRect:rect
-                                                   options:options
-                                                     owner:self
-                                                  userInfo:nil]);
-  [self addTrackingArea:trackingArea->get()];
-}
-
-- (void)mouseEntered:(NSEvent*)theEvent {
-  [self mouseMoved:theEvent];
-}
-
-- (void)mouseExited:(NSEvent*)theEvent {
-  [self mouseMoved:theEvent];
-}
-
-- (void)mouseMoved:(NSEvent*)theEvent {
-  MouseLocation location = [self mouseLocationForEvent:theEvent];
-  [rightCell_ setIsMouseInside:NO];
-  [leftCell_ setIsMouseInside:NO];
-  if (location == kInsideLeftCell)
-    [leftCell_ setIsMouseInside:YES];
-  else if (location == kInsideRightCell)
-    [rightCell_ setIsMouseInside:YES];
-  [self setNeedsDisplay:YES];
-}
-
-- (void)mouseDown:(NSEvent*)theEvent {
-  MouseLocation downLocation = [self mouseLocationForEvent:theEvent];
-  NSCell* focusCell = nil;
-  NSRect rect;
-  if (downLocation == kInsideLeftCell) {
-    focusCell = leftCell_.get();
-    rect = [self leftCellRect];
-  } else if (downLocation == kInsideRightCell) {
-    focusCell = rightCell_.get();
-    rect = [self rightCellRect];
-  }
-
-  do {
-    MouseLocation location = [self mouseLocationForEvent:theEvent];
-    if (location != kNotInside) {
-      [focusCell setHighlighted:YES];
-      [self setNeedsDisplay:YES];
-
-      if ([focusCell trackMouse:theEvent
-                         inRect:rect
-                         ofView:self
-                   untilMouseUp:NO]) {
-        [focusCell setState:![focusCell state]];
-        [self setNeedsDisplay:YES];
-        break;
-      } else {
-        // The above -trackMouse call returned NO, so we know that
-        // the mouse left the cell before a mouse up event occurred.
-        [focusCell setHighlighted:NO];
-        [self setNeedsDisplay:YES];
-      }
-    }
-    const NSUInteger mask = NSLeftMouseUpMask | NSLeftMouseDraggedMask;
-    theEvent = [[self window] nextEventMatchingMask:mask];
-  } while ([theEvent type] != NSLeftMouseUp);
-}
-
-- (MouseLocation)mouseLocationForEvent:(NSEvent*)theEvent {
-  MouseLocation location = kNotInside;
-  NSPoint mousePoint =
-      [self convertPoint:[theEvent locationInWindow] fromView:nil];
-  if ([self mouse:mousePoint inRect:[leftCell_ rect]])
-    location = kInsideLeftCell;
-  else if ([self mouse:mousePoint inRect:[self rightCellRect]])
-    location = kInsideRightCell;
-  return location;
-}
-
-- (void)sizeToFit {
-  NSSize leftSize = [leftCell_ cellSize];
-  NSSize rightSize = [rightCell_ cellSize];
-  NSSize size = NSMakeSize(
-      std::ceil(std::max(leftSize.width + rightSize.width,
-                         constrained_window_button::kButtonMinWidth)),
-      std::ceil(std::max(leftSize.height, rightSize.height)));
-  [self setFrameSize:size];
-}
-
-- (NSRect)leftCellRect {
-  return [leftCell_ rect];
-}
-
-- (NSRect)rightCellRect {
-  NSRect leftFrame, rightFrame;
-  NSDivideRect([self bounds], &leftFrame, &rightFrame,
-               NSWidth([self leftCellRect]), NSMinXEdge);
-  return rightFrame;
-}
-
-// Accessor for Testing.
-- (NSMenu*)menu {
-  return [rightCell_ menu];
-}
-
-@end
-
-@implementation SplitButtonTitleCell
-
-- (void)drawBezelWithFrame:(NSRect)frame inView:(NSView*)controlView {
-  DrawBezel(self, kRounded, kAngled, frame, controlView);
-}
-
-- (NSRect)rect {
-  NSSize size = [self cellSize];
-  return NSMakeRect(0, 0, std::ceil(size.width), std::ceil(size.height));
-}
-
-@end
-
-@implementation SplitButtonPopUpCell
-
-@synthesize isMouseInside = isMouseInside_;
-
-- (id)initWithMenuDelegate:(ui::SimpleMenuModel::Delegate*)menuDelegate {
-  if (self = [super initTextCell:@"" pullsDown:YES]) {
-    [self setControlSize:NSSmallControlSize];
-    [self setArrowPosition:NSPopUpArrowAtCenter];
-    [self setBordered:NO];
-    [self setBackgroundColor:[NSColor clearColor]];
-    menuModel_.reset(new ui::SimpleMenuModel(menuDelegate));
-    menuModel_->AddItemWithStringId(0, IDS_PERMISSION_CUSTOMIZE);
-    menuController_.reset([[MenuControllerCocoa alloc]
-                 initWithModel:menuModel_.get()
-        useWithPopUpButtonCell:NO]);
-    [self setMenu:[menuController_ menu]];
-    [self setUsesItemFromMenu:NO];
-  }
-  return self;
-}
-
-- (void)drawBorderAndBackgroundWithFrame:(NSRect)frame
-                                  inView:(NSView*)controlView {
-  // The arrow, which is what should be drawn by the base class, is drawn
-  // during -drawBezelWithFrame.  The only way to draw our own border with
-  // the default arrow is to make the cell unbordered, and draw the border
-  // from -drawBorderAndBackgroundWithFrame, rather than simply overriding
-  // -drawBezelWithFrame.
-  DrawBezel(self, kAngled, kRounded, frame, controlView);
-  [super drawBorderAndBackgroundWithFrame:NSOffsetRect(frame, -4, 0)
-                                   inView:controlView];
-}
-
-- (NSRect)rect {
-  NSSize size = [self cellSize];
-  return NSMakeRect(0, 0, std::ceil(size.width), std::ceil(size.height));
-}
-
-@end
diff --git a/chrome/browser/ui/cocoa/translate/translate_bubble_controller.h b/chrome/browser/ui/cocoa/translate/translate_bubble_controller.h
deleted file mode 100644
index be74fa2..0000000
--- a/chrome/browser/ui/cocoa/translate/translate_bubble_controller.h
+++ /dev/null
@@ -1,57 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef CHROME_BROWSER_UI_COCOA_TRANSLATE_TRANSLATE_BUBBLE_CONTROLLER_H_
-#define CHROME_BROWSER_UI_COCOA_TRANSLATE_TRANSLATE_BUBBLE_CONTROLLER_H_
-
-#import <Cocoa/Cocoa.h>
-
-#include <memory>
-
-#include "base/mac/scoped_nsobject.h"
-#include "chrome/browser/translate/chrome_translate_client.h"
-#import "chrome/browser/ui/cocoa/omnibox_decoration_bubble_controller.h"
-#include "components/translate/core/common/translate_errors.h"
-
-@class BrowserWindowController;
-
-class TranslateBubbleModel;
-
-namespace content {
-class WebContents;
-}
-
-// Displays the Translate bubble. The Translate bubble is a bubble which
-// pops up when clicking the Translate icon on Omnibox. This bubble
-// allows us to translate a foreign page into user-selected language,
-// revert this, and configure the translate setting.
-@interface TranslateBubbleController
-    : OmniboxDecorationBubbleController<NSTextViewDelegate> {
-  // The views of each state. The keys are TranslateBubbleModel::ViewState,
-  // and the values are NSView*.
-  base::scoped_nsobject<NSDictionary<NSNumber*, NSView*>> views_;
-
-  // The 'Try again' button on the error panel.
-  NSButton* tryAgainButton_;
-}
-
-@property(readonly, nonatomic) const content::WebContents* webContents;
-@property(readonly, nonatomic) const TranslateBubbleModel* model;
-
-- (id)initWithParentWindow:(BrowserWindowController*)controller
-                     model:(std::unique_ptr<TranslateBubbleModel>)model
-               webContents:(content::WebContents*)webContents;
-- (void)switchView:(TranslateBubbleModel::ViewState)viewState;
-- (void)switchToErrorView:(translate::TranslateErrors::Type)errorType;
-
-@end
-
-// The methods on this category are used internally by the controller and are
-// only exposed for testing purposes. DO NOT USE OTHERWISE.
-@interface TranslateBubbleController (ExposedForTesting)
-- (IBAction)handleCloseButtonPressed:(id)sender;
-- (IBAction)handleTranslateButtonPressed:(id)sender;
-@end
-
-#endif  // CHROME_BROWSER_UI_COCOA_TRANSLATE_TRANSLATE_BUBBLE_CONTROLLER_H_
diff --git a/chrome/browser/ui/cocoa/translate/translate_bubble_controller.mm b/chrome/browser/ui/cocoa/translate/translate_bubble_controller.mm
deleted file mode 100644
index 1ecb0ae..0000000
--- a/chrome/browser/ui/cocoa/translate/translate_bubble_controller.mm
+++ /dev/null
@@ -1,962 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#import "chrome/browser/ui/cocoa/translate/translate_bubble_controller.h"
-
-#include <utility>
-
-#include "base/mac/foundation_util.h"
-#include "base/mac/scoped_nsobject.h"
-#include "base/macros.h"
-#include "base/metrics/histogram_macros.h"
-#include "base/strings/sys_string_conversions.h"
-#include "chrome/browser/ui/chrome_pages.h"
-#import "chrome/browser/ui/cocoa/browser_window_controller.h"
-#import "chrome/browser/ui/cocoa/bubble_combobox.h"
-#include "chrome/browser/ui/cocoa/chrome_style.h"
-#import "chrome/browser/ui/cocoa/hover_close_button.h"
-#import "chrome/browser/ui/cocoa/info_bubble_view.h"
-#import "chrome/browser/ui/cocoa/info_bubble_window.h"
-#import "chrome/browser/ui/cocoa/location_bar/location_bar_view_mac.h"
-#import "chrome/browser/ui/cocoa/location_bar/translate_decoration.h"
-#include "chrome/browser/ui/translate/language_combobox_model.h"
-#include "chrome/browser/ui/translate/translate_bubble_model.h"
-#include "chrome/browser/ui/translate/translate_bubble_view_state_transition.h"
-#include "chrome/common/url_constants.h"
-#include "chrome/grit/generated_resources.h"
-#include "chrome/grit/theme_resources.h"
-#include "components/strings/grit/components_strings.h"
-#include "components/translate/core/browser/translate_ui_delegate.h"
-#include "content/public/browser/browser_context.h"
-#include "content/public/common/referrer.h"
-#include "skia/ext/skia_utils_mac.h"
-#include "ui/base/cocoa/cocoa_base_utils.h"
-#import "ui/base/cocoa/controls/hyperlink_button_cell.h"
-#include "ui/base/cocoa/controls/hyperlink_text_view.h"
-#import "ui/base/cocoa/window_size_constants.h"
-#include "ui/base/l10n/l10n_util.h"
-#include "ui/base/models/combobox_model.h"
-
-// TODO(hajimehoshi): This class is almost same as that of views. Refactor them.
-class TranslateDenialComboboxModel : public ui::ComboboxModel {
- public:
-  explicit TranslateDenialComboboxModel(
-      const base::string16& original_language_name) {
-    // Dummy menu item, which is shown on the top of a NSPopUpButton. The top
-    // text of the denial pop up menu should be IDS_TRANSLATE_BUBBLE_DENY, while
-    // it is impossible to use it here because NSPopUpButtons' addItemWithTitle
-    // removes a duplicated menu item. Instead, the title will be set later by
-    // NSMenuItem's setTitle.
-    items_.push_back(base::string16());
-
-    // Menu items in the drop down menu.
-    items_.push_back(l10n_util::GetStringFUTF16(
-        IDS_TRANSLATE_BUBBLE_NEVER_TRANSLATE_LANG,
-        original_language_name));
-    items_.push_back(l10n_util::GetStringUTF16(
-        IDS_TRANSLATE_BUBBLE_NEVER_TRANSLATE_SITE));
-  }
-  ~TranslateDenialComboboxModel() override {}
-
- private:
-  // ComboboxModel:
-  int GetItemCount() const override { return items_.size(); }
-  base::string16 GetItemAt(int index) override { return items_[index]; }
-  bool IsItemSeparatorAt(int index) override { return false; }
-  int GetDefaultIndex() const override { return 0; }
-
-  std::vector<base::string16> items_;
-
-  DISALLOW_COPY_AND_ASSIGN(TranslateDenialComboboxModel);
-};
-
-const CGFloat kTranslateBubbleIconWidth = 30;
-const CGFloat kTranslateBubbleIconHeight = 30;
-const CGFloat kTranslateBubbleIconPadding = 12;
-
-const CGFloat kTranslateBubbleWindowWidth = 320;
-
-// Padding between the window frame and content.
-const CGFloat kTranslateBubbleFramePadding = 16;
-
-const CGFloat kTranslateBubbleRelatedControlHorizontalSpacing = -2;
-
-const CGFloat kTranslateBubbleRelatedControlVerticalSpacing = 8;
-const CGFloat kTranslateBubbleUnrelatedControlVerticalSpacing = 20;
-
-const CGFloat kTranslateBubbleContentWidth =
-    kTranslateBubbleWindowWidth - 2 * kTranslateBubbleFramePadding;
-
-@interface TranslateBubbleController()
-
-- (void)performLayout;
-- (NSView*)newBeforeTranslateView;
-- (NSView*)newTranslatingView;
-- (NSView*)newAfterTranslateView;
-- (NSView*)newErrorView;
-- (NSView*)newAdvancedView;
-- (void)updateAdvancedView;
-- (void)updateAlwaysCheckboxes;
-- (NSImageView*)addIcon:(NSView*)view;
-- (NSTextView*)addStyledTextView:(NSString*)message
-                          toView:(NSView*)view
-                      withRanges:(std::vector<NSRange>)ranges
-                        delegate:(id<NSTextViewDelegate>)delegate;
-- (NSTextField*)addText:(NSString*)text
-                 toView:(NSView*)view;
-- (NSButton*)addLinkButtonWithText:(NSString*)text
-                            action:(SEL)action
-                            toView:(NSView*)view;
-- (NSButton*)addButton:(NSString*)title
-                action:(SEL)action
-                toView:(NSView*)view;
-- (NSButton*)addCheckbox:(NSString*)title
-                  action:(SEL)action
-                  toView:(NSView*)view;
-- (NSButton*)addCloseButton:(NSView*)view action:(SEL)action;
-- (NSPopUpButton*)addPopUpButton:(ui::ComboboxModel*)model
-                          action:(SEL)action
-                          toView:(NSView*)view;
-- (IBAction)handleAlwaysTranslateCheckboxPressed:(id)sender;
-- (IBAction)handleDoneButtonPressed:(id)sender;
-- (IBAction)handleCancelButtonPressed:(id)sender;
-- (IBAction)handleCloseButtonPressed:(id)sender;
-- (IBAction)handleShowOriginalButtonPressed:(id)sender;
-- (IBAction)handleTryAgainButtonPressed:(id)sender;
-- (IBAction)handleAdvancedLinkButtonPressed:(id)sender;
-- (IBAction)handleLanguageSettingsLinkButtonPressed:(id)sender;
-- (IBAction)handleDenialPopUpButtonNeverTranslateLanguageSelected:(id)sender;
-- (IBAction)handleDenialPopUpButtonNeverTranslateSiteSelected:(id)sender;
-- (IBAction)handleSourceLanguagePopUpButtonSelectedItemChanged:(id)sender;
-- (IBAction)handleTargetLanguagePopUpButtonSelectedItemChanged:(id)sender;
-- (BOOL)textView:(NSTextView*)aTextView
-    clickedOnLink:(id)link
-          atIndex:(NSUInteger)charIndex;
-@end
-
-@implementation TranslateBubbleController {
- @private
-  content::WebContents* webContents_;
-  std::unique_ptr<TranslateBubbleModel> model_;
-
-  // The 'Done' or 'Translate' button on the advanced (option) panel.
-  NSButton* advancedDoneButton_;
-
-  // The 'Cancel' button on the advanced (option) panel.
-  NSButton* advancedCancelButton_;
-
-  // The 'Always translate' checkbox on the before panel.
-  // This is nil when the current WebContents is in an incognito window.
-  NSButton* beforeAlwaysTranslateCheckbox_;
-
-  // The 'Always translate' checkbox on the advanced (option) panel.
-  // This is nil when the current WebContents is in an incognito window.
-  NSButton* advancedAlwaysTranslateCheckbox_;
-
-  // The '[x]' close button on the upper right side of the before panel.
-  NSButton* closeButton_;
-
-  // The combobox model which is used to deny translation at the view before
-  // translate.
-  std::unique_ptr<TranslateDenialComboboxModel> translateDenialComboboxModel_;
-
-  // The combobox model for source languages on the advanced (option) panel.
-  std::unique_ptr<LanguageComboboxModel> sourceLanguageComboboxModel_;
-
-  // The combobox model for target languages on the advanced (option) panel.
-  std::unique_ptr<LanguageComboboxModel> targetLanguageComboboxModel_;
-
-  // Whether the translation is actually executed once at least.
-  BOOL translateExecuted_;
-
-  // The state of the 'Always ...' checkboxes.
-  BOOL shouldAlwaysTranslate_;
-}
-
-@synthesize webContents = webContents_;
-
-- (id)initWithParentWindow:(BrowserWindowController*)controller
-                     model:(std::unique_ptr<TranslateBubbleModel>)model
-               webContents:(content::WebContents*)webContents {
-  NSWindow* parentWindow = [controller window];
-
-  // Use an arbitrary size; it will be changed in performLayout.
-  NSRect contentRect = ui::kWindowSizeDeterminedLater;
-  base::scoped_nsobject<InfoBubbleWindow> window(
-      [[InfoBubbleWindow alloc] initWithContentRect:contentRect
-                                          styleMask:NSBorderlessWindowMask
-                                            backing:NSBackingStoreBuffered
-                                              defer:NO]);
-
-  // Disable animations - otherwise, the window/controller will outlive the web
-  // contents it's associated with.
-  [window setAllowedAnimations:info_bubble::kAnimateNone];
-
-  if ((self = [super initWithWindow:window
-                       parentWindow:parentWindow
-                         anchoredAt:NSZeroPoint])) {
-    webContents_ = webContents;
-    model_ = std::move(model);
-
-    shouldAlwaysTranslate_ = model_->ShouldAlwaysTranslate();
-    if (!webContents_->GetBrowserContext()->IsOffTheRecord()) {
-      shouldAlwaysTranslate_ =
-          model_->ShouldAlwaysTranslateBeCheckedByDefault();
-    }
-
-    if (model_->GetViewState() !=
-        TranslateBubbleModel::VIEW_STATE_BEFORE_TRANSLATE) {
-      translateExecuted_ = YES;
-    }
-
-    views_.reset([@{
-        @(TranslateBubbleModel::VIEW_STATE_BEFORE_TRANSLATE):
-            [self newBeforeTranslateView],
-        @(TranslateBubbleModel::VIEW_STATE_TRANSLATING):
-            [self newTranslatingView],
-        @(TranslateBubbleModel::VIEW_STATE_AFTER_TRANSLATE):
-            [self newAfterTranslateView],
-        @(TranslateBubbleModel::VIEW_STATE_ERROR):
-            [self newErrorView],
-        @(TranslateBubbleModel::VIEW_STATE_ADVANCED):
-            [self newAdvancedView],
-    } retain]);
-
-    // The [X] Close Button.
-    closeButton_ = [self addCloseButton:[[self window] contentView]
-                                 action:@selector(handleCloseButtonPressed:)];
-
-    [self performLayout];
-    translate::ReportUiAction(translate::BUBBLE_SHOWN);
-  }
-  return self;
-}
-
-- (void)windowWillClose:(NSNotification*)notification {
-  model_->OnBubbleClosing();
-  [super windowWillClose:notification];
-}
-
-- (NSView*)currentView {
-  NSNumber* key = @(model_->GetViewState());
-  NSView* view = [views_ objectForKey:key];
-  DCHECK(view);
-  return view;
-}
-
-- (const TranslateBubbleModel*)model {
-  return model_.get();
-}
-
-- (void)showWindow:(id)sender {
-  BrowserWindowController* browserWindowController = [BrowserWindowController
-      browserWindowControllerForWindow:self.parentWindow];
-  LocationBarViewMac* locationBar = [browserWindowController locationBarBridge];
-  if (locationBar) {
-    NSPoint anchorPoint =
-        locationBar->GetBubblePointForDecoration([self decorationForBubble]);
-    anchorPoint =
-        ui::ConvertPointFromWindowToScreen([self parentWindow], anchorPoint);
-    [self setAnchorPoint:anchorPoint];
-  }
-  [super showWindow:sender];
-}
-
-- (LocationBarDecoration*)decorationForBubble {
-  BrowserWindowController* browserWindowController = [BrowserWindowController
-      browserWindowControllerForWindow:self.parentWindow];
-  LocationBarViewMac* locationBar = [browserWindowController locationBarBridge];
-  return locationBar ? locationBar->translate_decoration() : nullptr;
-}
-
-- (void)switchView:(TranslateBubbleModel::ViewState)viewState {
-  if (model_->GetViewState() == viewState)
-    return;
-
-  model_->SetViewState(viewState);
-  [self performLayout];
-}
-
-- (void)switchToErrorView:(translate::TranslateErrors::Type)errorType {
-  [self switchView:TranslateBubbleModel::VIEW_STATE_ERROR];
-  model_->ShowError(errorType);
-}
-
-- (void)performLayout {
-  [self updateAlwaysCheckboxes];
-  NSWindow* window = [self window];
-  [[window contentView] setSubviews:@[ [self currentView], closeButton_ ]];
-
-  CGFloat height = NSHeight([[self currentView] frame]) +
-                   2 * kTranslateBubbleFramePadding +
-                   info_bubble::kBubbleArrowHeight;
-
-  NSRect windowFrame = [window contentRectForFrameRect:[[self window] frame]];
-  NSRect newWindowFrame = [window
-      frameRectForContentRect:NSMakeRect(NSMinX(windowFrame),
-                                         NSMaxY(windowFrame) - height,
-                                         kTranslateBubbleWindowWidth, height)];
-
-  // Adjust the origin of closeButton.
-  CGFloat closeX = kTranslateBubbleWindowWidth -
-                   chrome_style::kCloseButtonPadding -
-                   chrome_style::GetCloseButtonSize();
-  CGFloat closeY = height - kTranslateBubbleFramePadding -
-                   chrome_style::kCloseButtonPadding -
-                   info_bubble::kBubbleArrowHeight;
-  [closeButton_ setFrameOrigin:NSMakePoint(closeX, closeY)];
-
-  [window setFrame:newWindowFrame
-           display:YES
-           animate:[[self window] isVisible]];
-}
-
-- (NSView*)newBeforeTranslateView {
-  NSRect contentFrame =
-      NSMakeRect(kTranslateBubbleFramePadding, kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth, 0);
-  NSView* view = [[NSView alloc] initWithFrame:contentFrame];
-
-  base::string16 originalLanguageName =
-      model_->GetLanguageNameAt(model_->GetOriginalLanguageIndex());
-  base::string16 targetLanguageName =
-      model_->GetLanguageNameAt(model_->GetTargetLanguageIndex());
-
-  std::vector<size_t> offsets;
-  NSString* message = l10n_util::GetNSStringF(
-      IDS_TRANSLATE_BUBBLE_BEFORE_TRANSLATE_NEW, originalLanguageName,
-      targetLanguageName, &offsets);
-  std::vector<NSRange> ranges;
-  ranges.push_back(NSMakeRange(offsets[0], originalLanguageName.length()));
-  ranges.push_back(NSMakeRange(offsets[1], targetLanguageName.length()));
-
-  NSTextView* textLabel = [self addStyledTextView:message
-                                           toView:view
-                                       withRanges:ranges
-                                         delegate:self];
-
-  NSString* title =
-      l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ACCEPT);
-  NSButton* translateButton =
-      [self addButton:title
-               action:@selector(handleTranslateButtonPressed:)
-               toView:view];
-  [translateButton setKeyEquivalent:@"\r"];
-
-  // TODO(hajimehoshi): When TranslateDenialComboboxModel is factored out as a
-  // common model, ui::MenuModel will be used here.
-  translateDenialComboboxModel_.reset(
-      new TranslateDenialComboboxModel(originalLanguageName));
-  NSPopUpButton* denyPopUpButton =
-      [self addPopUpButton:translateDenialComboboxModel_.get()
-                    action:nil
-                    toView:view];
-  [denyPopUpButton setPullsDown:YES];
-  [[denyPopUpButton itemAtIndex:1] setTarget:self];
-  [[denyPopUpButton itemAtIndex:1]
-      setAction:@selector(
-                    handleDenialPopUpButtonNeverTranslateLanguageSelected:)];
-  [[denyPopUpButton itemAtIndex:2] setTarget:self];
-  [[denyPopUpButton itemAtIndex:2]
-      setAction:@selector(handleDenialPopUpButtonNeverTranslateSiteSelected:)];
-
-  title = base::SysUTF16ToNSString(
-      l10n_util::GetStringUTF16(IDS_TRANSLATE_BUBBLE_OPTIONS_MENU_BUTTON));
-  [[denyPopUpButton itemAtIndex:0] setTitle:title];
-
-  // Adjust width for the first item.
-  base::scoped_nsobject<NSMenu> originalMenu([[denyPopUpButton menu] copy]);
-  [denyPopUpButton removeAllItems];
-  [denyPopUpButton addItemWithTitle:[[originalMenu itemAtIndex:0] title]];
-  [denyPopUpButton sizeToFit];
-  [denyPopUpButton setMenu:originalMenu];
-
-  // 'Always translate' checkbox
-  if (!webContents_->GetBrowserContext()->IsOffTheRecord()) {
-    title =
-        l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ALWAYS_DO_THIS);
-    SEL action = @selector(handleAlwaysTranslateCheckboxPressed:);
-    beforeAlwaysTranslateCheckbox_ =
-        [self addCheckbox:title action:action toView:view];
-  }
-
-  NSImageView* icon = [self addIcon:view];
-
-  // Layout
-  CGFloat yPos = 0;
-
-  [translateButton
-      setFrameOrigin:NSMakePoint(kTranslateBubbleContentWidth -
-                                     NSWidth([translateButton frame]),
-                                 yPos)];
-
-  NSRect denyPopUpButtonFrame = [denyPopUpButton frame];
-  CGFloat diffY = [[denyPopUpButton cell]
-    titleRectForBounds:[denyPopUpButton bounds]].origin.y;
-  [denyPopUpButton
-      setFrameOrigin:NSMakePoint(
-                         NSMinX([translateButton frame]) -
-                             denyPopUpButtonFrame.size.width -
-                             kTranslateBubbleRelatedControlHorizontalSpacing,
-                         yPos + diffY)];
-
-  yPos += NSHeight([translateButton frame]) +
-          kTranslateBubbleUnrelatedControlVerticalSpacing;
-
-  if (beforeAlwaysTranslateCheckbox_) {
-    [beforeAlwaysTranslateCheckbox_
-        setFrameOrigin:NSMakePoint(kTranslateBubbleIconWidth +
-                                       kTranslateBubbleIconPadding,
-                                   yPos)];
-
-    yPos += NSHeight([beforeAlwaysTranslateCheckbox_ frame]) +
-            kTranslateBubbleRelatedControlVerticalSpacing;
-  }
-
-  [textLabel setFrameOrigin:NSMakePoint(kTranslateBubbleIconWidth +
-                                            kTranslateBubbleIconPadding,
-                                        yPos)];
-
-  yPos = NSMaxY([textLabel frame]);
-  [icon setFrameOrigin:NSMakePoint(0, yPos - kTranslateBubbleIconHeight)];
-  [view setFrameSize:NSMakeSize(kTranslateBubbleContentWidth, yPos)];
-
-  return view;
-}
-
-- (NSView*)newTranslatingView {
-  NSRect contentFrame =
-      NSMakeRect(kTranslateBubbleFramePadding, kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth, 0);
-  NSView* view = [[NSView alloc] initWithFrame:contentFrame];
-
-  NSString* message =
-      l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_TRANSLATING);
-  NSTextField* textLabel = [self addText:message
-                                  toView:view];
-  NSString* title =
-      l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_REVERT);
-  NSButton* showOriginalButton =
-      [self addButton:title
-               action:@selector(handleShowOriginalButtonPressed:)
-               toView:view];
-  [showOriginalButton setEnabled:NO];
-  NSImageView* icon = [self addIcon:view];
-
-  // Layout
-  // TODO(hajimehoshi): Use l10n_util::VerticallyReflowGroup.
-  CGFloat yPos = 0;
-
-  [showOriginalButton
-      setFrameOrigin:NSMakePoint(kTranslateBubbleContentWidth -
-                                     NSWidth([showOriginalButton frame]),
-                                 yPos)];
-
-  yPos += NSHeight([showOriginalButton frame]) +
-          kTranslateBubbleUnrelatedControlVerticalSpacing;
-
-  [textLabel setFrameOrigin:NSMakePoint(kTranslateBubbleIconWidth +
-                                            kTranslateBubbleIconPadding,
-                                        yPos)];
-
-  yPos = NSMaxY([textLabel frame]);
-  [icon setFrameOrigin:NSMakePoint(0, yPos - kTranslateBubbleIconHeight)];
-  [view setFrameSize:NSMakeSize(kTranslateBubbleContentWidth, yPos)];
-
-  return view;
-}
-
-- (NSView*)newAfterTranslateView {
-  NSRect contentFrame =
-      NSMakeRect(kTranslateBubbleFramePadding, kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth, 0);
-  NSView* view = [[NSView alloc] initWithFrame:contentFrame];
-
-  NSString* message =
-      l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_TRANSLATED);
-  NSTextField* textLabel = [self addText:message
-                                  toView:view];
-  message = l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ADVANCED_LINK);
-  NSButton* advancedLinkButton =
-      [self addLinkButtonWithText:message
-                           action:@selector(handleAdvancedLinkButtonPressed:)
-                           toView:view];
-  NSString* title =
-      l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_REVERT);
-  NSButton* showOriginalButton =
-      [self addButton:title
-               action:@selector(handleShowOriginalButtonPressed:)
-               toView:view];
-
-  NSImageView* icon = [self addIcon:view];
-
-  // Layout
-  CGFloat yPos = 0;
-
-  [showOriginalButton
-      setFrameOrigin:NSMakePoint(kTranslateBubbleContentWidth -
-                                     NSWidth([showOriginalButton frame]),
-                                 yPos)];
-
-  yPos += NSHeight([showOriginalButton frame]) +
-          kTranslateBubbleUnrelatedControlVerticalSpacing;
-
-  [textLabel setFrameOrigin:NSMakePoint(kTranslateBubbleIconWidth +
-                                            kTranslateBubbleIconPadding,
-                                        yPos)];
-  [advancedLinkButton setFrameOrigin:NSMakePoint(
-      NSMaxX([textLabel frame]), yPos)];
-
-  yPos = NSMaxY([textLabel frame]);
-  [icon setFrameOrigin:NSMakePoint(0, yPos - kTranslateBubbleIconHeight)];
-  [view setFrameSize:NSMakeSize(kTranslateBubbleContentWidth, yPos)];
-
-  return view;
-}
-
-- (NSView*)newErrorView {
-  NSRect contentFrame =
-      NSMakeRect(kTranslateBubbleFramePadding, kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth, 0);
-  NSView* view = [[NSView alloc] initWithFrame:contentFrame];
-
-  NSString* message =
-      l10n_util::GetNSString(IDS_TRANSLATE_BUBBLE_COULD_NOT_TRANSLATE);
-  NSTextField* textLabel = [self addText:message toView:view];
-  message = l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ADVANCED_LINK);
-  NSButton* advancedLinkButton =
-      [self addLinkButtonWithText:message
-                           action:@selector(handleAdvancedLinkButtonPressed:)
-                           toView:view];
-  NSString* title =
-      l10n_util::GetNSString(IDS_TRANSLATE_BUBBLE_TRY_AGAIN);
-  tryAgainButton_ = [self addButton:title
-                             action:@selector(handleTryAgainButtonPressed:)
-                             toView:view];
-
-  NSImageView* icon = [self addIcon:view];
-
-  // Layout
-  CGFloat yPos = 0;
-
-  [tryAgainButton_
-      setFrameOrigin:NSMakePoint(kTranslateBubbleContentWidth -
-                                     NSWidth([tryAgainButton_ frame]),
-                                 yPos)];
-
-  yPos += NSHeight([tryAgainButton_ frame]) +
-          kTranslateBubbleUnrelatedControlVerticalSpacing;
-
-  [textLabel setFrameOrigin:NSMakePoint(kTranslateBubbleIconWidth +
-                                            kTranslateBubbleIconPadding,
-                                        yPos)];
-  [advancedLinkButton
-      setFrameOrigin:NSMakePoint(NSMaxX([textLabel frame]), yPos)];
-
-  yPos = NSMaxY([textLabel frame]);
-  [icon setFrameOrigin:NSMakePoint(0, yPos - kTranslateBubbleIconHeight)];
-  [view setFrameSize:NSMakeSize(kTranslateBubbleContentWidth, yPos)];
-
-  return view;
-}
-
-- (NSView*)newAdvancedView {
-  NSRect contentFrame =
-      NSMakeRect(kTranslateBubbleFramePadding, kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth, 0);
-  NSView* view = [[NSView alloc] initWithFrame:contentFrame];
-
-  NSString* title = l10n_util::GetNSStringWithFixup(
-      IDS_TRANSLATE_BUBBLE_PAGE_LANGUAGE);
-  NSTextField* sourceLanguageLabel = [self addText:title
-                                            toView:view];
-  title = l10n_util::GetNSStringWithFixup(
-      IDS_TRANSLATE_BUBBLE_TRANSLATION_LANGUAGE);
-  NSTextField* targetLanguageLabel = [self addText:title
-                                            toView:view];
-
-  // combobox
-  int sourceDefaultIndex = model_->GetOriginalLanguageIndex();
-  int targetDefaultIndex = model_->GetTargetLanguageIndex();
-  sourceLanguageComboboxModel_.reset(
-      new LanguageComboboxModel(sourceDefaultIndex, model_.get()));
-  targetLanguageComboboxModel_.reset(
-      new LanguageComboboxModel(targetDefaultIndex, model_.get()));
-  SEL action = @selector(handleSourceLanguagePopUpButtonSelectedItemChanged:);
-  NSPopUpButton* sourcePopUpButton =
-      [self addPopUpButton:sourceLanguageComboboxModel_.get()
-                    action:action
-                    toView:view];
-  action = @selector(handleTargetLanguagePopUpButtonSelectedItemChanged:);
-  NSPopUpButton* targetPopUpButton =
-      [self addPopUpButton:targetLanguageComboboxModel_.get()
-                    action:action
-                    toView:view];
-
-  // 'Always translate' checkbox
-  if (!webContents_->GetBrowserContext()->IsOffTheRecord()) {
-    NSString* title =
-        l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ALWAYS);
-    action = @selector(handleAlwaysTranslateCheckboxPressed:);
-    advancedAlwaysTranslateCheckbox_ =
-        [self addCheckbox:title action:action toView:view];
-  }
-
-  // Buttons
-  advancedDoneButton_ =
-      [self addButton:l10n_util::GetNSStringWithFixup(IDS_DONE)
-               action:@selector(handleDoneButtonPressed:)
-               toView:view];
-  [advancedDoneButton_ setKeyEquivalent:@"\r"];
-  advancedCancelButton_ =
-      [self addButton:l10n_util::GetNSStringWithFixup(IDS_CANCEL)
-               action:@selector(handleCancelButtonPressed:)
-               toView:view];
-
-  NSString* message = l10n_util::GetNSStringWithFixup(
-        IDS_TRANSLATE_BUBBLE_LANGUAGE_SETTINGS);
-  action = @selector(handleLanguageSettingsLinkButtonPressed:);
-  NSButton* languageSettingsLinkButton =
-      [self addLinkButtonWithText:message
-                           action:action
-                           toView:view];
-
-  // Layout
-  CGFloat textLabelWidth = NSWidth([sourceLanguageLabel frame]);
-  if (textLabelWidth < NSWidth([targetLanguageLabel frame]))
-    textLabelWidth = NSWidth([targetLanguageLabel frame]);
-
-  CGFloat yPos = 0;
-
-  [advancedDoneButton_ setFrameOrigin:NSMakePoint(0, yPos)];
-  [advancedCancelButton_ setFrameOrigin:NSMakePoint(0, yPos)];
-
-  [languageSettingsLinkButton setFrameOrigin:NSMakePoint(0, yPos)];
-
-  // Vertical center the languageSettingsLinkButton with the
-  // advancedDoneButton_. Move the link position by 1px to make the baseline of
-  // the text inside the link vertically align with the text inside the buttons.
-  yPos = 1 + floor((NSHeight([advancedDoneButton_ frame]) -
-         NSHeight([languageSettingsLinkButton frame])) / 2);
-  [languageSettingsLinkButton setFrameOrigin:NSMakePoint(0, yPos)];
-
-  yPos = NSHeight([advancedDoneButton_ frame]) +
-         kTranslateBubbleUnrelatedControlVerticalSpacing;
-
-  if (advancedAlwaysTranslateCheckbox_) {
-    [advancedAlwaysTranslateCheckbox_
-        setFrameOrigin:NSMakePoint(textLabelWidth, yPos)];
-
-    yPos += NSHeight([advancedAlwaysTranslateCheckbox_ frame]) +
-            kTranslateBubbleRelatedControlVerticalSpacing;
-  }
-
-  CGFloat diffY = [[sourcePopUpButton cell]
-                   titleRectForBounds:[sourcePopUpButton bounds]].origin.y;
-
-  [targetLanguageLabel setFrameOrigin:NSMakePoint(
-      textLabelWidth - NSWidth([targetLanguageLabel frame]), yPos + diffY)];
-
-  NSRect frame = [targetPopUpButton frame];
-  frame.origin = NSMakePoint(textLabelWidth, yPos);
-  frame.size.width =
-      (kTranslateBubbleWindowWidth - 2 * kTranslateBubbleFramePadding) -
-      textLabelWidth;
-  [targetPopUpButton setFrame:frame];
-
-  yPos += NSHeight([targetPopUpButton frame]) +
-          kTranslateBubbleRelatedControlVerticalSpacing;
-
-  [sourceLanguageLabel setFrameOrigin:NSMakePoint(
-      textLabelWidth - NSWidth([sourceLanguageLabel frame]), yPos + diffY)];
-
-  frame = [sourcePopUpButton frame];
-  frame.origin = NSMakePoint(textLabelWidth, yPos);
-  frame.size.width = NSWidth([targetPopUpButton frame]);
-  [sourcePopUpButton setFrame:frame];
-
-  [view setFrameSize:NSMakeSize(kTranslateBubbleContentWidth,
-                                NSMaxY([sourcePopUpButton frame]) +
-                                    kTranslateBubbleIconPadding)];
-
-  [self updateAdvancedView];
-
-  return view;
-}
-
-- (void)updateAdvancedView {
-  NSString* title;
-  if (model_->IsPageTranslatedInCurrentLanguages())
-    title = l10n_util::GetNSStringWithFixup(IDS_DONE);
-  else
-    title = l10n_util::GetNSStringWithFixup(IDS_TRANSLATE_BUBBLE_ACCEPT);
-  [advancedDoneButton_ setTitle:title];
-  [advancedDoneButton_ sizeToFit];
-
-  NSRect frame = [advancedDoneButton_ frame];
-  frame.origin.x =
-      (kTranslateBubbleWindowWidth - 2 * kTranslateBubbleFramePadding) -
-      NSWidth(frame);
-  [advancedDoneButton_ setFrameOrigin:frame.origin];
-
-  frame = [advancedCancelButton_ frame];
-  frame.origin.x = NSMinX([advancedDoneButton_ frame]) - NSWidth(frame) -
-                   kTranslateBubbleRelatedControlHorizontalSpacing;
-  [advancedCancelButton_ setFrameOrigin:frame.origin];
-}
-
-- (void)updateAlwaysCheckboxes {
-  NSInteger state = shouldAlwaysTranslate_ ? NSOnState : NSOffState;
-  [beforeAlwaysTranslateCheckbox_ setState:state];
-  [advancedAlwaysTranslateCheckbox_ setState:state];
-}
-
-- (NSImageView*)addIcon:(NSView*)view {
-  NSRect imageFrame =
-      NSMakeRect(0, 0, kTranslateBubbleIconWidth, kTranslateBubbleIconHeight);
-  base::scoped_nsobject<NSImageView> image(
-      [[NSImageView alloc] initWithFrame:imageFrame]);
-  [image setImage:(ui::ResourceBundle::GetSharedInstance()
-                       .GetImageNamed(IDR_TRANSLATE_BUBBLE_ICON)
-                       .ToNSImage())];
-  [view addSubview:image];
-  return image.get();
-}
-
-- (NSTextView*)addStyledTextView:(NSString*)message
-                          toView:(NSView*)view
-                      withRanges:(std::vector<NSRange>)ranges
-                        delegate:(id<NSTextViewDelegate>)delegate {
-  NSRect frame =
-      NSMakeRect(kTranslateBubbleFramePadding + kTranslateBubbleIconWidth +
-                     kTranslateBubbleIconPadding,
-                 kTranslateBubbleFramePadding,
-                 kTranslateBubbleContentWidth - kTranslateBubbleIconWidth -
-                     kTranslateBubbleIconPadding,
-                 0);
-  base::scoped_nsobject<HyperlinkTextView> styledText(
-      [[HyperlinkTextView alloc] initWithFrame:frame]);
-  [styledText setMessage:message
-                withFont:[NSFont systemFontOfSize:[NSFont smallSystemFontSize]]
-            messageColor:(skia::SkColorToCalibratedNSColor(SK_ColorBLACK))];
-  [styledText setDelegate:delegate];
-
-  NSColor* linkColor =
-      skia::SkColorToCalibratedNSColor(chrome_style::GetLinkColor());
-  // Create the link with no underlining.
-  [styledText setLinkTextAttributes:nil];
-  NSTextStorage* storage = [styledText textStorage];
-  for (const auto& range : ranges) {
-    [styledText addLinkRange:range withURL:nil linkColor:linkColor];
-    [storage addAttribute:NSUnderlineStyleAttributeName
-                    value:@(NSUnderlineStyleNone)
-                    range:range];
-  }
-
-  [view addSubview:styledText];
-  [styledText setVerticallyResizable:YES];
-  [styledText sizeToFit];
-  return styledText.get();
-}
-
-- (NSTextField*)addText:(NSString*)text
-                 toView:(NSView*)view {
-  base::scoped_nsobject<NSTextField> textField(
-      [[NSTextField alloc] initWithFrame:NSZeroRect]);
-  [textField setEditable:NO];
-  [textField setSelectable:YES];
-  [textField setDrawsBackground:NO];
-  [textField setBezeled:NO];
-  [textField setStringValue:text];
-  NSFont* font = [NSFont systemFontOfSize:[NSFont smallSystemFontSize]];
-  [textField setFont:font];
-  [textField setAutoresizingMask:NSViewWidthSizable];
-  [view addSubview:textField.get()];
-
-  [textField sizeToFit];
-  return textField.get();
-}
-
-- (NSButton*)addLinkButtonWithText:(NSString*)text
-                            action:(SEL)action
-                            toView:(NSView*)view {
-  base::scoped_nsobject<NSButton> button(
-      [[HyperlinkButtonCell buttonWithString:text] retain]);
-
-  [button setButtonType:NSMomentaryPushInButton];
-  [button setBezelStyle:NSRegularSquareBezelStyle];
-  [button setFont:[NSFont systemFontOfSize:[NSFont smallSystemFontSize]]];
-  [button sizeToFit];
-  [button setTarget:self];
-  [button setAction:action];
-
-  [view addSubview:button.get()];
-
-  return button.get();
-}
-
-- (NSButton*)addButton:(NSString*)title
-                action:(SEL)action
-                toView:(NSView*)view {
-  base::scoped_nsobject<NSButton> button(
-      [[NSButton alloc] initWithFrame:NSZeroRect]);
-  [button setFont:[NSFont systemFontOfSize:[NSFont smallSystemFontSize]]];
-  [button setTitle:title];
-  [button setBezelStyle:NSRoundedBezelStyle];
-  [[button cell] setControlSize:NSSmallControlSize];
-  [button sizeToFit];
-  [button setTarget:self];
-  [button setAction:action];
-
-  [view addSubview:button.get()];
-
-  return button.get();
-}
-
-- (NSButton*)addCheckbox:(NSString*)title
-                  action:(SEL)action
-                  toView:(NSView*)view {
-  base::scoped_nsobject<NSButton> button(
-      [[NSButton alloc] initWithFrame:NSZeroRect]);
-  [button setFont:[NSFont systemFontOfSize:[NSFont smallSystemFontSize]]];
-  [button setTitle:title];
-  [[button cell] setControlSize:NSSmallControlSize];
-  [button setButtonType:NSSwitchButton];
-  [button sizeToFit];
-  [button setTarget:self];
-  [button setAction:action];
-
-  [view addSubview:button.get()];
-
-  return button.get();
-}
-
-- (NSButton*)addCloseButton:(NSView*)view action:(SEL)action {
-  const int extent = chrome_style::GetCloseButtonSize();
-  NSRect frame = NSMakeRect(0, 0, extent, extent);
-  base::scoped_nsobject<NSButton> button(
-      [[WebUIHoverCloseButton alloc] initWithFrame:frame]);
-  [button setTarget:self];
-  [button setAction:action];
-  [view addSubview:button.get()];
-  return button.get();
-}
-
-- (NSPopUpButton*)addPopUpButton:(ui::ComboboxModel*)model
-                          action:(SEL)action
-                          toView:(NSView*)view {
-  base::scoped_nsobject<NSPopUpButton> button(
-      [[BubbleCombobox alloc] initWithFrame:NSZeroRect
-                                  pullsDown:NO
-                                      model:model]);
-  [button setTarget:self];
-  [button setAction:action];
-  [button sizeToFit];
-  [view addSubview:button.get()];
-  return button.get();
-}
-
-- (IBAction)handleTranslateButtonPressed:(id)sender {
-  model_->SetAlwaysTranslate(shouldAlwaysTranslate_);
-  translate::ReportUiAction(translate::TRANSLATE_BUTTON_CLICKED);
-  translateExecuted_ = YES;
-  model_->Translate();
-}
-
-- (IBAction)handleAlwaysTranslateCheckboxPressed:(NSButton*)sender {
-  shouldAlwaysTranslate_ = [sender state] == NSOnState;
-  translate::ReportUiAction(shouldAlwaysTranslate_
-                                ? translate::ALWAYS_TRANSLATE_CHECKED
-                                : translate::ALWAYS_TRANSLATE_UNCHECKED);
-}
-
-- (IBAction)handleDoneButtonPressed:(id)sender {
-  translate::ReportUiAction(translate::DONE_BUTTON_CLICKED);
-  model_->SetAlwaysTranslate(shouldAlwaysTranslate_);
-  if (model_->IsPageTranslatedInCurrentLanguages()) {
-    model_->GoBackFromAdvanced();
-    [self performLayout];
-  } else {
-    translateExecuted_ = true;
-    model_->Translate();
-    [self switchView:TranslateBubbleModel::VIEW_STATE_TRANSLATING];
-  }
-}
-
-- (IBAction)handleCancelButtonPressed:(id)sender {
-  translate::ReportUiAction(translate::CANCEL_BUTTON_CLICKED);
-  model_->GoBackFromAdvanced();
-  [self performLayout];
-}
-
-- (IBAction)handleCloseButtonPressed:(id)sender {
-  model_->DeclineTranslation();
-  translate::ReportUiAction(translate::CLOSE_BUTTON_CLICKED);
-  [self close];
-}
-
-- (IBAction)handleShowOriginalButtonPressed:(id)sender {
-  translate::ReportUiAction(translate::SHOW_ORIGINAL_BUTTON_CLICKED);
-  model_->RevertTranslation();
-  [self close];
-}
-
-- (IBAction)handleTryAgainButtonPressed:(id)sender {
-  model_->Translate();
-  translate::ReportUiAction(translate::TRY_AGAIN_BUTTON_CLICKED);
-}
-
-- (IBAction)handleAdvancedLinkButtonPressed:(id)sender {
-  translate::ReportUiAction(translate::ADVANCED_LINK_CLICKED);
-  [self switchView:TranslateBubbleModel::VIEW_STATE_ADVANCED];
-}
-
-- (IBAction)handleLanguageSettingsLinkButtonPressed:(id)sender {
-  GURL url = chrome::GetSettingsUrl(chrome::kLanguageOptionsSubPage);
-  webContents_->OpenURL(content::OpenURLParams(
-      url, content::Referrer(), WindowOpenDisposition::NEW_FOREGROUND_TAB,
-      ui::PAGE_TRANSITION_LINK, false));
-  translate::ReportUiAction(translate::SETTINGS_LINK_CLICKED);
-  [self close];
-}
-
-- (IBAction)handleDenialPopUpButtonNeverTranslateLanguageSelected:(id)sender {
-  translate::ReportUiAction(translate::NEVER_TRANSLATE_LANGUAGE_MENU_CLICKED);
-  model_->DeclineTranslation();
-  model_->SetNeverTranslateLanguage(true);
-  [self close];
-}
-
-- (IBAction)handleDenialPopUpButtonNeverTranslateSiteSelected:(id)sender {
-  translate::ReportUiAction(translate::NEVER_TRANSLATE_SITE_MENU_CLICKED);
-  model_->DeclineTranslation();
-  model_->SetNeverTranslateSite(true);
-  [self close];
-}
-
-- (IBAction)handleSourceLanguagePopUpButtonSelectedItemChanged:(id)sender {
-  translate::ReportUiAction(translate::SOURCE_LANGUAGE_MENU_CLICKED);
-  NSPopUpButton* button = base::mac::ObjCCastStrict<NSPopUpButton>(sender);
-  model_->UpdateOriginalLanguageIndex([button indexOfSelectedItem]);
-  [self updateAdvancedView];
-}
-
-- (IBAction)handleTargetLanguagePopUpButtonSelectedItemChanged:(id)sender {
-  translate::ReportUiAction(translate::TARGET_LANGUAGE_MENU_CLICKED);
-  NSPopUpButton* button = base::mac::ObjCCastStrict<NSPopUpButton>(sender);
-  model_->UpdateTargetLanguageIndex([button indexOfSelectedItem]);
-  [self updateAdvancedView];
-}
-
-// The NSTextViewDelegate method which called when user click on the
-// source or target language on the before translate view.
-- (BOOL)textView:(NSTextView*)aTextView
-    clickedOnLink:(id)link
-          atIndex:(NSUInteger)charIndex {
-  translate::ReportUiAction(translate::ADVANCED_LINK_CLICKED);
-  [self switchView:TranslateBubbleModel::VIEW_STATE_ADVANCED];
-  return YES;
-}
-
-@end
diff --git a/chrome/browser/ui/cocoa/translate/translate_bubble_controller_unittest.mm b/chrome/browser/ui/cocoa/translate/translate_bubble_controller_unittest.mm
deleted file mode 100644
index e8b97e0b..0000000
--- a/chrome/browser/ui/cocoa/translate/translate_bubble_controller_unittest.mm
+++ /dev/null
@@ -1,203 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "chrome/browser/ui/cocoa/translate/translate_bubble_controller.h"
-
-#include "base/test/metrics/histogram_tester.h"
-#include "base/test/scoped_feature_list.h"
-#import "chrome/browser/ui/browser.h"
-#include "chrome/browser/ui/browser_window.h"
-#import "chrome/browser/ui/cocoa/browser_window_controller.h"
-#import "chrome/browser/ui/cocoa/info_bubble_window.h"
-#import "chrome/browser/ui/cocoa/test/cocoa_profile_test.h"
-#include "chrome/browser/ui/cocoa/test/run_loop_testing.h"
-#include "chrome/browser/ui/translate/translate_bubble_model.h"
-#include "chrome/common/url_constants.h"
-#include "content/public/browser/site_instance.h"
-#include "ui/base/ui_base_features.h"
-
-@implementation BrowserWindowController (ForTesting)
-
-- (TranslateBubbleController*)translateBubbleController{
-  return translateBubbleController_;
-}
-
-@end
-
-@implementation TranslateBubbleController (ForTesting)
-
-- (NSView*)errorView {
-  NSNumber* key = @(TranslateBubbleModel::VIEW_STATE_ERROR);
-  return [views_ objectForKey:key];
-}
-- (NSButton*)tryAgainButton {
-  return tryAgainButton_;
-}
-
-@end
-
-class TranslateBubbleControllerTest : public CocoaProfileTest {
- public:
-  TranslateBubbleControllerTest() {}
-
-  // CocoaProfileTest:
-  void SetUp() override {
-    // This file only tests Cocoa UI and can be deleted when kSecondaryUiMd is
-    // default.
-    scoped_feature_list_.InitAndDisableFeature(features::kSecondaryUiMd);
-
-    CocoaProfileTest::SetUp();
-
-    site_instance_ = content::SiteInstance::Create(profile());
-
-    NSWindow* nativeWindow = browser()->window()->GetNativeWindow();
-    bwc_ =
-        [BrowserWindowController browserWindowControllerForWindow:nativeWindow];
-    web_contents_ = AppendToTabStrip();
-  }
-
-  content::WebContents* AppendToTabStrip() {
-    std::unique_ptr<content::WebContents> web_contents =
-        content::WebContents::Create(content::WebContents::CreateParams(
-            profile(), site_instance_.get()));
-    content::WebContents* raw_web_contents = web_contents.get();
-    browser()->tab_strip_model()->AppendWebContents(std::move(web_contents),
-                                                    /*foreground=*/true);
-    return raw_web_contents;
-  }
-
-  BrowserWindowController* bwc() { return bwc_; }
-
-  TranslateBubbleController* bubble() {
-    return [bwc() translateBubbleController];
-  }
-
-  void ShowBubble() {
-    ASSERT_FALSE(bubble());
-    translate::TranslateStep step = translate::TRANSLATE_STEP_BEFORE_TRANSLATE;
-
-    [bwc_ showTranslateBubbleForWebContents:web_contents_
-                                       step:step
-                                  errorType:translate::TranslateErrors::NONE];
-
-    // Ensure that there are no closing animations.
-    InfoBubbleWindow* window = (InfoBubbleWindow*)[bubble() window];
-    [window setAllowedAnimations:info_bubble::kAnimateNone];
-  }
-
-  void SwitchToErrorView() {
-    translate::TranslateStep step = translate::TRANSLATE_STEP_TRANSLATE_ERROR;
-    [bwc_
-        showTranslateBubbleForWebContents:web_contents_
-                                     step:step
-                                errorType:translate::TranslateErrors::NETWORK];
-  }
-
-  void CloseBubble() {
-    [bubble() close];
-    chrome::testing::NSRunLoopRunAllPending();
-  }
-
- private:
-  base::test::ScopedFeatureList scoped_feature_list_;
-  scoped_refptr<content::SiteInstance> site_instance_;
-  BrowserWindowController* bwc_;
-  content::WebContents* web_contents_;
-
-  DISALLOW_COPY_AND_ASSIGN(TranslateBubbleControllerTest);
-};
-
-TEST_F(TranslateBubbleControllerTest, ShowAndClose) {
-  EXPECT_FALSE(bubble());
-
-  ShowBubble();
-  EXPECT_TRUE(bubble());
-
-  CloseBubble();
-  EXPECT_FALSE(bubble());
-}
-
-TEST_F(TranslateBubbleControllerTest, SwitchToErrorView) {
-  EXPECT_FALSE(bubble());
-  ShowBubble();
-  const TranslateBubbleModel* model = [bubble() model];
-
-  EXPECT_TRUE(bubble());
-  EXPECT_EQ(TranslateBubbleModel::ViewState::VIEW_STATE_BEFORE_TRANSLATE,
-            model->GetViewState());
-
-  SwitchToErrorView();
-
-  NSView* errorView = [bubble() errorView];
-  // We should have 4 subview inside the error view:
-  // A NSTextField, a NSImageView and two NSButton.
-  EXPECT_EQ(4UL, [[errorView subviews] count]);
-
-  // one of the subview should be "Try again" button.
-  EXPECT_TRUE([[errorView subviews] containsObject:[bubble() tryAgainButton]]);
-  EXPECT_EQ(TranslateBubbleModel::ViewState::VIEW_STATE_ERROR,
-            model->GetViewState());
-  EXPECT_TRUE(bubble());
-  CloseBubble();
-}
-
-TEST_F(TranslateBubbleControllerTest, SwitchViews) {
-  // A basic test which just switch between views to make sure no crash.
-  EXPECT_FALSE(bubble());
-
-  ShowBubble();
-  EXPECT_TRUE(bubble());
-
-  // Switch to during translating view.
-  [bubble()
-      switchView:(TranslateBubbleModel::ViewState::VIEW_STATE_TRANSLATING)];
-
-  EXPECT_TRUE(bubble());
-
-  // Switch to after translating view.
-  [bubble()
-      switchView:(TranslateBubbleModel::ViewState::VIEW_STATE_AFTER_TRANSLATE)];
-
-  EXPECT_TRUE(bubble());
-
-  // Switch to advanced view.
-  [bubble() switchView:(TranslateBubbleModel::ViewState::VIEW_STATE_ADVANCED)];
-
-  EXPECT_TRUE(bubble());
-
-  // Switch to before translating view.
-  [bubble() switchView:
-                (TranslateBubbleModel::ViewState::VIEW_STATE_BEFORE_TRANSLATE)];
-
-  EXPECT_TRUE(bubble());
-
-  CloseBubble();
-  EXPECT_FALSE(bubble());
-}
-
-TEST_F(TranslateBubbleControllerTest, CloseRegistersDecline) {
-  const char kDeclineTranslateDismissUI[] =
-      "Translate.DeclineTranslateDismissUI";
-  const char kDeclineTranslate[] = "Translate.DeclineTranslate";
-
-  // A simple close without any interactions registers as a dismissal.
-  {
-    base::HistogramTester histogram_tester;
-    ShowBubble();
-    CloseBubble();
-    histogram_tester.ExpectTotalCount(kDeclineTranslateDismissUI, 1);
-    histogram_tester.ExpectTotalCount(kDeclineTranslate, 0);
-  }
-
-  // A close while pressing e.g. 'x', registers as decline.
-  {
-    base::HistogramTester histogram_tester;
-    ShowBubble();
-    [bubble() handleCloseButtonPressed:nil];
-
-    CloseBubble();
-    histogram_tester.ExpectTotalCount(kDeclineTranslateDismissUI, 0);
-    histogram_tester.ExpectTotalCount(kDeclineTranslate, 1);
-  }
-}
diff --git a/chrome/browser/ui/cocoa/translate/translate_bubble_test_utils_views_cocoa.mm b/chrome/browser/ui/cocoa/translate/translate_bubble_test_utils_views_cocoa.mm
deleted file mode 100644
index 0574cd8..0000000
--- a/chrome/browser/ui/cocoa/translate/translate_bubble_test_utils_views_cocoa.mm
+++ /dev/null
@@ -1,67 +0,0 @@
-// Copyright 2015 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "chrome/browser/ui/translate/translate_bubble_test_utils.h"
-
-#include "chrome/browser/ui/browser.h"
-#include "chrome/browser/ui/browser_window.h"
-#include "chrome/browser/ui/cocoa/browser_dialogs_views_mac.h"
-#include "chrome/browser/ui/cocoa/browser_window_controller.h"
-#include "chrome/browser/ui/cocoa/translate/translate_bubble_controller.h"
-#include "chrome/browser/ui/translate/translate_bubble_model.h"
-#include "chrome/browser/ui/views/translate/translate_bubble_view.h"
-#include "ui/events/keycodes/dom/dom_code.h"
-#include "ui/views/controls/button/label_button.h"
-
-// TODO(groby): Share with translate_bubble_controller_unittest.mm
-@implementation BrowserWindowController (ForTesting)
-
-- (TranslateBubbleController*)translateBubbleController {
-  return translateBubbleController_;
-}
-
-@end
-
-namespace translate {
-
-namespace test_utils {
-
-const TranslateBubbleModel* GetCurrentModel(Browser* browser) {
-  DCHECK(browser);
-  if (chrome::ShowAllDialogsWithViewsToolkit()) {
-    TranslateBubbleView* view = TranslateBubbleView::GetCurrentBubble();
-    return view ? view->model() : nullptr;
-  }
-
-  NSWindow* native_window = browser->window()->GetNativeWindow();
-  BrowserWindowController* controller =
-      [BrowserWindowController browserWindowControllerForWindow:native_window];
-  return [[controller translateBubbleController] model];
-}
-
-void PressTranslate(Browser* browser) {
-  DCHECK(browser);
-
-  if (chrome::ShowAllDialogsWithViewsToolkit()) {
-    TranslateBubbleView* bubble = TranslateBubbleView::GetCurrentBubble();
-    DCHECK(bubble);
-
-    views::LabelButton button(nullptr, base::string16());
-    button.set_id(TranslateBubbleView::BUTTON_ID_TRANSLATE);
-
-    bubble->ButtonPressed(&button,
-                          ui::KeyEvent(ui::ET_KEY_PRESSED, ui::VKEY_RETURN,
-                                       ui::DomCode::ENTER, ui::EF_NONE));
-    return;
-  }
-
-  NSWindow* native_window = browser->window()->GetNativeWindow();
-  BrowserWindowController* controller =
-      [BrowserWindowController browserWindowControllerForWindow:native_window];
-  [[controller translateBubbleController] handleTranslateButtonPressed:nil];
-}
-
-}  // namespace test_utils
-
-}  // namespace translate
diff --git a/chrome/browser/ui/desktop_ios_promotion/desktop_ios_promotion_util.cc b/chrome/browser/ui/desktop_ios_promotion/desktop_ios_promotion_util.cc
index 5e41940f..63e7925 100644
--- a/chrome/browser/ui/desktop_ios_promotion/desktop_ios_promotion_util.cc
+++ b/chrome/browser/ui/desktop_ios_promotion/desktop_ios_promotion_util.cc
@@ -14,12 +14,10 @@
 #include "chrome/browser/browser_process.h"
 #include "chrome/browser/profiles/profile.h"
 #include "chrome/browser/profiles/profiles_state.h"
-#include "chrome/browser/sync/profile_sync_service_factory.h"
 #include "chrome/common/chrome_features.h"
 #include "chrome/common/chrome_switches.h"
 #include "chrome/grit/chromium_strings.h"
 #include "chrome/grit/generated_resources.h"
-#include "components/browser_sync/profile_sync_service.h"
 #include "components/pref_registry/pref_registry_syncable.h"
 #include "components/prefs/pref_registry_simple.h"
 #include "components/prefs/pref_service.h"
@@ -93,16 +91,15 @@
   if (signin_error_controller && signin_error_controller->HasError())
     return false;
 
-  const browser_sync::ProfileSyncService* sync_service =
-      ProfileSyncServiceFactory::GetForProfile(profile);
   // Promotion should only show for english locale.
   PrefService* local_state = g_browser_process->local_state();
   std::string locale = base::i18n::GetConfiguredLocale();
   if (locale != "en-US" && locale != "en-CA")
     return false;
   if (!base::FeatureList::IsEnabled(features::kDesktopIOSPromotion) ||
-      !sync_service || !sync_service->IsSyncAllowed())
+      !profile->IsSyncAllowed()) {
     return false;
+  }
 
   // Check if the specific entrypoint is enabled by Finch.
   std::string targeted_entry_point = base::GetFieldTrialParamValueByFeature(
diff --git a/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.cc b/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.cc
index 73f9d37..5d731de 100644
--- a/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.cc
+++ b/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.cc
@@ -11,6 +11,7 @@
 #include "chrome/browser/ui/views/chrome_constrained_window_views_client.h"
 #include "chrome/browser/ui/views/chrome_views_delegate.h"
 #include "chrome/browser/ui/views/harmony/chrome_layout_provider.h"
+#include "chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.h"
 #include "components/constrained_window/constrained_window_views.h"
 #include "services/service_manager/sandbox/switches.h"
 
@@ -53,8 +54,6 @@
 #include "ash/public/interfaces/constants.mojom.h"
 #include "content/public/common/content_switches.h"
 #include "ui/base/ui_base_features.h"
-#else  // defined(OS_CHROMEOS)
-#include "chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.h"
 #endif  // defined(OS_CHROMEOS)
 
 ChromeBrowserMainExtraPartsViews::ChromeBrowserMainExtraPartsViews() {
@@ -173,18 +172,14 @@
 }
 
 void ChromeBrowserMainExtraPartsViews::PostBrowserStart() {
-#if !defined(OS_CHROMEOS)
   relaunch_notification_controller_ =
       std::make_unique<RelaunchNotificationController>(
           UpgradeDetector::GetInstance());
-#endif
 }
 
 void ChromeBrowserMainExtraPartsViews::PostMainMessageLoopRun() {
-#if !defined(OS_CHROMEOS)
   // The relaunch notification controller acts on timer-based events. Tear it
   // down explicitly here to avoid a case where such an event arrives during
   // shutdown.
   relaunch_notification_controller_.reset();
-#endif
 }
diff --git a/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.h b/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.h
index 434a1f0..697ac4d 100644
--- a/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.h
+++ b/chrome/browser/ui/views/chrome_browser_main_extra_parts_views.h
@@ -27,9 +27,7 @@
 }
 #endif
 
-#if !defined(OS_CHROMEOS)
 class RelaunchNotificationController;
-#endif
 
 class ChromeBrowserMainExtraPartsViews : public ChromeBrowserMainExtraParts {
  public:
@@ -60,11 +58,9 @@
   std::unique_ptr<views::MusClient> mus_client_;
 #endif
 
-#if !defined(OS_CHROMEOS)
   // Manages the relaunch notification prompts.
   std::unique_ptr<RelaunchNotificationController>
       relaunch_notification_controller_;
-#endif
 
   DISALLOW_COPY_AND_ASSIGN(ChromeBrowserMainExtraPartsViews);
 };
diff --git a/chrome/browser/ui/views/frame/browser_non_client_frame_view_mac.mm b/chrome/browser/ui/views/frame/browser_non_client_frame_view_mac.mm
index bc1ff473..69c5df8c 100644
--- a/chrome/browser/ui/views/frame/browser_non_client_frame_view_mac.mm
+++ b/chrome/browser/ui/views/frame/browser_non_client_frame_view_mac.mm
@@ -73,6 +73,18 @@
   // calling through private APIs.
   DCHECK(tabstrip);
 
+  const bool restored = !frame()->IsMaximized() && !frame()->IsFullscreen();
+  gfx::Rect bounds = gfx::Rect(0, GetTopInset(restored), width(),
+                               tabstrip->GetPreferredSize().height());
+  bounds.Inset(GetTabStripLeftInset(), 0,
+               GetAfterTabstripItemWidth() + GetTabstripPadding(), 0);
+  return bounds;
+}
+
+int BrowserNonClientFrameViewMac::GetTopInset(bool restored) const {
+  if (!browser_view()->IsTabStripVisible())
+    return 0;
+
   // Calculate the y offset for the tab strip because in fullscreen mode the tab
   // strip may need to move under the slide down menu bar.
   CGFloat y_offset = kTabstripTopInset;
@@ -99,15 +111,7 @@
     }
   }
 
-  gfx::Rect bounds =
-      gfx::Rect(0, y_offset, width(), tabstrip->GetPreferredSize().height());
-  bounds.Inset(GetTabStripLeftInset(), 0,
-               GetAfterTabstripItemWidth() + GetTabstripPadding(), 0);
-  return bounds;
-}
-
-int BrowserNonClientFrameViewMac::GetTopInset(bool restored) const {
-  return browser_view()->IsTabStripVisible() ? kTabstripTopInset : 0;
+  return y_offset;
 }
 
 int BrowserNonClientFrameViewMac::GetAfterTabstripItemWidth() const {
diff --git a/chrome/browser/ui/views/frame/browser_view_layout.cc b/chrome/browser/ui/views/frame/browser_view_layout.cc
index 0e5d086..19f3694 100644
--- a/chrome/browser/ui/views/frame/browser_view_layout.cc
+++ b/chrome/browser/ui/views/frame/browser_view_layout.cc
@@ -323,15 +323,7 @@
   vertical_layout_rect_ = browser_view->GetLocalBounds();
   int top_inset = delegate_->GetTopInsetInBrowserView(false);
   int top = LayoutTabStripRegion(top_inset);
-  int possible_fullscreen_tab_strip_offset = 0;
   if (delegate_->IsTabStripVisible()) {
-#if defined(OS_MACOSX)
-    // Tab strip should always start from the top inset unless it is under the
-    // system menu bar during full screen mode on Mac. In this case, the tab
-    // strip and other top UI will move under the menu bar while the content
-    // should stay put.
-    possible_fullscreen_tab_strip_offset = tab_strip_->bounds().y() - top_inset;
-#endif
     // By passing true to GetTopInsetInBrowserView(), we position the tab
     // background to vertically align with the frame background image of a
     // restored-mode frame, even in a maximized window.  Then in the frame code,
@@ -354,7 +346,6 @@
   UpdateTopContainerBounds();
 
   int bottom = LayoutDownloadShelf(browser_view->height());
-  top -= possible_fullscreen_tab_strip_offset;
   // Treat a detached bookmark bar as if the web contents container is shifted
   // upwards and overlaps it.
   int active_top_margin = GetContentsOffsetForBookmarkBar();
diff --git a/chrome/browser/ui/views/location_bar/icon_label_bubble_view.cc b/chrome/browser/ui/views/location_bar/icon_label_bubble_view.cc
index 914e1fa1..3240412 100644
--- a/chrome/browser/ui/views/location_bar/icon_label_bubble_view.cc
+++ b/chrome/browser/ui/views/location_bar/icon_label_bubble_view.cc
@@ -50,6 +50,9 @@
 IconLabelBubbleView::SeparatorView::SeparatorView(IconLabelBubbleView* owner) {
   DCHECK(owner);
   owner_ = owner;
+
+  SetPaintToLayer();
+  layer()->SetFillsBoundsOpaquely(false);
 }
 
 void IconLabelBubbleView::SeparatorView::OnPaint(gfx::Canvas* canvas) {
@@ -63,15 +66,18 @@
                    gfx::PointF(x, GetLocalBounds().bottom()), separator_color);
 }
 
-void IconLabelBubbleView::SeparatorView::OnImplicitAnimationsCompleted() {
-  if (layer() && layer()->opacity() == 1.0f)
-    DestroyLayer();
-}
-
 void IconLabelBubbleView::SeparatorView::UpdateOpacity() {
   if (!visible())
     return;
 
+  // When using focus rings are visible we should hide the separator instantly
+  // when the IconLabelBubbleView is focused. Otherwise we should follow the
+  // inkdrop.
+  if (views::PlatformStyle::kPreferFocusRings && owner_->HasFocus()) {
+    layer()->SetOpacity(0.0f);
+    return;
+  }
+
   views::InkDrop* ink_drop = owner_->GetInkDrop();
   DCHECK(ink_drop);
 
@@ -88,10 +94,6 @@
     duration = kIconLabelBubbleFadeInDurationMs;
   }
 
-  if (!layer())
-    SetPaintToLayer();
-  layer()->SetFillsBoundsOpaquely(false);
-
   if (disable_animation_for_test_) {
     layer()->SetOpacity(opacity);
   } else {
@@ -99,7 +101,6 @@
     animation.SetTransitionDuration(
         base::TimeDelta::FromMilliseconds(duration));
     animation.SetTweenType(gfx::Tween::Type::EASE_IN);
-    animation.AddObserver(this);
     layer()->SetOpacity(opacity);
   }
 }
@@ -377,6 +378,16 @@
   OnActivate(event);
 }
 
+void IconLabelBubbleView::OnFocus() {
+  separator_view_->UpdateOpacity();
+  Button::OnFocus();
+}
+
+void IconLabelBubbleView::OnBlur() {
+  separator_view_->UpdateOpacity();
+  Button::OnBlur();
+}
+
 void IconLabelBubbleView::OnWidgetDestroying(views::Widget* widget) {
   widget->RemoveObserver(this);
 }
diff --git a/chrome/browser/ui/views/location_bar/icon_label_bubble_view.h b/chrome/browser/ui/views/location_bar/icon_label_bubble_view.h
index 4d3254d..e45806a 100644
--- a/chrome/browser/ui/views/location_bar/icon_label_bubble_view.h
+++ b/chrome/browser/ui/views/location_bar/icon_label_bubble_view.h
@@ -9,7 +9,6 @@
 
 #include "base/macros.h"
 #include "base/strings/string16.h"
-#include "ui/compositor/layer_animation_observer.h"
 #include "ui/gfx/geometry/insets.h"
 #include "ui/gfx/geometry/size.h"
 #include "ui/views/animation/ink_drop_host_view.h"
@@ -38,17 +37,13 @@
   static constexpr int kTrailingPaddingPreMd = 2;
 
   // A view that draws the separator.
-  class SeparatorView : public views::View,
-                        public ui::ImplicitAnimationObserver {
+  class SeparatorView : public views::View {
    public:
     explicit SeparatorView(IconLabelBubbleView* owner);
 
     // views::View:
     void OnPaint(gfx::Canvas* canvas) override;
 
-    // ui::ImplicitAnimationObserver:
-    void OnImplicitAnimationsCompleted() override;
-
     // Updates the opacity based on the ink drop's state.
     void UpdateOpacity();
 
@@ -154,6 +149,8 @@
   bool IsTriggerableEvent(const ui::Event& event) override;
   bool ShouldUpdateInkDropOnClickCanceled() const override;
   void NotifyClick(const ui::Event& event) override;
+  void OnFocus() override;
+  void OnBlur() override;
 
   // views::WidgetObserver:
   void OnWidgetDestroying(views::Widget* widget) override;
diff --git a/chrome/browser/ui/views/omnibox/omnibox_result_view.cc b/chrome/browser/ui/views/omnibox/omnibox_result_view.cc
index 677eb81..60b6e2f 100644
--- a/chrome/browser/ui/views/omnibox/omnibox_result_view.cc
+++ b/chrome/browser/ui/views/omnibox/omnibox_result_view.cc
@@ -178,12 +178,14 @@
                                          match_.contents_class);
     suggestion_view_->description()->SetText(match_.description,
                                              match_.description_class);
-    // Explicitly re-apply default styling - high contrast modes use different
-    // text colors depending on selection state.
-    suggestion_view_->content()->ApplyTextColor(
-        OmniboxPart::RESULTS_TEXT_DEFAULT);
-    suggestion_view_->description()->ApplyTextColor(
-        OmniboxPart::RESULTS_TEXT_DEFAULT);
+
+    // Normally, OmniboxTextView caches its appearance, but in high contrast
+    // selected-ness changes the text colors, so the styling of the text part of
+    // the results needs to be recomputed.
+    if (high_contrast) {
+      suggestion_view_->content()->ReapplyStyling();
+      suggestion_view_->description()->ReapplyStyling();
+    }
   }
 
   AutocompleteMatch* keyword_match = match_.associated_keyword.get();
diff --git a/chrome/browser/ui/views/omnibox/omnibox_text_view.cc b/chrome/browser/ui/views/omnibox/omnibox_text_view.cc
index 4d0cb05..5d87db9 100644
--- a/chrome/browser/ui/views/omnibox/omnibox_text_view.cc
+++ b/chrome/browser/ui/views/omnibox/omnibox_text_view.cc
@@ -224,36 +224,7 @@
       std::make_unique<ACMatchClassifications>(classifications);
   render_text_ = CreateRenderText(text);
 
-  const size_t text_length = render_text_->text().length();
-  for (size_t i = 0; i < classifications.size(); ++i) {
-    const size_t text_start = classifications[i].offset;
-    if (text_start >= text_length)
-      break;
-
-    const size_t text_end =
-        (i < (classifications.size() - 1))
-            ? std::min(classifications[i + 1].offset, text_length)
-            : text_length;
-    const gfx::Range current_range(text_start, text_end);
-
-    // Calculate style-related data.
-    if (classifications[i].style & ACMatchClassification::MATCH)
-      render_text_->ApplyWeight(gfx::Font::Weight::BOLD, current_range);
-
-    OmniboxPart part = OmniboxPart::RESULTS_TEXT_DEFAULT;
-    if (classifications[i].style & ACMatchClassification::URL) {
-      part = OmniboxPart::RESULTS_TEXT_URL;
-      render_text_->SetDirectionalityMode(gfx::DIRECTIONALITY_AS_URL);
-    } else if (classifications[i].style & ACMatchClassification::DIM) {
-      part = OmniboxPart::RESULTS_TEXT_DIMMED;
-    } else if (classifications[i].style & ACMatchClassification::INVISIBLE) {
-      part = OmniboxPart::RESULTS_TEXT_INVISIBLE;
-    }
-    render_text_->ApplyColor(result_view_->GetColor(part), current_range);
-  }
-
-  UpdateLineHeight();
-  SetPreferredSize(CalculatePreferredSize());
+  ReapplyStyling();
 }
 
 void OmniboxTextView::SetText(const SuggestionAnswer::ImageLine& line,
@@ -309,6 +280,40 @@
   return font_height_;
 }
 
+void OmniboxTextView::ReapplyStyling() {
+  const ACMatchClassifications& classifications = *cached_classifications_;
+  const size_t text_length = render_text_->text().length();
+  for (size_t i = 0; i < classifications.size(); ++i) {
+    const size_t text_start = classifications[i].offset;
+    if (text_start >= text_length)
+      break;
+
+    const size_t text_end =
+        (i < (classifications.size() - 1))
+            ? std::min(classifications[i + 1].offset, text_length)
+            : text_length;
+    const gfx::Range current_range(text_start, text_end);
+
+    // Calculate style-related data.
+    if (classifications[i].style & ACMatchClassification::MATCH)
+      render_text_->ApplyWeight(gfx::Font::Weight::BOLD, current_range);
+
+    OmniboxPart part = OmniboxPart::RESULTS_TEXT_DEFAULT;
+    if (classifications[i].style & ACMatchClassification::URL) {
+      part = OmniboxPart::RESULTS_TEXT_URL;
+      render_text_->SetDirectionalityMode(gfx::DIRECTIONALITY_AS_URL);
+    } else if (classifications[i].style & ACMatchClassification::DIM) {
+      part = OmniboxPart::RESULTS_TEXT_DIMMED;
+    } else if (classifications[i].style & ACMatchClassification::INVISIBLE) {
+      part = OmniboxPart::RESULTS_TEXT_INVISIBLE;
+    }
+    render_text_->ApplyColor(result_view_->GetColor(part), current_range);
+  }
+
+  UpdateLineHeight();
+  SetPreferredSize(CalculatePreferredSize());
+}
+
 std::unique_ptr<gfx::RenderText> OmniboxTextView::CreateRenderText(
     const base::string16& text) const {
   auto render_text = gfx::RenderText::CreateHarfBuzzInstance();
diff --git a/chrome/browser/ui/views/omnibox/omnibox_text_view.h b/chrome/browser/ui/views/omnibox/omnibox_text_view.h
index 4a5bc29..01ccc0b 100644
--- a/chrome/browser/ui/views/omnibox/omnibox_text_view.h
+++ b/chrome/browser/ui/views/omnibox/omnibox_text_view.h
@@ -63,6 +63,10 @@
   // multiple lines.
   int GetLineHeight() const;
 
+  // Reapplies text styling to the results text, based on the types of the match
+  // parts.
+  void ReapplyStyling();
+
  private:
   std::unique_ptr<gfx::RenderText> CreateRenderText(
       const base::string16& text) const;
diff --git a/chrome/browser/ui/views/overlay/overlay_window_views.cc b/chrome/browser/ui/views/overlay/overlay_window_views.cc
index 546f9af..d5845ef 100644
--- a/chrome/browser/ui/views/overlay/overlay_window_views.cc
+++ b/chrome/browser/ui/views/overlay/overlay_window_views.cc
@@ -604,6 +604,10 @@
   views::Widget::OnNativeWidgetSizeChanged(new_size);
 }
 
+void OverlayWindowViews::OnNativeWidgetDestroyed() {
+  controller_->OnWindowDestroyed();
+}
+
 void OverlayWindowViews::TogglePlayPause() {
   // Retrieve expected active state based on what command was sent in
   // TogglePlayPause() since the IPC message may not have been propogated
diff --git a/chrome/browser/ui/views/overlay/overlay_window_views.h b/chrome/browser/ui/views/overlay/overlay_window_views.h
index ca28d91..da9e1f3 100644
--- a/chrome/browser/ui/views/overlay/overlay_window_views.h
+++ b/chrome/browser/ui/views/overlay/overlay_window_views.h
@@ -51,6 +51,7 @@
   void OnNativeBlur() override;
   void OnNativeWidgetMove() override;
   void OnNativeWidgetSizeChanged(const gfx::Size& new_size) override;
+  void OnNativeWidgetDestroyed() override;
 
   // Gets the bounds of the controls.
   gfx::Rect GetCloseControlsBounds();
diff --git a/chrome/browser/ui/views/payments/order_summary_view_controller.cc b/chrome/browser/ui/views/payments/order_summary_view_controller.cc
index f718d1d6d..0f421b4 100644
--- a/chrome/browser/ui/views/payments/order_summary_view_controller.cc
+++ b/chrome/browser/ui/views/payments/order_summary_view_controller.cc
@@ -151,6 +151,10 @@
   return button;
 }
 
+bool OrderSummaryViewController::ShouldShowSecondaryButton() {
+  return false;
+}
+
 base::string16 OrderSummaryViewController::GetSheetTitle() {
   return l10n_util::GetStringUTF16(IDS_PAYMENTS_ORDER_SUMMARY_LABEL);
 }
diff --git a/chrome/browser/ui/views/payments/order_summary_view_controller.h b/chrome/browser/ui/views/payments/order_summary_view_controller.h
index 9f25d33..e085fa3 100644
--- a/chrome/browser/ui/views/payments/order_summary_view_controller.h
+++ b/chrome/browser/ui/views/payments/order_summary_view_controller.h
@@ -40,6 +40,7 @@
  private:
   // PaymentRequestSheetController:
   std::unique_ptr<views::Button> CreatePrimaryButton() override;
+  bool ShouldShowSecondaryButton() override;
   base::string16 GetSheetTitle() override;
   void FillContentView(views::View* content_view) override;
   void UpdatePayButtonState(bool enabled);
diff --git a/chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.cc b/chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.cc
index 3696241e..d52ea1ed 100644
--- a/chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.cc
+++ b/chrome/browser/ui/views/relaunch_notification/relaunch_notification_controller.cc
@@ -360,7 +360,7 @@
 
   widget_ = RelaunchRecommendedBubbleView::ShowBubble(
       browser, upgrade_detector_->upgrade_detected_time(),
-      base::BindRepeating(&chrome::AttemptRestart));
+      base::BindRepeating(&chrome::AttemptRelaunch));
 
   // Monitor the widget so that |widget_| can be cleared on close.
   widget_->AddObserver(this);
@@ -384,7 +384,7 @@
 
   widget_ = RelaunchRequiredDialogView::Show(
       browser, timer_.desired_run_time(),
-      base::BindRepeating(&chrome::AttemptRestart));
+      base::BindRepeating(&chrome::AttemptRelaunch));
 
   // Monitor the widget so that |widget_| can be cleared on close.
   widget_->AddObserver(this);
@@ -396,5 +396,5 @@
 }
 
 void RelaunchNotificationController::OnRelaunchDeadlineExpired() {
-  chrome::AttemptRestart();
+  chrome::AttemptRelaunch();
 }
diff --git a/chrome/browser/ui/views/tabs/new_tab_button.cc b/chrome/browser/ui/views/tabs/new_tab_button.cc
index ceefd28..401ae5a 100644
--- a/chrome/browser/ui/views/tabs/new_tab_button.cc
+++ b/chrome/browser/ui/views/tabs/new_tab_button.cc
@@ -465,13 +465,13 @@
       // The new tab background is mirrored in RTL mode, but the theme
       // background should never be mirrored. Mirror it here to compensate.
       float x_scale = 1.0f;
-      int x = GetMirroredX() + background_offset_.x();
-      const gfx::Size size = GetContentsBounds().size();
+      const gfx::Rect& contents_bounds = GetContentsBounds();
+      int x = GetMirroredX() + contents_bounds.x() + background_offset_.x();
       if (base::i18n::IsRTL()) {
         x_scale = -1.0f;
         // Offset by |width| such that the same region is painted as if there
         // was no flip.
-        x += size.width();
+        x += contents_bounds.width();
       }
 
       const bool succeeded = canvas->InitPaintFlagsForTiling(
diff --git a/chrome/browser/ui/webui/ukm/ukm_internals_ui.cc b/chrome/browser/ui/webui/ukm/ukm_internals_ui.cc
index 1f25b7d..0b69b7b 100644
--- a/chrome/browser/ui/webui/ukm/ukm_internals_ui.cc
+++ b/chrome/browser/ui/webui/ukm/ukm_internals_ui.cc
@@ -85,6 +85,8 @@
 
 }  // namespace
 
+// Changes to this class should be in sync with its iOS equivalent
+// ios/chrome/browser/ui/webui/ukm_internals_ui.cc
 UkmInternalsUI::UkmInternalsUI(content::WebUI* web_ui)
     : content::WebUIController(web_ui) {
   ukm::UkmService* ukm_service =
diff --git a/chrome/browser/vr/service/browser_xr_device.cc b/chrome/browser/vr/service/browser_xr_device.cc
index 6fd9c9c..e32daa766 100644
--- a/chrome/browser/vr/service/browser_xr_device.cc
+++ b/chrome/browser/vr/service/browser_xr_device.cc
@@ -113,21 +113,19 @@
     base::WeakPtr<VRDisplayHost> display,
     device::mojom::XRDeviceRuntimeSessionOptionsPtr options,
     device::mojom::VRDisplayHost::RequestSessionCallback callback,
-    device::mojom::XRPresentationConnectionPtr connection,
+    device::mojom::XRSessionPtr session,
     device::mojom::XRSessionControllerPtr immersive_session_controller) {
-  if (connection && display) {
+  if (session && display) {
     if (options->immersive) {
       presenting_display_host_ = display.get();
       immersive_session_controller_ = std::move(immersive_session_controller);
     }
 
-    device::mojom::XRSessionPtr xr_session = device::mojom::XRSession::New();
-    xr_session->connection = std::move(connection);
-    std::move(callback).Run(std::move(xr_session));
+    std::move(callback).Run(std::move(session));
   } else {
     std::move(callback).Run(nullptr);
-    if (connection) {
-      // The device has been removed, but we still got a connection, so make
+    if (session) {
+      // The device has been removed, but we still got a session, so make
       // sure to clean up this weird state.
       immersive_session_controller_ = std::move(immersive_session_controller);
       StopImmersiveSession();
diff --git a/chrome/browser/vr/service/browser_xr_device.h b/chrome/browser/vr/service/browser_xr_device.h
index de78a5a..242ee26 100644
--- a/chrome/browser/vr/service/browser_xr_device.h
+++ b/chrome/browser/vr/service/browser_xr_device.h
@@ -57,7 +57,7 @@
       base::WeakPtr<VRDisplayHost> display,
       device::mojom::XRDeviceRuntimeSessionOptionsPtr options,
       device::mojom::VRDisplayHost::RequestSessionCallback callback,
-      device::mojom::XRPresentationConnectionPtr connection,
+      device::mojom::XRSessionPtr session,
       device::mojom::XRSessionControllerPtr immersive_session_controller);
 
   device::mojom::XRRuntimePtr device_;
diff --git a/chrome/browser/vr/service/vr_display_host.cc b/chrome/browser/vr/service/vr_display_host.cc
index b22645ba7..f2937a3 100644
--- a/chrome/browser/vr/service/vr_display_host.cc
+++ b/chrome/browser/vr/service/vr_display_host.cc
@@ -124,25 +124,11 @@
                                      std::move(display_info));
 }
 
-void VRDisplayHost::OnARSessionCreated(
-    vr::BrowserXrDevice* device,
-    device::mojom::VRDisplayHost::RequestSessionCallback callback,
-    device::mojom::XRSessionPtr session) {
-  if (!session) {
-    std::move(callback).Run(nullptr);
-    return;
-  }
-
-  device->GetRuntime()->RequestMagicWindowSession(
-      base::BindOnce(&VRDisplayHost::OnMagicWindowSessionCreated,
-                     weak_ptr_factory_.GetWeakPtr(), std::move(callback)));
-}
-
 void VRDisplayHost::OnMagicWindowSessionCreated(
     device::mojom::VRDisplayHost::RequestSessionCallback callback,
-    device::mojom::VRMagicWindowProviderPtr magic_window_provider,
+    device::mojom::XRSessionPtr session,
     device::mojom::XRSessionControllerPtr controller) {
-  if (!magic_window_provider) {
+  if (!session) {
     std::move(callback).Run(nullptr);
     return;
   }
@@ -152,10 +138,7 @@
 
   magic_window_controllers_.AddPtr(std::move(controller));
 
-  device::mojom::XRSessionPtr xr_session = device::mojom::XRSession::New();
-  xr_session->magic_window_provider = magic_window_provider.PassInterface();
-
-  std::move(callback).Run(std::move(xr_session));
+  std::move(callback).Run(std::move(session));
 }
 
 VRDisplayHost::~VRDisplayHost() {
@@ -214,28 +197,16 @@
       ReportRequestPresent();
     }
 
-    // Ensure we will notify the immersive device when we are destroyed, so
-    // lingering presentation state will be cleaned up.
-    DCHECK(device == immersive_device_);
-
     device->RequestSession(this, std::move(runtime_options),
                            std::move(callback));
-  } else if (runtime_options->provide_passthrough_camera) {
-    // WebXrHitTest enabled means we are requesting an AR session.  This means
-    // we make two requests to the device - one to request permissions and start
-    // the runtime, then a followup to actually get the magic window provider.
-    // TODO(offenwanger): Clean this up and make only one request.
-    // base::Unretained(device) is safe because either device is alive when we
-    // are called back, or we are called back with failure.
-    device->RequestSession(
-        this, std::move(runtime_options),
-        base::BindOnce(&VRDisplayHost::OnARSessionCreated,
-                       weak_ptr_factory_.GetWeakPtr(), base::Unretained(device),
-                       std::move(callback)));
   } else {
-    device->GetRuntime()->RequestMagicWindowSession(
-        base::BindOnce(&VRDisplayHost::OnMagicWindowSessionCreated,
-                       weak_ptr_factory_.GetWeakPtr(), std::move(callback)));
+    base::OnceCallback<void(device::mojom::XRSessionPtr,
+                            device::mojom::XRSessionControllerPtr)>
+        magic_window_callback =
+            base::BindOnce(&VRDisplayHost::OnMagicWindowSessionCreated,
+                           weak_ptr_factory_.GetWeakPtr(), std::move(callback));
+    device->GetRuntime()->RequestSession(std::move(runtime_options),
+                                         std::move(magic_window_callback));
   }
 }
 
diff --git a/chrome/browser/vr/service/vr_display_host.h b/chrome/browser/vr/service/vr_display_host.h
index 2e8e9e6..e8b673e3 100644
--- a/chrome/browser/vr/service/vr_display_host.h
+++ b/chrome/browser/vr/service/vr_display_host.h
@@ -73,12 +73,8 @@
   bool InternalSupportsSession(device::mojom::XRSessionOptions* options);
   void OnMagicWindowSessionCreated(
       device::mojom::VRDisplayHost::RequestSessionCallback callback,
-      device::mojom::VRMagicWindowProviderPtr session,
+      device::mojom::XRSessionPtr session,
       device::mojom::XRSessionControllerPtr controller);
-  void OnARSessionCreated(
-      vr::BrowserXrDevice* device,
-      device::mojom::VRDisplayHost::RequestSessionCallback callback,
-      device::mojom::XRSessionPtr session);
 
   // TODO(https://crbug.com/837538): Instead, check before returning this
   // object.
diff --git a/chrome/browser/vr/ui_unittest.cc b/chrome/browser/vr/ui_unittest.cc
index 1664f6d55..d0a4eeeb 100644
--- a/chrome/browser/vr/ui_unittest.cc
+++ b/chrome/browser/vr/ui_unittest.cc
@@ -467,17 +467,15 @@
   EXPECT_EQ(initial_position, content_group->LocalTransform());
 }
 
-TEST_F(UiTest, SecurityIconClickTriggersUnsupportedMode) {
+TEST_F(UiTest, SecurityIconClickShouldShowPageInfo) {
   CreateScene(kNotInWebVr);
 
   // Initial state.
   VerifyOnlyElementsVisible("Initial", kElementsVisibleInBrowsing);
 
-  // Clicking on security icon should trigger unsupported mode.
-  EXPECT_CALL(*browser_,
-              OnUnsupportedMode(UiUnsupportedMode::kUnhandledPageInfo));
-  browser_->OnUnsupportedMode(UiUnsupportedMode::kUnhandledPageInfo);
-  VerifyOnlyElementsVisible("Prompt invisible", kElementsVisibleInBrowsing);
+  EXPECT_CALL(*browser_, ShowPageInfo);
+  auto* security_icon = scene_->GetUiElementByName(kUrlBarSecurityButton);
+  ClickElement(security_icon);
 }
 
 TEST_F(UiTest, ClickingOmniboxTriggersUnsupportedMode) {
diff --git a/chrome/common/media_router/providers/cast/cast_media_source.cc b/chrome/common/media_router/providers/cast/cast_media_source.cc
index 06f0047..8f97a0d 100644
--- a/chrome/common/media_router/providers/cast/cast_media_source.cc
+++ b/chrome/common/media_router/providers/cast/cast_media_source.cc
@@ -4,6 +4,7 @@
 
 #include "chrome/common/media_router/providers/cast/cast_media_source.h"
 
+#include "base/strings/string_number_conversions.h"
 #include "base/strings/string_split.h"
 #include "base/strings/utf_string_conversions.h"
 #include "build/build_config.h"
@@ -18,18 +19,19 @@
 
 namespace {
 
-constexpr char kMirroringAppId[] = "0F5096E8";
-constexpr char kAudioMirroringAppId[] = "85CDB22F";
-
 // Parameter keys used by new Cast URLs.
 constexpr char kCapabilitiesKey[] = "capabilities";
 constexpr char kBroadcastNamespaceKey[] = "broadcastNamespace";
 constexpr char kBroadcastMessageKey[] = "broadcastMessage";
+constexpr char kClientIdKey[] = "clientId";
+constexpr char kLaunchTimeoutKey[] = "launchTimeout";
 
 // Parameter keys used by legacy Cast URLs.
 constexpr char kLegacyAppIdKey[] = "__castAppId__";
 constexpr char kLegacyBroadcastNamespaceKey[] = "__castBroadcastNamespace__";
 constexpr char kLegacyBroadcastMessageKey[] = "__castBroadcastMessage__";
+constexpr char kLegacyClientIdKey[] = "__castClientId__";
+constexpr char kLegacyLaunchTimeoutKey[] = "__castLaunchTimeout__";
 
 // TODO(imcheng): Move to common utils?
 std::string DecodeURLComponent(const std::string& encoded) {
@@ -61,8 +63,9 @@
 std::unique_ptr<CastMediaSource> CastMediaSourceForTabMirroring(
     const MediaSource::Id& source_id) {
   return std::make_unique<CastMediaSource>(
-      source_id, std::vector<CastAppInfo>({CastAppInfo(kMirroringAppId),
-                                           CastAppInfo(kAudioMirroringAppId)}));
+      source_id,
+      std::vector<CastAppInfo>({CastAppInfo(kCastStreamingAppId),
+                                CastAppInfo(kCastStreamingAudioAppId)}));
 }
 
 std::unique_ptr<CastMediaSource> CastMediaSourceForDesktopMirroring(
@@ -70,28 +73,33 @@
 // Desktop audio mirroring is only supported on some platforms.
 #if defined(OS_WIN) || defined(OS_CHROMEOS)
   return std::make_unique<CastMediaSource>(
-      source_id, std::vector<CastAppInfo>({CastAppInfo(kMirroringAppId),
-                                           CastAppInfo(kAudioMirroringAppId)}));
+      source_id,
+      std::vector<CastAppInfo>({CastAppInfo(kCastStreamingAppId),
+                                CastAppInfo(kCastStreamingAudioAppId)}));
 #else
   return std::make_unique<CastMediaSource>(
-      source_id, std::vector<CastAppInfo>({CastAppInfo(kMirroringAppId)}));
+      source_id, std::vector<CastAppInfo>({CastAppInfo(kCastStreamingAppId)}));
 #endif
 }
 
 std::unique_ptr<CastMediaSource> CreateFromURLParams(
     const MediaSource::Id& source_id,
     const std::vector<CastAppInfo>& app_infos,
+    const std::string& client_id,
     const std::string& broadcast_namespace,
-    const std::string& broadcast_message) {
+    const std::string& broadcast_message,
+    base::TimeDelta launch_timeout) {
   if (app_infos.empty())
     return nullptr;
 
   auto cast_source = std::make_unique<CastMediaSource>(source_id, app_infos);
+  cast_source->set_client_id(client_id);
   if (!broadcast_namespace.empty() && !broadcast_message.empty()) {
     cast_source->set_broadcast_request(
         BroadcastRequest(broadcast_namespace, broadcast_message));
   }
-
+  if (launch_timeout > base::TimeDelta())
+    cast_source->set_launch_timeout(launch_timeout);
   return cast_source;
 }
 
@@ -103,6 +111,8 @@
     return nullptr;
 
   std::string broadcast_namespace, broadcast_message, capabilities;
+  std::string client_id;
+  int launch_timeout_millis = 0;
   for (net::QueryIterator query_it(url); !query_it.IsAtEnd();
        query_it.Advance()) {
     std::string key = query_it.GetKey();
@@ -116,6 +126,12 @@
       broadcast_message = DecodeURLComponent(value);
     } else if (key == kCapabilitiesKey) {
       capabilities = value;
+    } else if (key == kClientIdKey) {
+      client_id = value;
+    } else if (key == kLaunchTimeoutKey) {
+      if (!base::StringToInt(value, &launch_timeout_millis) ||
+          launch_timeout_millis < 0)
+        launch_timeout_millis = 0;
     }
   }
 
@@ -129,8 +145,9 @@
     }
   }
 
-  return CreateFromURLParams(source_id, {app_info}, broadcast_namespace,
-                             broadcast_message);
+  return CreateFromURLParams(
+      source_id, {app_info}, client_id, broadcast_namespace, broadcast_message,
+      base::TimeDelta::FromMilliseconds(launch_timeout_millis));
 }
 
 std::unique_ptr<CastMediaSource> ParseLegacyCastUrl(
@@ -141,6 +158,8 @@
   // Legacy URLs can specify multiple apps.
   std::vector<std::string> app_id_params;
   std::string broadcast_namespace, broadcast_message;
+  std::string client_id;
+  int launch_timeout_millis = 0;
   for (const auto& key_value : parameters) {
     const auto& key = key_value.first;
     const auto& value = key_value.second;
@@ -151,6 +170,12 @@
     } else if (key == kLegacyBroadcastMessageKey) {
       // The broadcast message is URL-encoded.
       broadcast_message = DecodeURLComponent(value);
+    } else if (key == kLegacyClientIdKey) {
+      client_id = value;
+    } else if (key == kLegacyLaunchTimeoutKey) {
+      if (!base::StringToInt(value, &launch_timeout_millis) ||
+          launch_timeout_millis < 0)
+        launch_timeout_millis = 0;
     }
   }
 
@@ -187,8 +212,9 @@
   if (app_infos.empty())
     return nullptr;
 
-  return CreateFromURLParams(source_id, app_infos, broadcast_namespace,
-                             broadcast_message);
+  return CreateFromURLParams(
+      source_id, app_infos, client_id, broadcast_namespace, broadcast_message,
+      base::TimeDelta::FromMilliseconds(launch_timeout_millis));
 }
 
 }  // namespace
@@ -226,9 +252,6 @@
 }
 
 CastMediaSource::CastMediaSource(const MediaSource::Id& source_id,
-                                 const CastAppInfo& app_info)
-    : source_id_(source_id), app_infos_({app_info}) {}
-CastMediaSource::CastMediaSource(const MediaSource::Id& source_id,
                                  const std::vector<CastAppInfo>& app_infos)
     : source_id_(source_id), app_infos_(app_infos) {}
 CastMediaSource::CastMediaSource(const CastMediaSource& other) = default;
diff --git a/chrome/common/media_router/providers/cast/cast_media_source.h b/chrome/common/media_router/providers/cast/cast_media_source.h
index ae090b5e2..e46b725 100644
--- a/chrome/common/media_router/providers/cast/cast_media_source.h
+++ b/chrome/common/media_router/providers/cast/cast_media_source.h
@@ -16,6 +16,16 @@
 
 namespace media_router {
 
+static constexpr char kCastStreamingAppId[] = "0F5096E8";
+static constexpr char kCastStreamingAudioAppId[] = "85CDB22F";
+
+// Placeholder app ID advertised by the multizone leader in a receiver status
+// message.
+static constexpr char kMultizoneLeaderAppId[] = "MultizoneLeader";
+
+static constexpr base::TimeDelta kDefaultLaunchTimeout =
+    base::TimeDelta::FromSeconds(60);
+
 // Represents a Cast app and its capabilitity requirements.
 struct CastAppInfo {
   explicit CastAppInfo(const std::string& app_id);
@@ -39,10 +49,8 @@
   // Returns the parsed form of |source|, or nullptr if it cannot be parsed.
   static std::unique_ptr<CastMediaSource> From(const MediaSource::Id& source);
 
-  explicit CastMediaSource(const MediaSource::Id& source_id,
-                           const CastAppInfo& app_info);
-  explicit CastMediaSource(const MediaSource::Id& source_id,
-                           const std::vector<CastAppInfo>& app_infos);
+  CastMediaSource(const MediaSource::Id& source_id,
+                  const std::vector<CastAppInfo>& app_infos);
   CastMediaSource(const CastMediaSource& other);
   ~CastMediaSource();
 
@@ -55,6 +63,12 @@
 
   const MediaSource::Id& source_id() const { return source_id_; }
   const std::vector<CastAppInfo>& app_infos() const { return app_infos_; }
+  const std::string& client_id() const { return client_id_; }
+  void set_client_id(const std::string& client_id) { client_id_ = client_id; }
+  base::TimeDelta launch_timeout() const { return launch_timeout_; }
+  void set_launch_timeout(base::TimeDelta launch_timeout) {
+    launch_timeout_ = launch_timeout;
+  }
   const base::Optional<cast_channel::BroadcastRequest>& broadcast_request()
       const {
     return broadcast_request_;
@@ -68,6 +82,9 @@
   // TODO(imcheng): Fill in other parameters.
   MediaSource::Id source_id_;
   std::vector<CastAppInfo> app_infos_;
+  base::TimeDelta launch_timeout_ = kDefaultLaunchTimeout;
+  // Empty if not set.
+  std::string client_id_;
   base::Optional<cast_channel::BroadcastRequest> broadcast_request_;
 };
 
diff --git a/chrome/common/media_router/providers/cast/cast_media_source_unittest.cc b/chrome/common/media_router/providers/cast/cast_media_source_unittest.cc
index 94a9204..54181fdc 100644
--- a/chrome/common/media_router/providers/cast/cast_media_source_unittest.cc
+++ b/chrome/common/media_router/providers/cast/cast_media_source_unittest.cc
@@ -13,7 +13,9 @@
   MediaSource::Id source_id(
       "cast:ABCDEFAB?capabilities=video_out,audio_out"
       "&broadcastNamespace=namespace"
-      "&broadcastMessage=message");
+      "&broadcastMessage=message"
+      "&clientId=12345"
+      "&launchTimeout=30000");
   std::unique_ptr<CastMediaSource> source = CastMediaSource::From(source_id);
   ASSERT_TRUE(source);
   EXPECT_EQ(source_id, source->source_id());
@@ -27,13 +29,17 @@
   ASSERT_TRUE(broadcast_request);
   EXPECT_EQ("namespace", broadcast_request->broadcast_namespace);
   EXPECT_EQ("message", broadcast_request->message);
+  EXPECT_EQ("12345", source->client_id());
+  EXPECT_EQ(base::TimeDelta::FromMilliseconds(30000), source->launch_timeout());
 }
 
 TEST(CastMediaSourceTest, FromLegacyCastURL) {
   MediaSource::Id source_id(
       "https://google.com/cast#__castAppId__=ABCDEFAB(video_out,audio_out)"
       "/__castBroadcastNamespace__=namespace"
-      "/__castBroadcastMessage__=message");
+      "/__castBroadcastMessage__=message"
+      "/__castClientId__=12345"
+      "/__castLaunchTimeout__=30000");
   std::unique_ptr<CastMediaSource> source = CastMediaSource::From(source_id);
   ASSERT_TRUE(source);
   EXPECT_EQ(source_id, source->source_id());
@@ -47,6 +53,8 @@
   ASSERT_TRUE(broadcast_request);
   EXPECT_EQ("namespace", broadcast_request->broadcast_namespace);
   EXPECT_EQ("message", broadcast_request->message);
+  EXPECT_EQ("12345", source->client_id());
+  EXPECT_EQ(base::TimeDelta::FromMilliseconds(30000), source->launch_timeout());
 }
 
 TEST(CastMediaSourceTest, FromPresentationURL) {
@@ -57,6 +65,8 @@
   ASSERT_EQ(2u, source->app_infos().size());
   EXPECT_EQ("0F5096E8", source->app_infos()[0].app_id);
   EXPECT_EQ("85CDB22F", source->app_infos()[1].app_id);
+  EXPECT_TRUE(source->client_id().empty());
+  EXPECT_EQ(kDefaultLaunchTimeout, source->launch_timeout());
 }
 
 TEST(CastMediaSourceTest, FromMirroringURN) {
@@ -67,6 +77,8 @@
   ASSERT_EQ(2u, source->app_infos().size());
   EXPECT_EQ("0F5096E8", source->app_infos()[0].app_id);
   EXPECT_EQ("85CDB22F", source->app_infos()[1].app_id);
+  EXPECT_TRUE(source->client_id().empty());
+  EXPECT_EQ(kDefaultLaunchTimeout, source->launch_timeout());
 }
 
 TEST(CastMediaSourceTest, FromInvalidSource) {
diff --git a/chrome/common/pref_names.cc b/chrome/common/pref_names.cc
index 70fddd7..d56dc0ba 100644
--- a/chrome/common/pref_names.cc
+++ b/chrome/common/pref_names.cc
@@ -1997,11 +1997,9 @@
 #endif  // !defined(OS_ANDROID)
 
 #if !defined(OS_ANDROID)
-#if !defined(OS_CHROMEOS)
 // Pref name for the policy controlling the way in which users are notified of
 // the need to relaunch the browser for a pending update.
 const char kRelaunchNotification[] = "browser.relaunch_notification";
-#endif  // !defined(OS_CHROMEOS)
 // Pref name for the policy controlling the time period over which users are
 // notified of the need to relaunch the browser for a pending update. Values
 // are in milliseconds.
diff --git a/chrome/common/pref_names.h b/chrome/common/pref_names.h
index c41685c..d2184936b 100644
--- a/chrome/common/pref_names.h
+++ b/chrome/common/pref_names.h
@@ -728,9 +728,7 @@
 #endif  // !defined(OS_ANDROID)
 
 #if !defined(OS_ANDROID)
-#if !defined(OS_CHROMEOS)
 extern const char kRelaunchNotification[];
-#endif  // !defined(OS_CHROMEOS)
 extern const char kRelaunchNotificationPeriod[];
 #endif  // !defined(OS_ANDROID)
 
diff --git a/chrome/test/BUILD.gn b/chrome/test/BUILD.gn
index db3480c6..2f8e1bf 100644
--- a/chrome/test/BUILD.gn
+++ b/chrome/test/BUILD.gn
@@ -1474,6 +1474,8 @@
         "../browser/ui/views/payments/shipping_address_editor_view_controller_browsertest.cc",
         "../browser/ui/views/payments/shipping_option_view_controller_browsertest.cc",
         "../browser/ui/views/profiles/profile_chooser_view_browsertest.cc",
+        "../browser/ui/views/relaunch_notification/relaunch_recommended_bubble_view_browsertest.cc",
+        "../browser/ui/views/relaunch_notification/relaunch_required_dialog_view_browsertest.cc",
         "../browser/ui/views/safe_browsing/password_reuse_modal_warning_dialog_browsertest.cc",
         "../browser/ui/views/select_file_dialog_extension_browsertest.cc",
         "../browser/ui/views/sync/profile_signin_confirmation_dialog_views_browsertest.cc",
@@ -1527,12 +1529,6 @@
           "../browser/ui/views/web_dialog_view_browsertest.cc",
         ]
       }
-      if (!is_chromeos) {
-        sources += [
-          "../browser/ui/views/relaunch_notification/relaunch_recommended_bubble_view_browsertest.cc",
-          "../browser/ui/views/relaunch_notification/relaunch_required_dialog_view_browsertest.cc",
-        ]
-      }
     }
 
     if (is_linux && !is_component_build) {
@@ -3205,6 +3201,7 @@
       "../browser/media/router/mojo/media_sink_service_status_unittest.cc",
       "../browser/media/router/providers/cast/cast_app_availability_tracker_unittest.cc",
       "../browser/media/router/providers/cast/cast_app_discovery_service_unittest.cc",
+      "../browser/media/router/providers/cast/cast_internal_message_util_unittest.cc",
       "../browser/media/router/providers/cast/cast_media_route_provider_metrics_unittest.cc",
       "../browser/media/router/providers/cast/cast_media_route_provider_unittest.cc",
       "../browser/media/router/providers/cast/dual_media_sink_service_unittest.cc",
@@ -3472,7 +3469,6 @@
       "../browser/extensions/api/signed_in_devices/id_mapping_helper_unittest.cc",
       "../browser/extensions/api/signed_in_devices/signed_in_devices_api_unittest.cc",
       "../browser/extensions/api/signed_in_devices/signed_in_devices_manager_unittest.cc",
-      "../browser/extensions/api/socket/combined_socket_unittest.cc",
       "../browser/extensions/api/socket/socket_api_unittest.cc",
       "../browser/extensions/api/socket/tcp_socket_unittest.cc",
       "../browser/extensions/api/socket/tls_socket_unittest.cc",
@@ -4168,7 +4164,6 @@
         "../browser/ui/cocoa/omnibox/omnibox_popup_view_mac_unittest.mm",
         "../browser/ui/cocoa/omnibox/omnibox_view_mac_unittest.mm",
         "../browser/ui/cocoa/page_info/page_info_bubble_controller_unittest.mm",
-        "../browser/ui/cocoa/page_info/permission_selector_button_unittest.mm",
         "../browser/ui/cocoa/profiles/avatar_button_controller_unittest.mm",
         "../browser/ui/cocoa/profiles/avatar_button_unittest.mm",
         "../browser/ui/cocoa/profiles/avatar_icon_controller_unittest.mm",
@@ -4202,7 +4197,6 @@
         "../browser/ui/cocoa/touchbar/browser_window_default_touch_bar_unittest.mm",
         "../browser/ui/cocoa/touchbar/credit_card_autofill_touch_bar_controller_unittest.mm",
         "../browser/ui/cocoa/touchbar/suggested_text_touch_bar_controller_unittest.mm",
-        "../browser/ui/cocoa/translate/translate_bubble_controller_unittest.mm",
         "../browser/ui/cocoa/url_drop_target_unittest.mm",
         "../browser/ui/cocoa/vertical_gradient_view_unittest.mm",
         "../browser/ui/cocoa/view_resizer_pong.h",
@@ -4362,14 +4356,11 @@
       "../browser/ui/views/payments/payment_request_item_list_unittest.cc",
       "../browser/ui/views/payments/validating_textfield_unittest.cc",
       "../browser/ui/views/payments/view_stack_unittest.cc",
+      "../browser/ui/views/relaunch_notification/relaunch_notification_controller_unittest.cc",
+      "../browser/ui/views/relaunch_notification/relaunch_required_dialog_view_unittest.cc",
     ]
     if (is_chromeos) {
       sources += [ "../browser/ui/views/ime_driver/input_method_bridge_chromeos_unittest.cc" ]
-    } else {
-      sources += [
-        "../browser/ui/views/relaunch_notification/relaunch_notification_controller_unittest.cc",
-        "../browser/ui/views/relaunch_notification/relaunch_required_dialog_view_unittest.cc",
-      ]
     }
     if (!is_chromeos && (!is_mac || mac_views_browser)) {
       sources += [
@@ -5108,10 +5099,6 @@
         # toolkit-views secondary UI. It should not be deleted with the Cocoa
         # bubble (but it can be deleted with the Cocoa browser window).
         "../browser/ui/cocoa/permission_bubble/permission_bubble_cocoa_interactive_uitest.mm",
-
-        # The translate bubble tests only obsolete Cocoa UI and doesn't link.
-        # Ensure the corresponding file is deleted when removing the next line.
-        # "../browser/ui/cocoa/translate/translate_bubble_test_utils_views_cocoa.mm",
       ]
     }
 
diff --git a/chrome/test/base/dialog_test_browser_window.cc b/chrome/test/base/dialog_test_browser_window.cc
index bf88d63..d1110de3 100644
--- a/chrome/test/base/dialog_test_browser_window.cc
+++ b/chrome/test/base/dialog_test_browser_window.cc
@@ -4,6 +4,7 @@
 
 #include "chrome/test/base/dialog_test_browser_window.h"
 
+#include "build/build_config.h"
 #include "chrome/browser/ui/browser.h"
 #include "chrome/browser/ui/browser_list.h"
 #include "chrome/browser/ui/tabs/tab_strip_model.h"
@@ -52,6 +53,10 @@
 }
 
 gfx::Size DialogTestBrowserWindow::GetMaximumDialogSize() {
+#if defined(OS_MACOSX)
+  // Zero-size windows aren't allowed on Mac.
+  return gfx::Size(1, 1);
+#endif
   return gfx::Size();
 }
 
diff --git a/chrome/test/data/extensions/api_test/socket/api/background.js b/chrome/test/data/extensions/api_test/socket/api/background.js
index d093aa0..4e675fa8 100644
--- a/chrome/test/data/extensions/api_test/socket/api/background.js
+++ b/chrome/test/data/extensions/api_test/socket/api/background.js
@@ -200,16 +200,16 @@
     socket.read(acceptedSocketId, function(readInfo) {
       arrayBuffer2String(readInfo.data, function (s) {
         assertDataMatch(request, s);
-        succeeded = true;
-        // Test whether socket.getInfo correctly reflects the connection status
-        // if the peer has closed the connection.
-        setTimeout(function() {
+        // Rather than using a timeout, use another read to detect the peer
+        // termination.
+        socket.read(acceptedSocketId, function(readInfo2) {
+          chrome.test.assertEq(-2, readInfo2.resultCode);
           socket.getInfo(acceptedSocketId, function(info) {
             chrome.test.assertFalse(info.connected);
             socket.destroy(socketId);
             chrome.test.succeed();
           });
-        }, 500);
+        });
       });
     });
   }
diff --git a/chrome/test/data/policy/policy_test_cases.json b/chrome/test/data/policy/policy_test_cases.json
index 0383715..9dbacec 100644
--- a/chrome/test/data/policy/policy_test_cases.json
+++ b/chrome/test/data/policy/policy_test_cases.json
@@ -2453,7 +2453,7 @@
   },
 
   "RelaunchNotification": {
-    "os": ["win", "linux", "mac"],
+    "os": ["win", "linux", "mac", "chromeos"],
     "test_policy": { "RelaunchNotification": 1 },
     "pref_mappings": [
       { "pref": "browser.relaunch_notification",
diff --git a/components/autofill/core/browser/BUILD.gn b/components/autofill/core/browser/BUILD.gn
index 9753eec..16621e5 100644
--- a/components/autofill/core/browser/BUILD.gn
+++ b/components/autofill/core/browser/BUILD.gn
@@ -183,6 +183,8 @@
     "webdata/autofill_table_encryptor_factory.h",
     "webdata/autofill_wallet_metadata_syncable_service.cc",
     "webdata/autofill_wallet_metadata_syncable_service.h",
+    "webdata/autofill_wallet_sync_bridge.cc",
+    "webdata/autofill_wallet_sync_bridge.h",
     "webdata/autofill_wallet_syncable_service.cc",
     "webdata/autofill_wallet_syncable_service.h",
     "webdata/autofill_webdata.h",
diff --git a/components/autofill/core/browser/webdata/OWNERS b/components/autofill/core/browser/webdata/OWNERS
new file mode 100644
index 0000000..7385936a
--- /dev/null
+++ b/components/autofill/core/browser/webdata/OWNERS
@@ -0,0 +1 @@
+per-file *sync_bridge*=jkrcal@chromium.org
diff --git a/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.cc b/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.cc
new file mode 100644
index 0000000..3296c81
--- /dev/null
+++ b/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.cc
@@ -0,0 +1,90 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h"
+
+#include <utility>
+
+#include "base/logging.h"
+#include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
+#include "components/sync/model_impl/client_tag_based_model_type_processor.h"
+
+namespace autofill {
+
+namespace {
+
+// Address to this variable used as the user data key.
+static int kAutofillWalletSyncBridgeUserDataKey = 0;
+
+}  // namespace
+
+// static
+void AutofillWalletSyncBridge::CreateForWebDataServiceAndBackend(
+    const std::string& app_locale,
+    AutofillWebDataBackend* web_data_backend,
+    AutofillWebDataService* web_data_service) {
+  web_data_service->GetDBUserData()->SetUserData(
+      &kAutofillWalletSyncBridgeUserDataKey,
+      std::make_unique<AutofillWalletSyncBridge>(
+          std::make_unique<syncer::ClientTagBasedModelTypeProcessor>(
+              syncer::AUTOFILL_WALLET_DATA,
+              /*dump_stack=*/base::RepeatingClosure())));
+}
+
+// static
+syncer::ModelTypeSyncBridge* AutofillWalletSyncBridge::FromWebDataService(
+    AutofillWebDataService* web_data_service) {
+  return static_cast<AutofillWalletSyncBridge*>(
+      web_data_service->GetDBUserData()->GetUserData(
+          &kAutofillWalletSyncBridgeUserDataKey));
+}
+
+AutofillWalletSyncBridge::AutofillWalletSyncBridge(
+    std::unique_ptr<syncer::ModelTypeChangeProcessor> change_processor)
+    : ModelTypeSyncBridge(std::move(change_processor)) {}
+
+AutofillWalletSyncBridge::~AutofillWalletSyncBridge() {}
+
+std::unique_ptr<syncer::MetadataChangeList>
+AutofillWalletSyncBridge::CreateMetadataChangeList() {
+  NOTIMPLEMENTED();
+  return nullptr;
+}
+
+base::Optional<syncer::ModelError> AutofillWalletSyncBridge::MergeSyncData(
+    std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
+    syncer::EntityChangeList entity_data) {
+  return ApplySyncChanges(std::move(metadata_change_list),
+                          std::move(entity_data));
+}
+
+base::Optional<syncer::ModelError> AutofillWalletSyncBridge::ApplySyncChanges(
+    std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
+    syncer::EntityChangeList entity_data) {
+  NOTIMPLEMENTED();
+  return base::nullopt;
+}
+
+void AutofillWalletSyncBridge::GetData(StorageKeyList storage_keys,
+                                       DataCallback callback) {
+  NOTIMPLEMENTED();
+}
+
+void AutofillWalletSyncBridge::GetAllDataForDebugging(DataCallback callback) {
+  NOTIMPLEMENTED();
+}
+
+std::string AutofillWalletSyncBridge::GetClientTag(
+    const syncer::EntityData& entity_data) {
+  NOTIMPLEMENTED();
+  return "";
+}
+
+std::string AutofillWalletSyncBridge::GetStorageKey(
+    const syncer::EntityData& entity_data) {
+  NOTIMPLEMENTED();
+  return "";
+}
+
+}  // namespace autofill
diff --git a/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h b/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h
new file mode 100644
index 0000000..6037895
--- /dev/null
+++ b/components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h
@@ -0,0 +1,63 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef COMPONENTS_AUTOFILL_CORE_BROWSER_WEBDATA_AUTOFILL_WALLET_SYNC_BRIDGE_H_
+#define COMPONENTS_AUTOFILL_CORE_BROWSER_WEBDATA_AUTOFILL_WALLET_SYNC_BRIDGE_H_
+
+#include <memory>
+#include <string>
+
+#include "base/macros.h"
+#include "base/supports_user_data.h"
+#include "components/sync/model/metadata_change_list.h"
+#include "components/sync/model/model_error.h"
+#include "components/sync/model/model_type_change_processor.h"
+#include "components/sync/model/model_type_sync_bridge.h"
+
+namespace autofill {
+
+class AutofillWebDataBackend;
+class AutofillWebDataService;
+
+// Sync bridge responsible for propagating local changes to the processor and
+// applying remote changes to the local database.
+class AutofillWalletSyncBridge : public base::SupportsUserData::Data,
+                                 public syncer::ModelTypeSyncBridge {
+ public:
+  // Factory method that hides dealing with change_processor and also stores the
+  // created bridge within |web_data_service|. This method should only be
+  // called on |web_data_service|'s DB thread.
+  static void CreateForWebDataServiceAndBackend(
+      const std::string& app_locale,
+      AutofillWebDataBackend* webdata_backend,
+      AutofillWebDataService* web_data_service);
+
+  static syncer::ModelTypeSyncBridge* FromWebDataService(
+      AutofillWebDataService* web_data_service);
+
+  explicit AutofillWalletSyncBridge(
+      std::unique_ptr<syncer::ModelTypeChangeProcessor> change_processor);
+  ~AutofillWalletSyncBridge() override;
+
+  // ModelTypeSyncBridge implementation.
+  std::unique_ptr<syncer::MetadataChangeList> CreateMetadataChangeList()
+      override;
+  base::Optional<syncer::ModelError> MergeSyncData(
+      std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
+      syncer::EntityChangeList entity_data) override;
+  base::Optional<syncer::ModelError> ApplySyncChanges(
+      std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
+      syncer::EntityChangeList entity_changes) override;
+  void GetData(StorageKeyList storage_keys, DataCallback callback) override;
+  void GetAllDataForDebugging(DataCallback callback) override;
+  std::string GetClientTag(const syncer::EntityData& entity_data) override;
+  std::string GetStorageKey(const syncer::EntityData& entity_data) override;
+
+ private:
+  DISALLOW_COPY_AND_ASSIGN(AutofillWalletSyncBridge);
+};
+
+}  // namespace autofill
+
+#endif  // COMPONENTS_AUTOFILL_CORE_BROWSER_WEBDATA_AUTOFILL_WALLET_SYNC_BRIDGE_H_
diff --git a/components/browser_sync/profile_sync_components_factory_impl.cc b/components/browser_sync/profile_sync_components_factory_impl.cc
index 0cec2e2..713f268 100644
--- a/components/browser_sync/profile_sync_components_factory_impl.cc
+++ b/components/browser_sync/profile_sync_components_factory_impl.cc
@@ -14,6 +14,7 @@
 #include "components/autofill/core/browser/webdata/autocomplete_sync_bridge.h"
 #include "components/autofill/core/browser/webdata/autofill_profile_data_type_controller.h"
 #include "components/autofill/core/browser/webdata/autofill_profile_sync_bridge.h"
+#include "components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h"
 #include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
 #include "components/autofill/core/browser/webdata/web_data_model_type_controller.h"
 #include "components/browser_sync/browser_sync_switches.h"
@@ -151,9 +152,18 @@
     // enforced by the datatype controller. Register unless explicitly disabled.
     bool wallet_disabled = disabled_types.Has(syncer::AUTOFILL_WALLET_DATA);
     if (!wallet_disabled) {
-      controllers.push_back(std::make_unique<AutofillWalletDataTypeController>(
-          syncer::AUTOFILL_WALLET_DATA, db_thread_, error_callback,
-          sync_client_, web_data_service_));
+      if (base::FeatureList::IsEnabled(switches::kSyncUSSAutofillWalletData)) {
+        controllers.push_back(
+            std::make_unique<autofill::WebDataModelTypeController>(
+                syncer::AUTOFILL_WALLET_DATA, sync_client_, db_thread_,
+                web_data_service_,
+                base::BindRepeating(&AutofillProfileDelegateFromDataService)));
+      } else {
+        controllers.push_back(
+            std::make_unique<AutofillWalletDataTypeController>(
+                syncer::AUTOFILL_WALLET_DATA, db_thread_, error_callback,
+                sync_client_, web_data_service_));
+      }
     }
 
     // Wallet metadata sync depends on Wallet data sync. Register if Wallet data
diff --git a/components/cast_channel/cast_message_util.cc b/components/cast_channel/cast_message_util.cc
index 011ec8e..790c73c 100644
--- a/components/cast_channel/cast_message_util.cc
+++ b/components/cast_channel/cast_message_util.cc
@@ -34,13 +34,14 @@
 constexpr char kTypeNodeId[] = "type";
 constexpr char kRequestIdNodeId[] = "requestId";
 
-// Cast application protocol message types.
+// Cast application protocol message types. Keep in sync with CastMessageType.
 constexpr char kKeepAlivePingType[] = "PING";
 constexpr char kKeepAlivePongType[] = "PONG";
 constexpr char kGetAppAvailabilityRequestType[] = "GET_APP_AVAILABILITY";
 constexpr char kConnectionRequestType[] = "CONNECT";
 constexpr char kBroadcastRequestType[] = "APPLICATION_BROADCAST";
 constexpr char kLaunchRequestType[] = "LAUNCH";
+constexpr char kStopRequestType[] = "STOP";
 constexpr char kReceiverStatusType[] = "RECEIVER_STATUS";
 constexpr char kLaunchErrorType[] = "LAUNCH_ERROR";
 
@@ -107,6 +108,7 @@
   if (!message.has_payload_utf8())
     return nullptr;
 
+  // TODO(https://crbug.com/809249): Parse JSON using data_decoder service.
   return base::DictionaryValue::From(
       base::JSONReader::Read(message.payload_utf8()));
 }
@@ -146,6 +148,8 @@
       return kBroadcastRequestType;
     case CastMessageType::kLaunch:
       return kLaunchRequestType;
+    case CastMessageType::kStop:
+      return kStopRequestType;
     case CastMessageType::kReceiverStatus:
       return kReceiverStatusType;
     case CastMessageType::kLaunchError:
@@ -170,6 +174,8 @@
     return CastMessageType::kBroadcast;
   if (type == kLaunchRequestType)
     return CastMessageType::kLaunch;
+  if (type == kStopRequestType)
+    return CastMessageType::kStop;
   if (type == kReceiverStatusType)
     return CastMessageType::kReceiverStatus;
   if (type == kLaunchErrorType)
@@ -347,6 +353,17 @@
                            kPlatformReceiverId);
 }
 
+CastMessage CreateStopRequest(const std::string& source_id,
+                              int request_id,
+                              const std::string& session_id) {
+  Value dict(Value::Type::DICTIONARY);
+  dict.SetKey(kTypeNodeId, Value(kStopRequestType));
+  dict.SetKey(kRequestIdNodeId, Value(request_id));
+  dict.SetKey("sessionId", Value(session_id));
+  return CreateCastMessage(kReceiverNamespace, dict, source_id,
+                           kPlatformReceiverId);
+}
+
 CastMessage CreateCastMessage(const std::string& message_namespace,
                               const base::Value& message,
                               const std::string& source_id,
@@ -406,8 +423,7 @@
     default;
 LaunchSessionResponse::~LaunchSessionResponse() = default;
 
-LaunchSessionResponse GetLaunchSessionResponse(
-    const base::DictionaryValue& payload) {
+LaunchSessionResponse GetLaunchSessionResponse(const base::Value& payload) {
   const Value* type_value =
       payload.FindKeyOfType(kTypeNodeId, Value::Type::STRING);
   if (!type_value)
diff --git a/components/cast_channel/cast_message_util.h b/components/cast_channel/cast_message_util.h
index aedf0bc..c854352 100644
--- a/components/cast_channel/cast_message_util.h
+++ b/components/cast_channel/cast_message_util.h
@@ -27,6 +27,7 @@
   kConnect,    // Virtual connection request
   kBroadcast,  // Application broadcast / precache
   kLaunch,     // Session launch request
+  kStop,       // Session stop request
   kReceiverStatus,
   kLaunchError,
   kOther  // Add new types above |kOther|.
@@ -132,6 +133,10 @@
                                 const std::string& app_id,
                                 const std::string& locale);
 
+CastMessage CreateStopRequest(const std::string& source_id,
+                              int request_id,
+                              const std::string& session_id);
+
 // Creates a generic CastMessage with |message| as the string payload. Used for
 // app messages.
 CastMessage CreateCastMessage(const std::string& message_namespace,
@@ -168,16 +173,15 @@
   ~LaunchSessionResponse();
 
   Result result = Result::kUnknown;
-
   // Populated if |result| is |kOk|.
   base::Optional<base::Value> receiver_status;
 };
 
 // Parses |payload| into a LaunchSessionResponse. Returns an empty
 // LaunchSessionResponse if |payload| is not a properly formatted launch
-// response. |payload| must be from the string payload of a CastMessage.
-LaunchSessionResponse GetLaunchSessionResponse(
-    const base::DictionaryValue& payload);
+// response. |payload| must be a dictionary from the string payload of a
+// CastMessage.
+LaunchSessionResponse GetLaunchSessionResponse(const base::Value& payload);
 
 }  // namespace cast_channel
 
diff --git a/components/cast_channel/cast_message_util_unittest.cc b/components/cast_channel/cast_message_util_unittest.cc
index 9fdbfa8..8e629cfd 100644
--- a/components/cast_channel/cast_message_util_unittest.cc
+++ b/components/cast_channel/cast_message_util_unittest.cc
@@ -6,6 +6,7 @@
 
 #include "base/json/json_reader.h"
 #include "base/values.h"
+#include "components/cast_channel/proto/cast_channel.pb.h"
 #include "testing/gtest/include/gtest/gtest.h"
 
 namespace cast_channel {
@@ -34,8 +35,7 @@
     }
   )";
 
-  std::unique_ptr<base::DictionaryValue> value =
-      base::DictionaryValue::From(base::JSONReader::Read(payload));
+  std::unique_ptr<base::Value> value = base::JSONReader::Read(payload);
   ASSERT_TRUE(value);
 
   LaunchSessionResponse response = GetLaunchSessionResponse(*value);
@@ -51,8 +51,7 @@
     }
   )";
 
-  std::unique_ptr<base::DictionaryValue> value =
-      base::DictionaryValue::From(base::JSONReader::Read(payload));
+  std::unique_ptr<base::Value> value = base::JSONReader::Read(payload);
   ASSERT_TRUE(value);
 
   LaunchSessionResponse response = GetLaunchSessionResponse(*value);
@@ -70,8 +69,7 @@
     }
   )";
 
-  std::unique_ptr<base::DictionaryValue> value =
-      base::DictionaryValue::From(base::JSONReader::Read(payload));
+  std::unique_ptr<base::Value> value = base::JSONReader::Read(payload);
   ASSERT_TRUE(value);
 
   LaunchSessionResponse response = GetLaunchSessionResponse(*value);
@@ -79,4 +77,26 @@
   EXPECT_FALSE(response.receiver_status);
 }
 
+TEST(CastMessageUtilTest, CreateStopRequest) {
+  std::string expected_message = R"(
+    {
+      "type": "STOP",
+      "requestId": 123,
+      "sessionId": "sessionId"
+    }
+  )";
+
+  std::unique_ptr<base::Value> expected_value =
+      base::JSONReader::Read(expected_message);
+  ASSERT_TRUE(expected_value);
+
+  CastMessage message = CreateStopRequest("sourceId", 123, "sessionId");
+  ASSERT_TRUE(IsCastMessageValid(message));
+
+  std::unique_ptr<base::Value> actual_value =
+      base::JSONReader::Read(message.payload_utf8());
+  ASSERT_TRUE(actual_value);
+  EXPECT_EQ(*expected_value, *actual_value);
+}
+
 }  // namespace cast_channel
diff --git a/components/consent_auditor/consent_auditor.h b/components/consent_auditor/consent_auditor.h
index 30e61ac..fa62710d 100644
--- a/components/consent_auditor/consent_auditor.h
+++ b/components/consent_auditor/consent_auditor.h
@@ -31,7 +31,7 @@
   GOOGLE_LOCATION_SERVICE = 3,
   CHROME_UNIFIED_CONSENT = 4,
 
-  FEATURE_LAST = GOOGLE_LOCATION_SERVICE
+  FEATURE_LAST = CHROME_UNIFIED_CONSENT
 };
 
 // Whether a consent is given or not given.
diff --git a/components/consent_auditor/consent_sync_bridge_impl.cc b/components/consent_auditor/consent_sync_bridge_impl.cc
index 05c60ed..264f32cb 100644
--- a/components/consent_auditor/consent_sync_bridge_impl.cc
+++ b/components/consent_auditor/consent_sync_bridge_impl.cc
@@ -51,16 +51,6 @@
   return entity_data;
 }
 
-// TODO(vitaliii): Delete this function both here and in UserEventSyncBridge.
-std::unique_ptr<EntityData> CopyToEntityData(
-    const UserConsentSpecifics specifics) {
-  auto entity_data = std::make_unique<EntityData>();
-  entity_data->non_unique_name =
-      base::Int64ToString(specifics.client_consent_time_usec());
-  *entity_data->specifics.mutable_user_consent() = specifics;
-  return entity_data;
-}
-
 }  // namespace
 
 ConsentSyncBridgeImpl::ConsentSyncBridgeImpl(
@@ -296,11 +286,12 @@
   }
 
   auto batch = std::make_unique<MutableDataBatch>();
-  UserConsentSpecifics specifics;
   for (const Record& r : *data_records) {
-    if (specifics.ParseFromString(r.value)) {
-      DCHECK_EQ(r.id, GetStorageKeyFromSpecifics(specifics));
-      batch->Put(r.id, CopyToEntityData(specifics));
+    auto specifics = std::make_unique<UserConsentSpecifics>();
+
+    if (specifics->ParseFromString(r.value)) {
+      DCHECK_EQ(r.id, GetStorageKeyFromSpecifics(*specifics));
+      batch->Put(r.id, MoveToEntityData(std::move(specifics)));
     } else {
       change_processor()->ReportError(
           {FROM_HERE, "Failed deserializing user events."});
diff --git a/components/password_manager/core/browser/password_bubble_experiment.cc b/components/password_manager/core/browser/password_bubble_experiment.cc
index 3c446d4..27022f7a 100644
--- a/components/password_manager/core/browser/password_bubble_experiment.cc
+++ b/components/password_manager/core/browser/password_bubble_experiment.cc
@@ -60,9 +60,14 @@
 bool ShouldShowChromeSignInPasswordPromo(
     PrefService* prefs,
     const syncer::SyncService* sync_service) {
-  if (!sync_service || !sync_service->IsSyncAllowed() ||
-      sync_service->IsFirstSetupComplete())
+  if (!sync_service ||
+      sync_service->HasDisableReason(
+          syncer::SyncService::DISABLE_REASON_PLATFORM_OVERRIDE) ||
+      sync_service->HasDisableReason(
+          syncer::SyncService::DISABLE_REASON_ENTERPRISE_POLICY) ||
+      sync_service->IsFirstSetupComplete()) {
     return false;
+  }
   // Don't show the promo more than 3 times.
   constexpr int kThreshold = 3;
   return !prefs->GetBoolean(
diff --git a/components/password_manager/core/browser/password_form_manager.cc b/components/password_manager/core/browser/password_form_manager.cc
index 3482396..8f58dadd 100644
--- a/components/password_manager/core/browser/password_form_manager.cc
+++ b/components/password_manager/core/browser/password_form_manager.cc
@@ -150,8 +150,7 @@
                              true /* should_migrate_http_passwords */,
                              true /* should_query_suppressed_https_forms */)),
       form_fetcher_(form_fetcher ? form_fetcher : owned_form_fetcher_.get()),
-      votes_uploader_(client, observed_form.IsPossibleChangePasswordForm()),
-      is_main_frame_secure_(client->IsMainFrameSecure()) {
+      votes_uploader_(client, observed_form.IsPossibleChangePasswordForm()) {
   // Non-HTML forms should not need any interaction with the renderer, and hence
   // no driver. Note that cloned PasswordFormManager instances can have HTML
   // forms without drivers as well.
@@ -625,17 +624,14 @@
 
   // Look for the actually submitted credentials in the list of previously saved
   // credentials that were available to autofilling.
-  // This first match via FindBestSavedMatch focuses on matches by username and
-  // falls back to password based matches if |submitted_form_| has no username
-  // filled.
   const PasswordForm* saved_form = FindBestSavedMatch(submitted_form_.get());
-  if (saved_form != nullptr) {
+  if (saved_form) {
     // The user signed in with a login we autofilled.
     pending_credentials_ = *saved_form;
     SetPasswordOverridden(pending_credentials_.password_value !=
                           password_to_save.first);
 
-    if (IsPendingCredentialsPublicSuffixMatch()) {
+    if (pending_credentials_.is_public_suffix_match) {
       // If the autofilled credentials were a PSL match or credentials stored
       // from Android apps, store a copy with the current origin and signon
       // realm. This ensures that on the next visit, a precise match is found.
@@ -647,10 +643,6 @@
       // If this isn't updated, then password generation uploads are off for
       // sites where PSL matching is required to fill the login form, as two
       // PASSWORD votes are uploaded per saved password instead of one.
-      //
-      // TODO(gcasto): It would be nice if other state were shared such that if
-      // say a password was updated on one match it would update on all related
-      // passwords. This is a much larger change.
       UpdateMetadataForUsage(&pending_credentials_);
 
       // Update |pending_credentials_| in order to be able correctly save it.
@@ -667,8 +659,7 @@
       // actually correspond to two different accounts (see
       // http://crbug.com/385619). In that case the user should be asked again
       // before saving the password. This is ensured by setting
-      // |password_overriden_| on |pending_credentials_| to false and setting
-      // |origin| and |signon_realm| to correct values.
+      // |password_overriden_| on |pending_credentials_| to false.
       //
       // There is still the edge case when the autofilled credentials represent
       // the same account as |submitted_form_| but the stored password
@@ -746,14 +737,8 @@
     }
   }
 
-  if (!IsValidAndroidFacetURI(pending_credentials_.signon_realm)) {
+  if (!IsValidAndroidFacetURI(pending_credentials_.signon_realm))
     pending_credentials_.action = submitted_form_->action;
-    // If the user selected credentials we autofilled from a PasswordForm
-    // that contained no action URL (IE6/7 imported passwords, for example),
-    // bless it with the action URL from the observed form. See b/1107719.
-    if (pending_credentials_.action.is_empty())
-      pending_credentials_.action = observed_form_.action;
-  }
 
   pending_credentials_.password_value = password_to_save.first;
   pending_credentials_.preferred = submitted_form_->preferred;
@@ -771,7 +756,9 @@
     pending_credentials_.display_name = submitted_form_->display_name;
     pending_credentials_.federation_origin = submitted_form_->federation_origin;
     pending_credentials_.icon_url = submitted_form_->icon_url;
-    // Take the correct signon_realm for federated credentials.
+    // It's important to override |signon_realm| for federated credentials
+    // because it has format "federation://" + origin_host + "/" +
+    // federation_host
     pending_credentials_.signon_realm = submitted_form_->signon_realm;
   }
 
diff --git a/components/password_manager/core/browser/password_form_manager.h b/components/password_manager/core/browser/password_form_manager.h
index 547c462..339b764 100644
--- a/components/password_manager/core/browser/password_form_manager.h
+++ b/components/password_manager/core/browser/password_form_manager.h
@@ -437,10 +437,6 @@
 
   VotesUploader votes_uploader_;
 
-  // True if the main frame's visible URL, at the time this PasswordFormManager
-  // was created, is secure.
-  bool is_main_frame_secure_ = false;
-
   // Takes care of recording metrics and events for this PasswordFormManager.
   // Make sure to call Init before using |*this|, to ensure it is not null.
   scoped_refptr<PasswordFormMetricsRecorder> metrics_recorder_;
diff --git a/components/password_manager/core/common/password_manager_features.cc b/components/password_manager/core/common/password_manager_features.cc
index 437840a..7322313 100644
--- a/components/password_manager/core/common/password_manager_features.cc
+++ b/components/password_manager/core/common/password_manager_features.cc
@@ -71,11 +71,17 @@
 // selection, rather than autofilling on page load, with highlighting of fields.
 const base::Feature kFillOnAccountSelect = {"fill-on-account-select",
                                             base::FEATURE_DISABLED_BY_DEFAULT};
-// Enables new password form parsing mechanism, details in
-// go/new-cpm-design-refactoring.
+
+// Enables new password form parsing mechanism for filling passwords, details in
+// https://goo.gl/QodPH1
 const base::Feature kNewPasswordFormParsing = {
     "new-password-form-parsing", base::FEATURE_DISABLED_BY_DEFAULT};
 
+// Enables new password form parsing mechanism for saving passwords, details in
+// https://goo.gl/QodPH1
+const base::Feature kNewPasswordFormParsingForSaving = {
+    "new-password-form-parsing-for-saving", base::FEATURE_DISABLED_BY_DEFAULT};
+
 // Field trial identifier for password generation requirements.
 const char* kGenerationRequirementsFieldTrial =
     "PasswordGenerationRequirements";
diff --git a/components/password_manager/core/common/password_manager_features.h b/components/password_manager/core/common/password_manager_features.h
index cc59717..7eb1bf2 100644
--- a/components/password_manager/core/common/password_manager_features.h
+++ b/components/password_manager/core/common/password_manager_features.h
@@ -26,6 +26,7 @@
 extern const base::Feature kShowAllSavedPasswordsContextMenu;
 extern const base::Feature kFillOnAccountSelect;
 extern const base::Feature kNewPasswordFormParsing;
+extern const base::Feature kNewPasswordFormParsingForSaving;
 extern const base::Feature kPasswordExport;
 extern const base::Feature kPasswordImport;
 extern const base::Feature kPasswordSearchMobile;
diff --git a/components/policy/resources/policy_templates.json b/components/policy/resources/policy_templates.json
index c147691..b31014e6 100644
--- a/components/policy/resources/policy_templates.json
+++ b/components/policy/resources/policy_templates.json
@@ -11651,7 +11651,7 @@
         'dynamic_refresh': True,
         'per_profile': False,
       },
-      'supported_on': ['chrome.*:66-'],
+      'supported_on': ['chrome.*:66-', 'chrome_os:70-'],
       'type': 'int-enum',
       'schema': {
         'type': 'integer',
diff --git a/components/sync/BUILD.gn b/components/sync/BUILD.gn
index 24152e3..1766d58 100644
--- a/components/sync/BUILD.gn
+++ b/components/sync/BUILD.gn
@@ -858,6 +858,7 @@
     "engine/ui_model_worker_unittest.cc",
     "engine_impl/apply_control_data_updates_unittest.cc",
     "engine_impl/backoff_delay_provider_unittest.cc",
+    "engine_impl/cycle/data_type_debug_info_emitter_unittest.cc",
     "engine_impl/cycle/nudge_tracker_unittest.cc",
     "engine_impl/cycle/status_controller_unittest.cc",
     "engine_impl/debug_info_event_listener_unittest.cc",
diff --git a/components/sync/PRESUBMIT.py b/components/sync/PRESUBMIT.py
index bca86cb..0d138565 100644
--- a/components/sync/PRESUBMIT.py
+++ b/components/sync/PRESUBMIT.py
@@ -48,7 +48,7 @@
 
 # Start and end regexes for finding the EntitySpecifics definition in
 # sync.proto.
-PROTO_DEFINITION_START_PATTERN = '^message EntitySpecifics'
+PROTO_DEFINITION_START_PATTERN = '^  oneof specifics_variant \{'
 PROTO_DEFINITION_END_PATTERN = '^\}'
 
 # Start and end regexes for finding the ModelTypeInfoMap definition
@@ -60,6 +60,7 @@
 # model_type.cc is where the ModelTypeInfoMap is
 # sync.proto is where the proto definitions for ModelTypes are.
 PROTO_FILE_PATH = './protocol/sync.proto'
+PROTO_FILE_NAME = 'sync.proto'
 MODEL_TYPE_FILE_NAME = 'model_type.cc'
 
 SYNC_SOURCE_FILES = (r'^components[\\/]sync[\\/].*\.(cc|h)$',)
@@ -214,12 +215,11 @@
       if '//' in line or len(split_proto_line) < 3:
         continue
 
-      field_typename = split_proto_line[1]
-      field_identifier = split_proto_line[2]
+      field_typename = split_proto_line[0]
+      field_identifier = split_proto_line[1]
       proto_field_definitions[field_typename] = field_identifier
   return proto_field_definitions
 
-
 def StripTrailingS(string):
   return string.rstrip('sS')
 
@@ -384,8 +384,9 @@
   results = []
   results += CheckChangeLintsClean(input_api, output_api)
   for f in input_api.AffectedFiles():
-    if f.LocalPath().endswith(MODEL_TYPE_FILE_NAME):
-      return CheckModelTypeInfoMap(input_api, output_api, f)
+    if (f.LocalPath().endswith(MODEL_TYPE_FILE_NAME) or
+        f.LocalPath().endswith(PROTO_FILE_NAME)):
+      results += CheckModelTypeInfoMap(input_api, output_api, f)
   return results
 
 def CheckChangeOnUpload(input_api, output_api):
diff --git a/components/sync/PRESUBMIT_test.py b/components/sync/PRESUBMIT_test.py
index 6164a18a..df563a0 100755
--- a/components/sync/PRESUBMIT_test.py
+++ b/components/sync/PRESUBMIT_test.py
@@ -70,16 +70,19 @@
 # test presubmit parsing of EntitySpecifics definition in that file.
 MOCK_PROTOFILE_CONTENTS = ('\n'
   'message EntitySpecifics {\n'
-  '//comment\n'
+  '  //comment\n'
   '\n'
-  'optional AutofillSpecifics autofill = 123;\n'
-  'optional AppSpecifics app = 456;\n'
-  'optional AppSettingSpecifics app_setting = 789;\n'
-  'optional ExtensionSettingSpecifics extension_setting = 910;\n'
-  'optional ManagedUserSharedSettingSpecifics managed_user_shared_setting'
+  '  oneof specifics_variant {\n'
+  '    AutofillSpecifics autofill = 123;\n'
+  '    AppSpecifics app = 456;\n'
+  '    AppSettingSpecifics app_setting = 789;\n'
+  '    ExtensionSettingSpecifics extension_setting = 910;\n'
+  '    ManagedUserSharedSettingSpecifics managed_user_shared_setting'
                                                                     ' = 915;\n'
-  '//comment\n'
-  '}\n')
+  '    //comment\n'
+  '  }\n'
+  '}\n'
+  )
 
 
 # Format string used as the contents of a mock model_type.cc
diff --git a/components/sync/base/data_type_histogram.cc b/components/sync/base/data_type_histogram.cc
index 18546e4..42b8b34a 100644
--- a/components/sync/base/data_type_histogram.cc
+++ b/components/sync/base/data_type_histogram.cc
@@ -11,9 +11,7 @@
 void SyncRecordMemoryKbHistogram(const std::string& histogram_name_prefix,
                                  syncer::ModelType model_type,
                                  size_t value) {
-  std::string type_string;
-  if (RealModelTypeToNotificationType(model_type, &type_string)) {
-    std::string full_histogram_name = histogram_name_prefix + type_string;
-    base::UmaHistogramCounts1M(full_histogram_name, value / 1024);
-  }
+  std::string type_string = ModelTypeToHistogramSuffix(model_type);
+  std::string full_histogram_name = histogram_name_prefix + type_string;
+  base::UmaHistogramCounts1M(full_histogram_name, value / 1024);
 }
diff --git a/components/sync/base/model_type.h b/components/sync/base/model_type.h
index 7903bba9..ea9a153 100644
--- a/components/sync/base/model_type.h
+++ b/components/sync/base/model_type.h
@@ -331,10 +331,14 @@
 
 // TODO(sync): The functions below badly need some cleanup.
 
-// Returns a pointer to a string with application lifetime that represents
-// the name of |model_type|.
+// Returns a string with application lifetime that represents the name of
+// |model_type|.
 const char* ModelTypeToString(ModelType model_type);
 
+// Returns a string with application lifetime that is used as the histogram
+// suffix for |model_type|.
+const char* ModelTypeToHistogramSuffix(ModelType model_type);
+
 // Some histograms take an integer parameter that represents a model type.
 // The mapping from ModelType to integer is defined here. It should match the
 // mapping from integer to labels defined in histograms.xml.
diff --git a/components/sync/driver/sync_driver_switches.cc b/components/sync/driver/sync_driver_switches.cc
index e409510..697cd0f 100644
--- a/components/sync/driver/sync_driver_switches.cc
+++ b/components/sync/driver/sync_driver_switches.cc
@@ -74,4 +74,8 @@
 const base::Feature kSyncUSSAutofillProfile{"SyncUSSAutofillProfile",
                                             base::FEATURE_DISABLED_BY_DEFAULT};
 
+// Enable USS implementation of autofill wallet datatype.
+const base::Feature kSyncUSSAutofillWalletData{
+    "SyncUSSAutofillWalletData", base::FEATURE_DISABLED_BY_DEFAULT};
+
 }  // namespace switches
diff --git a/components/sync/driver/sync_driver_switches.h b/components/sync/driver/sync_driver_switches.h
index 77c3e17..b01df2b2 100644
--- a/components/sync/driver/sync_driver_switches.h
+++ b/components/sync/driver/sync_driver_switches.h
@@ -29,6 +29,7 @@
 extern const base::Feature kSyncUSSBookmarks;
 extern const base::Feature kSyncUSSSessions;
 extern const base::Feature kSyncUSSAutofillProfile;
+extern const base::Feature kSyncUSSAutofillWalletData;
 
 }  // namespace switches
 
diff --git a/components/sync/driver/sync_service.cc b/components/sync/driver/sync_service.cc
index 2ec82a3..e04efc7 100644
--- a/components/sync/driver/sync_service.cc
+++ b/components/sync/driver/sync_service.cc
@@ -17,11 +17,6 @@
   return GetDisableReasons() == DISABLE_REASON_NONE;
 }
 
-bool SyncService::IsSyncAllowed() const {
-  return !HasDisableReason(DISABLE_REASON_PLATFORM_OVERRIDE) &&
-         !HasDisableReason(DISABLE_REASON_ENTERPRISE_POLICY);
-}
-
 bool SyncService::IsEngineInitialized() const {
   switch (GetState()) {
     case State::DISABLED:
diff --git a/components/sync/driver/sync_service.h b/components/sync/driver/sync_service.h
index d7922988..3d270305 100644
--- a/components/sync/driver/sync_service.h
+++ b/components/sync/driver/sync_service.h
@@ -187,11 +187,6 @@
   bool IsEngineInitialized() const;
 
   // DEPRECATED! Use GetDisableReasons/HasDisableReason instead.
-  // Equivalent to "!HasDisableReason(DISABLE_REASON_PLATFORM_OVERRIDE) &&
-  // !HasDisableReason(DISABLE_REASON_ENTERPRISE_POLICY)".
-  bool IsSyncAllowed() const;
-
-  // DEPRECATED! Use GetDisableReasons/HasDisableReason instead.
   // Equivalent to having no disable reasons, i.e.
   // "GetDisableReasons() == DISABLE_REASON_NONE".
   bool CanSyncStart() const;
diff --git a/components/sync/engine_impl/cycle/data_type_debug_info_emitter.cc b/components/sync/engine_impl/cycle/data_type_debug_info_emitter.cc
index 68c3a17..93bc48b 100644
--- a/components/sync/engine_impl/cycle/data_type_debug_info_emitter.cc
+++ b/components/sync/engine_impl/cycle/data_type_debug_info_emitter.cc
@@ -4,14 +4,64 @@
 
 #include "components/sync/engine_impl/cycle/data_type_debug_info_emitter.h"
 
+#include <string>
+
+#include "base/metrics/histogram.h"
 #include "components/sync/engine/cycle/type_debug_info_observer.h"
 
 namespace syncer {
 
+namespace {
+
+const char kModelTypeEntityChangeHistogramPrefix[] =
+    "Sync.ModelTypeEntityChange.";
+
+// Values corrospond to a UMA histogram, do not modify, or delete any values.
+// Add new values only directly before COUNT.
+enum ModelTypeEntityChange {
+  LOCAL_DELETION = 0,
+  LOCAL_CREATION = 1,
+  LOCAL_UPDATE = 2,
+  REMOTE_DELETION = 3,
+  REMOTE_UPDATE = 4,
+  MODEL_TYPE_ENTITY_CHANGE_COUNT = 5
+};
+
+void EmitNewChangesToUma(int count,
+                         ModelTypeEntityChange bucket,
+                         base::HistogramBase* histogram) {
+  DCHECK_GE(count, 0);
+  if (count > 0) {
+    histogram->AddCount(bucket, count);
+  }
+}
+
+// We'll add many values in this histogram. Since the name of the histogram is
+// not static here, we cannot use the (efficient) macros that use caching of the
+// histogram object. The helper functions like UmaHistogramEnumeration are not
+// efficient and thus we need re-implement the code here, caching the resulting
+// histogram object.
+base::HistogramBase* GetModelTypeEntityChangeHistogram(ModelType type) {
+  std::string type_string = ModelTypeToHistogramSuffix(type);
+  std::string full_histogram_name =
+      kModelTypeEntityChangeHistogramPrefix + type_string;
+  return base::LinearHistogram::FactoryGet(
+      /*name=*/full_histogram_name, /*minimum=*/1,
+      /*maximum=*/MODEL_TYPE_ENTITY_CHANGE_COUNT,
+      /*bucket_count=*/MODEL_TYPE_ENTITY_CHANGE_COUNT + 1,
+      /*flagz=*/base::HistogramBase::kUmaTargetedHistogramFlag);
+}
+
+}  // namespace
+
 DataTypeDebugInfoEmitter::DataTypeDebugInfoEmitter(
     ModelType type,
     base::ObserverList<TypeDebugInfoObserver>* observers)
-    : type_(type), type_debug_info_observers_(observers) {}
+    : type_(type),
+      type_debug_info_observers_(observers),
+      histogram_(GetModelTypeEntityChangeHistogram(type)) {
+  DCHECK(histogram_);
+}
 
 DataTypeDebugInfoEmitter::~DataTypeDebugInfoEmitter() {}
 
@@ -26,6 +76,23 @@
 void DataTypeDebugInfoEmitter::EmitCommitCountersUpdate() {
   for (auto& observer : *type_debug_info_observers_)
     observer.OnCommitCountersUpdated(type_, commit_counters_);
+
+  // Emit the newly added counts to UMA.
+  EmitNewChangesToUma(
+      /*count=*/commit_counters_.num_creation_commits_attempted -
+          emitted_commit_counters_.num_creation_commits_attempted,
+      /*bucket=*/LOCAL_CREATION, histogram_);
+  EmitNewChangesToUma(
+      /*count=*/commit_counters_.num_deletion_commits_attempted -
+          emitted_commit_counters_.num_deletion_commits_attempted,
+      /*bucket=*/LOCAL_DELETION, histogram_);
+  EmitNewChangesToUma(
+      /*count=*/commit_counters_.num_update_commits_attempted -
+          emitted_commit_counters_.num_update_commits_attempted,
+      /*bucket=*/LOCAL_UPDATE, histogram_);
+
+  // Mark the current state of the counters as uploaded to UMA.
+  emitted_commit_counters_ = commit_counters_;
 }
 
 const UpdateCounters& DataTypeDebugInfoEmitter::GetUpdateCounters() const {
@@ -39,6 +106,27 @@
 void DataTypeDebugInfoEmitter::EmitUpdateCountersUpdate() {
   for (auto& observer : *type_debug_info_observers_)
     observer.OnUpdateCountersUpdated(type_, update_counters_);
+
+  // Emit the newly added counts to UMA.
+  EmitNewChangesToUma(
+      /*count=*/update_counters_.num_tombstone_updates_received -
+          emitted_update_counters_.num_tombstone_updates_received,
+      /*bucket=*/REMOTE_DELETION, histogram_);
+  // The REMOTE_UPDATE type is not explicitly stored, we need to compute it as a
+  // diff of (all - deletions).
+  int emitted_remote_updates_count =
+      emitted_update_counters_.num_updates_received -
+      emitted_update_counters_.num_tombstone_updates_received;
+  int remote_updates_count = update_counters_.num_updates_received -
+                             update_counters_.num_tombstone_updates_received;
+  EmitNewChangesToUma(
+      /*count=*/remote_updates_count - emitted_remote_updates_count,
+      /*bucket=*/REMOTE_UPDATE, histogram_);
+
+  // Mark the current state of the counters as uploaded to UMA.
+  emitted_update_counters_ = update_counters_;
 }
 
+void DataTypeDebugInfoEmitter::EmitStatusCountersUpdate() {}
+
 }  // namespace syncer
diff --git a/components/sync/engine_impl/cycle/data_type_debug_info_emitter.h b/components/sync/engine_impl/cycle/data_type_debug_info_emitter.h
index e49be1b6..2aac422 100644
--- a/components/sync/engine_impl/cycle/data_type_debug_info_emitter.h
+++ b/components/sync/engine_impl/cycle/data_type_debug_info_emitter.h
@@ -13,6 +13,10 @@
 #include "components/sync/engine/cycle/commit_counters.h"
 #include "components/sync/engine/cycle/update_counters.h"
 
+namespace base {
+class HistogramBase;
+}
+
 namespace syncer {
 
 class TypeDebugInfoObserver;
@@ -55,8 +59,10 @@
   // Triggers an update counters update to registered observers.
   void EmitUpdateCountersUpdate();
 
-  // Triggers a status counters update to registered observers.
-  virtual void EmitStatusCountersUpdate() = 0;
+  // Triggers a status counters update to registered observers. The default
+  // implementation does nothing and is present only to make this class
+  // non-abstract and thus unit-testable.
+  virtual void EmitStatusCountersUpdate();
 
  protected:
   const ModelType type_;
@@ -68,9 +74,20 @@
   base::ObserverList<TypeDebugInfoObserver>* type_debug_info_observers_;
 
  private:
+  // The actual up-to-date counters.
   CommitCounters commit_counters_;
   UpdateCounters update_counters_;
 
+  // The last state of the counters emitted to UMA. In the next round of
+  // emitting to UMA, we only need to upload the diff between the actual
+  // counters and the counts here.
+  CommitCounters emitted_commit_counters_;
+  UpdateCounters emitted_update_counters_;
+
+  // The histogram to record to; cached for efficiency because many histogram
+  // entries are recorded in this object during run-time.
+  base::HistogramBase* const histogram_;
+
   DISALLOW_COPY_AND_ASSIGN(DataTypeDebugInfoEmitter);
 };
 
diff --git a/components/sync/engine_impl/cycle/data_type_debug_info_emitter_unittest.cc b/components/sync/engine_impl/cycle/data_type_debug_info_emitter_unittest.cc
new file mode 100644
index 0000000..68cc6f7
--- /dev/null
+++ b/components/sync/engine_impl/cycle/data_type_debug_info_emitter_unittest.cc
@@ -0,0 +1,131 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "components/sync/engine_impl/cycle/data_type_debug_info_emitter.h"
+
+#include "base/test/metrics/histogram_tester.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+namespace syncer {
+namespace {
+
+TEST(DataTypeDebugInfoEmitterTest, ShouldEmitCommitsToUMAIfChanged) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  CommitCounters* counters = emitter.GetMutableCommitCounters();
+  counters->num_deletion_commits_attempted += 3;
+  counters->num_creation_commits_attempted += 2;
+  counters->num_update_commits_attempted += 1;
+
+  base::HistogramTester histogram_tester;
+  emitter.EmitCommitCountersUpdate();
+  EXPECT_EQ(
+      3, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*LOCAL_DELETION=*/0));
+  EXPECT_EQ(
+      2, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*LOCAL_CREATION=*/1));
+  EXPECT_EQ(1, histogram_tester.GetBucketCount(
+                   "Sync.ModelTypeEntityChange.BOOKMARK", /*LOCAL_UPDATE=*/2));
+}
+
+TEST(DataTypeDebugInfoEmitterTest, ShouldNotEmitCommitsToUMAIfNotChanged) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  base::HistogramTester histogram_tester;
+  emitter.EmitCommitCountersUpdate();
+  histogram_tester.ExpectTotalCount("Sync.ModelTypeEntityChange.BOOKMARK", 0);
+}
+
+// Tests that at each EmitCommitCountersUpdate() call, only the changes since
+// the last call to EmitCommitCountersUpdate() are reported to UMA.
+TEST(DataTypeDebugInfoEmitterTest, ShouldEmitCommitsToUMAIncrementally) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  CommitCounters* counters = emitter.GetMutableCommitCounters();
+  counters->num_deletion_commits_attempted += 3;
+  counters->num_creation_commits_attempted += 2;
+  counters->num_update_commits_attempted += 1;
+
+  // First emission - tested in the test above.
+  emitter.EmitCommitCountersUpdate();
+
+  counters = emitter.GetMutableCommitCounters();
+  counters->num_deletion_commits_attempted += 1;
+  counters->num_creation_commits_attempted += 2;
+  counters->num_update_commits_attempted += 3;
+
+  // Test the second emission that it only reports the increment in counters.
+  base::HistogramTester histogram_tester;
+  emitter.EmitCommitCountersUpdate();
+  EXPECT_EQ(
+      1, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*LOCAL_DELETION=*/0));
+  EXPECT_EQ(
+      2, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*LOCAL_CREATION=*/1));
+  EXPECT_EQ(3, histogram_tester.GetBucketCount(
+                   "Sync.ModelTypeEntityChange.BOOKMARK", /*LOCAL_UPDATE=*/2));
+}
+
+TEST(DataTypeDebugInfoEmitterTest, ShouldEmitUpdatesToUMAIfChanged) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  UpdateCounters* counters = emitter.GetMutableUpdateCounters();
+  counters->num_updates_received += 3;
+  counters->num_tombstone_updates_received += 1;
+
+  base::HistogramTester histogram_tester;
+  emitter.EmitUpdateCountersUpdate();
+  EXPECT_EQ(
+      1, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*REMOTE_DELETION=*/3));
+  EXPECT_EQ(
+      2, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*REMOTE_UPDATE=*/4));
+}
+
+TEST(DataTypeDebugInfoEmitterTest, ShouldNotEmitUpdatesToUMAIfNotChanged) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  base::HistogramTester histogram_tester;
+  emitter.EmitUpdateCountersUpdate();
+  histogram_tester.ExpectTotalCount("Sync.ModelTypeEntityChange.BOOKMARK", 0);
+}
+
+// Tests that at each EmitUpdateCountersUpdate() call, only the changes since
+// the last call to EmitUpdateCountersUpdate() are reported to UMA.
+TEST(DataTypeDebugInfoEmitterTest, ShouldEmitUpdatesToUMAIncrementally) {
+  base::ObserverList<TypeDebugInfoObserver> observers;
+  DataTypeDebugInfoEmitter emitter(BOOKMARKS, &observers);
+
+  UpdateCounters* counters = emitter.GetMutableUpdateCounters();
+  counters->num_updates_received += 3;
+  counters->num_tombstone_updates_received += 1;
+
+  // First emission - tested in the test above.
+  emitter.EmitUpdateCountersUpdate();
+
+  counters = emitter.GetMutableUpdateCounters();
+  counters->num_updates_received += 3;
+  counters->num_tombstone_updates_received += 2;
+
+  // Test the second emission that it only reports the increment in counters.
+  base::HistogramTester histogram_tester;
+  emitter.EmitUpdateCountersUpdate();
+  EXPECT_EQ(
+      2, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*REMOTE_DELETION=*/3));
+  EXPECT_EQ(
+      1, histogram_tester.GetBucketCount("Sync.ModelTypeEntityChange.BOOKMARK",
+                                         /*REMOTE_UPDATE=*/4));
+}
+
+}  // namespace
+}  // namespace syncer
diff --git a/components/sync/syncable/directory_backing_store.cc b/components/sync/syncable/directory_backing_store.cc
index 6b681c4b..8c97f4b 100644
--- a/components/sync/syncable/directory_backing_store.cc
+++ b/components/sync/syncable/directory_backing_store.cc
@@ -280,16 +280,14 @@
                                const int (&entries_counts)[MODEL_TYPE_COUNT]) {
   int total_entry_counts = 0;
   for (int i = FIRST_REAL_MODEL_TYPE; i < MODEL_TYPE_COUNT; ++i) {
-    std::string model_type;
-    if (RealModelTypeToNotificationType((ModelType)i, &model_type)) {
-      std::string full_histogram_name = "Sync.ModelTypeCount." + model_type;
-      base::HistogramBase* histogram = base::Histogram::FactoryGet(
-          full_histogram_name, 1, 1000000, 50,
-          base::HistogramBase::kUmaTargetedHistogramFlag);
-      if (histogram)
-        histogram->Add(entries_counts[i]);
-      total_entry_counts += entries_counts[i];
-    }
+    std::string model_type = ModelTypeToHistogramSuffix((ModelType)i);
+    std::string full_histogram_name = "Sync.ModelTypeCount." + model_type;
+    base::HistogramBase* histogram = base::Histogram::FactoryGet(
+        full_histogram_name, 1, 1000000, 50,
+        base::HistogramBase::kUmaTargetedHistogramFlag);
+    if (histogram)
+      histogram->Add(entries_counts[i]);
+    total_entry_counts += entries_counts[i];
   }
   UMA_HISTOGRAM_COUNTS("Sync.ModelTypeCount", total_entry_counts);
   UMA_HISTOGRAM_COUNTS("Sync.ExtraSyncDataCount",
diff --git a/components/sync/syncable/model_type.cc b/components/sync/syncable/model_type.cc
index b3b23a6..68ec049 100644
--- a/components/sync/syncable/model_type.cc
+++ b/components/sync/syncable/model_type.cc
@@ -32,7 +32,9 @@
 struct ModelTypeInfo {
   ModelType model_type;
   // Model Type notification string.
-  // This needs to match the corresponding proto message name in sync.proto
+  // This needs to match the corresponding proto message name in sync.proto. It
+  // is also used to identify the model type in the SyncModelType
+  // histogram_suffix in histograms.xml. Must always be kept in sync.
   const char* notification_type;
   // Root tag for Model Type
   // This should be the same as the model type but all lowercase.
@@ -49,8 +51,10 @@
 };
 
 // Below struct entries are in the same order as their definition in the
-// ModelType enum. Don't forget to update the ModelType enum when you make
-// changes to this list.
+// ModelType enum. When making changes to this list, don't forget to
+//  - update the ModelType enum,
+//  - update the SyncModelTypes enum in enums.xml, and
+//  - update the SyncModelType histogram suffix in histograms.xml.
 // Struct field values should be unique across the entire map.
 const ModelTypeInfo kModelTypeInfoMap[] = {
     {UNSPECIFIED, "", "", "Unspecified", -1, 0},
@@ -493,6 +497,17 @@
   return "Invalid";
 }
 
+const char* ModelTypeToHistogramSuffix(ModelType model_type) {
+  if (model_type >= UNSPECIFIED && model_type < MODEL_TYPE_COUNT) {
+    // We use the same string that is used for notification types because they
+    // satisfy all we need (being stable and explanatory).
+    return kModelTypeInfoMap[model_type].notification_type;
+  }
+  NOTREACHED() << "No known suffix for model type "
+               << static_cast<int>(model_type) << ".";
+  return "Invalid";
+}
+
 // The normal rules about histograms apply here.  Always append to the bottom of
 // the list, and be careful to not reuse integer values that have already been
 // assigned.
diff --git a/components/sync/user_events/user_event_sync_bridge.cc b/components/sync/user_events/user_event_sync_bridge.cc
index 6b6041d..846356c 100644
--- a/components/sync/user_events/user_event_sync_bridge.cc
+++ b/components/sync/user_events/user_event_sync_bridge.cc
@@ -59,15 +59,6 @@
   return entity_data;
 }
 
-std::unique_ptr<EntityData> CopyToEntityData(
-    const UserEventSpecifics specifics) {
-  auto entity_data = std::make_unique<EntityData>();
-  entity_data->non_unique_name =
-      base::Int64ToString(specifics.event_time_usec());
-  *entity_data->specifics.mutable_user_event() = specifics;
-  return entity_data;
-}
-
 }  // namespace
 
 UserEventSyncBridge::UserEventSyncBridge(
@@ -358,11 +349,12 @@
   }
 
   auto batch = std::make_unique<MutableDataBatch>();
-  UserEventSpecifics specifics;
   for (const Record& r : *data_records) {
-    if (specifics.ParseFromString(r.value)) {
-      DCHECK_EQ(r.id, GetStorageKeyFromSpecifics(specifics));
-      batch->Put(r.id, CopyToEntityData(specifics));
+    auto specifics = std::make_unique<UserEventSpecifics>();
+
+    if (specifics->ParseFromString(r.value)) {
+      DCHECK_EQ(r.id, GetStorageKeyFromSpecifics(*specifics));
+      batch->Put(r.id, MoveToEntityData(std::move(specifics)));
     } else {
       change_processor()->ReportError(
           {FROM_HERE, "Failed deserializing user events."});
diff --git a/components/test/data/ntp/render/1200x800_DefaultMV.png b/components/test/data/ntp/render/1200x800_DefaultMV.png
index 958b314..f453798 100644
--- a/components/test/data/ntp/render/1200x800_DefaultMV.png
+++ b/components/test/data/ntp/render/1200x800_DefaultMV.png
Binary files differ
diff --git a/components/ukm/debug/BUILD.gn b/components/ukm/debug/BUILD.gn
index f40504a..e4f6781 100644
--- a/components/ukm/debug/BUILD.gn
+++ b/components/ukm/debug/BUILD.gn
@@ -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("//third_party/closure_compiler/compile_js.gni")
+
 source_set("util") {
   sources = [
     "ukm_debug_data_extractor.cc",
@@ -13,3 +15,16 @@
     "//services/metrics/public/cpp:ukm_builders",
   ]
 }
+
+js_type_check("closure_compile") {
+  deps = [
+    ":ukm_internals",
+  ]
+}
+
+js_library("ukm_internals") {
+  deps = [
+    "//ui/webui/resources/js:cr",
+    "//ui/webui/resources/js:util",
+  ]
+}
diff --git a/components/ukm/debug/ukm_debug_data_extractor.cc b/components/ukm/debug/ukm_debug_data_extractor.cc
index fc768528..89fd400e 100644
--- a/components/ukm/debug/ukm_debug_data_extractor.cc
+++ b/components/ukm/debug/ukm_debug_data_extractor.cc
@@ -114,74 +114,5 @@
   return std::move(ukm_data);
 }
 
-// static
-std::string UkmDebugDataExtractor::GetHTMLData(const UkmService* ukm_service) {
-  std::string output;
-  output.append(R"""(<!DOCTYPE html>
-  <html>
-    <head>
-      <meta http-equiv="Content-Security-Policy"
-            content="object-src 'none'; script-src 'none'">
-      <title>UKM Debug</title>
-    </head>
-    <body>
-      <h1>UKM Debug page</h1>
-  )""");
-
-  if (ukm_service) {
-    output.append(
-        // 'id' attribute set so tests can extract this element.
-        base::StringPrintf("<p>IsEnabled:<span id='state'>%s</span></p>",
-                           ukm_service->recording_enabled_ ? "True" : "False"));
-    output.append(base::StringPrintf("<p>ClientId:<span id='clientid'>%" PRIu64
-                                     "</span></p>",
-                                     ukm_service->client_id_));
-    output.append(
-        base::StringPrintf("<p>SessionId:%d</p>", ukm_service->session_id_));
-
-    const auto& decode_map = ukm_service->decode_map_;
-    std::map<SourceId, SourceData> source_data;
-    for (const auto& kv : ukm_service->recordings_.sources) {
-      source_data[kv.first].source = kv.second.get();
-    }
-
-    for (const auto& v : ukm_service->recordings_.entries) {
-      source_data[v->source_id].entries.push_back(v.get());
-    }
-
-    output.append("<h2>Sources</h2>");
-    for (const auto& kv : source_data) {
-      const auto* src = kv.second.source;
-      if (src) {
-        output.append(base::StringPrintf("<h3>Id:%" PRId64 " Url:%s</h3>",
-                                         src->id(), src->url().spec().c_str()));
-      } else {
-        output.append(base::StringPrintf("<h3>Id:%" PRId64 "</h3>", kv.first));
-      }
-      for (auto* entry : kv.second.entries) {
-        const auto it = decode_map.find(entry->event_hash);
-        if (it == decode_map.end()) {
-          output.append(base::StringPrintf(
-              "<h4>Entry: Unknown %" PRIu64 "</h4>", entry->event_hash));
-          continue;
-        }
-        output.append(base::StringPrintf("<h4>Entry:%s</h4>", it->second.name));
-        for (const auto& metric : entry->metrics) {
-          output.append(base::StringPrintf(
-              "<h5>Metric:%s Value:%" PRId64 "</h5>",
-              GetName(it->second, metric.first).c_str(), metric.second));
-        }
-      }
-    }
-  }
-
-  output.append(R"""(
-    </body>
-  </html>
-  )""");
-
-  return output;
-}
-
 }  // namespace debug
 }  // namespace ukm
diff --git a/components/ukm/debug/ukm_debug_data_extractor.h b/components/ukm/debug/ukm_debug_data_extractor.h
index 0a55cdf..dc2810d 100644
--- a/components/ukm/debug/ukm_debug_data_extractor.h
+++ b/components/ukm/debug/ukm_debug_data_extractor.h
@@ -26,10 +26,6 @@
   // Returns UKM data structured in a DictionaryValue.
   static base::Value GetStructuredData(const UkmService* ukm_service);
 
-  // Returns UKM data as an HTML page.
-  // TODO(etiennep): Use GetStructuredData() instead.
-  static std::string GetHTMLData(const UkmService* ukm_service);
-
  private:
   DISALLOW_COPY_AND_ASSIGN(UkmDebugDataExtractor);
 };
diff --git a/chrome/browser/resources/ukm/ukm_internals.html b/components/ukm/debug/ukm_internals.html
similarity index 100%
rename from chrome/browser/resources/ukm/ukm_internals.html
rename to components/ukm/debug/ukm_internals.html
diff --git a/components/ukm/debug/ukm_internals.js b/components/ukm/debug/ukm_internals.js
new file mode 100644
index 0000000..85c1c7a
--- /dev/null
+++ b/components/ukm/debug/ukm_internals.js
@@ -0,0 +1,78 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+/**
+ * @typedef {{
+ *   name: string,
+ *   value: string
+ * }}
+ */
+var Metric;
+
+/**
+ * @typedef {{
+ *   name: string,
+ *   metrics: !Array<!Metric>
+ * }}
+ */
+var UkmEntry;
+
+/**
+ * @typedef {{
+ *   url: string,
+ *   id: string,
+ *   entries: !Array<UkmEntry>,
+ * }}
+ */
+var UkmDataSource;
+
+/**
+ * The Ukm data sent from the browser.
+ * @typedef {{
+ *   state: boolean,
+ *   client_id: string,
+ *   session_id: string,
+ *   sources: !Array<!UkmDataSource>,
+ * }}
+ */
+var UkmData;
+
+/**
+ * Fetches data from the Ukm service and updates the DOM to display it as a
+ * list.
+ */
+function updateUkmData() {
+  cr.sendWithPromise('requestUkmData').then((/** @type {UkmData} */ data) => {
+    $('state').innerText = data.state ? 'True' : 'False';
+    $('clientid').innerText = data.client_id;
+    $('sessionid').innerText = data.session_id;
+
+    let sourceDiv = $('sources');
+    for (const source of data.sources) {
+      const sourceElement = document.createElement('h3');
+      if (source.url !== undefined)
+        sourceElement.innerText = `Id: ${source.id} Url: ${source.url}`;
+      else
+        sourceElement.innerText = `Id: ${source.id}`;
+      sourceDiv.appendChild(sourceElement);
+
+      for (const entry of source.entries) {
+        const entryElement = document.createElement('h4');
+        entryElement.innerText = `Entry: ${entry.name}`;
+        sourceDiv.appendChild(entryElement);
+
+        if (entry.metrics === undefined)
+          continue;
+        for (const metric of entry.metrics) {
+          const metricElement = document.createElement('h5');
+          metricElement.innerText =
+              `Metric: ${metric.name} Value: ${metric.value}`;
+          sourceDiv.appendChild(metricElement);
+        }
+      }
+    }
+  });
+}
+
+updateUkmData();
diff --git a/components/unified_consent/unified_consent_service.cc b/components/unified_consent/unified_consent_service.cc
index 4747e0d..34795671 100644
--- a/components/unified_consent/unified_consent_service.cc
+++ b/components/unified_consent/unified_consent_service.cc
@@ -135,7 +135,10 @@
     return;
   }
 
-  DCHECK(sync_service_->IsSyncAllowed());
+  DCHECK(!sync_service_->HasDisableReason(
+      syncer::SyncService::DISABLE_REASON_PLATFORM_OVERRIDE));
+  DCHECK(!sync_service_->HasDisableReason(
+      syncer::SyncService::DISABLE_REASON_ENTERPRISE_POLICY));
   DCHECK(identity_manager_->HasPrimaryAccount());
   DCHECK_LT(MigrationState::NOT_INITIALIZED, GetMigrationState());
 
diff --git a/components/unified_consent/url_keyed_data_collection_consent_helper.cc b/components/unified_consent/url_keyed_data_collection_consent_helper.cc
index a504fdb..6ad5213 100644
--- a/components/unified_consent/url_keyed_data_collection_consent_helper.cc
+++ b/components/unified_consent/url_keyed_data_collection_consent_helper.cc
@@ -87,8 +87,10 @@
       sync_data_type_(sync_data_type),
       sync_data_type_upload_state_(
           syncer::GetUploadToGoogleState(sync_service_, sync_data_type_)) {
-  DCHECK(sync_service_);
-  sync_service_->AddObserver(this);
+  if (sync_service_)
+    sync_service_->AddObserver(this);
+  else
+    DCHECK_EQ(syncer::UploadState::NOT_ACTIVE, sync_data_type_upload_state_);
 }
 
 SyncBasedUrlKeyedDataCollectionConsentHelper::
diff --git a/components/unified_consent/url_keyed_data_collection_consent_helper_unittest.cc b/components/unified_consent/url_keyed_data_collection_consent_helper_unittest.cc
index a83653c..d677ccb 100644
--- a/components/unified_consent/url_keyed_data_collection_consent_helper_unittest.cc
+++ b/components/unified_consent/url_keyed_data_collection_consent_helper_unittest.cc
@@ -137,6 +137,16 @@
 }
 
 TEST_F(UrlKeyedDataCollectionConsentHelperTest,
+       AnonymizedDataCollection_UnifiedConsentDisabled_NullSyncService) {
+  std::unique_ptr<UrlKeyedDataCollectionConsentHelper> helper =
+      UrlKeyedDataCollectionConsentHelper::
+          NewAnonymizedDataCollectionConsentHelper(
+              false /* is_unified_consent_enabled */, &pref_service_,
+              nullptr /* sync_service */);
+  EXPECT_FALSE(helper->IsEnabled());
+}
+
+TEST_F(UrlKeyedDataCollectionConsentHelperTest,
        PersonalizeddDataCollection_UnifiedConsentEnabled) {
   std::unique_ptr<UrlKeyedDataCollectionConsentHelper> helper =
       UrlKeyedDataCollectionConsentHelper::
@@ -171,5 +181,25 @@
   helper->RemoveObserver(this);
 }
 
+TEST_F(UrlKeyedDataCollectionConsentHelperTest,
+       PersonalizedDataCollection_NullSyncService) {
+  {
+    std::unique_ptr<UrlKeyedDataCollectionConsentHelper> helper =
+        UrlKeyedDataCollectionConsentHelper::
+            NewPersonalizedDataCollectionConsentHelper(
+                false /* is_unified_consent_enabled */,
+                nullptr /* sync_service */);
+    EXPECT_FALSE(helper->IsEnabled());
+  }
+  {
+    std::unique_ptr<UrlKeyedDataCollectionConsentHelper> helper =
+        UrlKeyedDataCollectionConsentHelper::
+            NewPersonalizedDataCollectionConsentHelper(
+                true /* is_unified_consent_enabled */,
+                nullptr /* sync_service */);
+    EXPECT_FALSE(helper->IsEnabled());
+  }
+}
+
 }  // namespace
 }  // namespace unified_consent
diff --git a/components/viz/service/surfaces/surface_dependency_tracker.cc b/components/viz/service/surfaces/surface_dependency_tracker.cc
index a8773e6b..08a14bc 100644
--- a/components/viz/service/surfaces/surface_dependency_tracker.cc
+++ b/components/viz/service/surfaces/surface_dependency_tracker.cc
@@ -19,8 +19,6 @@
 void SurfaceDependencyTracker::RequestSurfaceResolution(Surface* surface) {
   DCHECK(surface->HasPendingFrame());
 
-  const CompositorFrame& pending_frame = surface->GetPendingFrame();
-
   if (IsSurfaceLate(surface)) {
     ActivateLateSurfaceSubtree(surface);
     return;
@@ -28,8 +26,7 @@
 
   // Activation dependencies that aren't currently known to the surface manager
   // or do not have an active CompositorFrame block this frame.
-  for (const SurfaceId& surface_id :
-       pending_frame.metadata.activation_dependencies) {
+  for (const SurfaceId& surface_id : surface->activation_dependencies()) {
     Surface* dependency = surface_manager_->GetSurfaceForId(surface_id);
     if (!dependency || !dependency->HasActiveFrame()) {
       blocked_surfaces_from_dependency_[surface_id.frame_sink_id()].insert(
@@ -75,12 +72,7 @@
   if (!surface->HasPendingFrame())
     return;
 
-  const CompositorFrame& pending_frame = surface->GetPendingFrame();
-
-  DCHECK(!pending_frame.metadata.activation_dependencies.empty());
-
-  for (const SurfaceId& surface_id :
-       pending_frame.metadata.activation_dependencies) {
+  for (const SurfaceId& surface_id : surface->activation_dependencies()) {
     auto it =
         blocked_surfaces_from_dependency_.find(surface_id.frame_sink_id());
     if (it == blocked_surfaces_from_dependency_.end())
@@ -110,10 +102,7 @@
 void SurfaceDependencyTracker::ActivateLateSurfaceSubtree(Surface* surface) {
   DCHECK(surface->HasPendingFrame());
 
-  const CompositorFrame& pending_frame = surface->GetPendingFrame();
-
-  for (const SurfaceId& surface_id :
-       pending_frame.metadata.activation_dependencies) {
+  for (const SurfaceId& surface_id : surface->activation_dependencies()) {
     Surface* dependency = surface_manager_->GetSurfaceForId(surface_id);
     if (dependency && dependency->HasPendingFrame())
       ActivateLateSurfaceSubtree(dependency);
@@ -125,8 +114,6 @@
 void SurfaceDependencyTracker::UpdateSurfaceDeadline(Surface* surface) {
   DCHECK(surface->HasPendingFrame());
 
-  const CompositorFrame& pending_frame = surface->GetPendingFrame();
-
   // Inherit the deadline from the first parent blocked on this surface.
   auto it = blocked_surfaces_from_dependency_.find(
       surface->surface_id().frame_sink_id());
@@ -146,8 +133,7 @@
          surface->has_deadline());
 
   // Recursively propagate the newly set deadline to children.
-  for (const SurfaceId& surface_id :
-       pending_frame.metadata.activation_dependencies) {
+  for (const SurfaceId& surface_id : surface->activation_dependencies()) {
     Surface* dependency = surface_manager_->GetSurfaceForId(surface_id);
     if (dependency && dependency->HasPendingFrame())
       UpdateSurfaceDeadline(dependency);
diff --git a/components/webdata_services/web_data_service_wrapper.cc b/components/webdata_services/web_data_service_wrapper.cc
index 8bcd4d3..0287a89 100644
--- a/components/webdata_services/web_data_service_wrapper.cc
+++ b/components/webdata_services/web_data_service_wrapper.cc
@@ -18,6 +18,7 @@
 #include "components/autofill/core/browser/webdata/autofill_profile_syncable_service.h"
 #include "components/autofill/core/browser/webdata/autofill_table.h"
 #include "components/autofill/core/browser/webdata/autofill_wallet_metadata_syncable_service.h"
+#include "components/autofill/core/browser/webdata/autofill_wallet_sync_bridge.h"
 #include "components/autofill/core/browser/webdata/autofill_wallet_syncable_service.h"
 #include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
 #include "components/autofill/core/common/autofill_features.h"
@@ -71,6 +72,9 @@
   }
 }
 
+// TODO(jkrcal): Rename this function when the last webdata sync type get
+// converted to USS, e.g. to InitSyncBridgesOnDBSequence(). Check also other
+// related functions.
 void InitSyncableAccountServicesOnDBSequence(
     scoped_refptr<base::SingleThreadTaskRunner> db_task_runner,
     const syncer::SyncableService::StartSyncFlare& sync_flare,
@@ -79,15 +83,21 @@
     const std::string& app_locale,
     autofill::AutofillWebDataBackend* autofill_backend) {
   DCHECK(db_task_runner->RunsTasksInCurrentSequence());
-  autofill::AutofillWalletSyncableService::CreateForWebDataServiceAndBackend(
-      autofill_web_data.get(), autofill_backend, app_locale);
+
+  if (base::FeatureList::IsEnabled(switches::kSyncUSSAutofillWalletData)) {
+    autofill::AutofillWalletSyncBridge::CreateForWebDataServiceAndBackend(
+        app_locale, autofill_backend, autofill_web_data.get());
+  } else {
+    autofill::AutofillWalletSyncableService::CreateForWebDataServiceAndBackend(
+        autofill_web_data.get(), autofill_backend, app_locale);
+    autofill::AutofillWalletSyncableService::FromWebDataService(
+        autofill_web_data.get())
+        ->InjectStartSyncFlare(sync_flare);
+  }
+
   autofill::AutofillWalletMetadataSyncableService::
       CreateForWebDataServiceAndBackend(autofill_web_data.get(),
                                         autofill_backend, app_locale);
-
-  autofill::AutofillWalletSyncableService::FromWebDataService(
-      autofill_web_data.get())
-      ->InjectStartSyncFlare(sync_flare);
 }
 
 }  // namespace
diff --git a/components/zucchini/BUILD.gn b/components/zucchini/BUILD.gn
index 77dc810..c871d9d1 100644
--- a/components/zucchini/BUILD.gn
+++ b/components/zucchini/BUILD.gn
@@ -78,8 +78,8 @@
     "rel32_finder.h",
     "rel32_utils.cc",
     "rel32_utils.h",
-    "reloc_utils.cc",
-    "reloc_utils.h",
+    "reloc_win32.cc",
+    "reloc_win32.h",
     "suffix_array.h",
     "target_pool.cc",
     "target_pool.h",
@@ -169,7 +169,7 @@
     "reference_set_unittest.cc",
     "rel32_finder_unittest.cc",
     "rel32_utils_unittest.cc",
-    "reloc_utils_unittest.cc",
+    "reloc_win32_unittest.cc",
     "suffix_array_unittest.cc",
     "target_pool_unittest.cc",
     "targets_affinity_unittest.cc",
diff --git a/components/zucchini/disassembler_win32.cc b/components/zucchini/disassembler_win32.cc
index a932bea2..7eba0f4 100644
--- a/components/zucchini/disassembler_win32.cc
+++ b/components/zucchini/disassembler_win32.cc
@@ -15,7 +15,7 @@
 #include "components/zucchini/buffer_source.h"
 #include "components/zucchini/rel32_finder.h"
 #include "components/zucchini/rel32_utils.h"
-#include "components/zucchini/reloc_utils.h"
+#include "components/zucchini/reloc_win32.h"
 
 namespace zucchini {
 
diff --git a/components/zucchini/reloc_utils.cc b/components/zucchini/reloc_win32.cc
similarity index 98%
rename from components/zucchini/reloc_utils.cc
rename to components/zucchini/reloc_win32.cc
index d6f69b9..2f993ad1 100644
--- a/components/zucchini/reloc_utils.cc
+++ b/components/zucchini/reloc_win32.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 "components/zucchini/reloc_utils.h"
+#include "components/zucchini/reloc_win32.h"
 
 #include <algorithm>
 #include <tuple>
diff --git a/components/zucchini/reloc_utils.h b/components/zucchini/reloc_win32.h
similarity index 97%
rename from components/zucchini/reloc_utils.h
rename to components/zucchini/reloc_win32.h
index aac1efb..207c6e7 100644
--- a/components/zucchini/reloc_utils.h
+++ b/components/zucchini/reloc_win32.h
@@ -2,8 +2,8 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef COMPONENTS_ZUCCHINI_RELOC_UTILS_H_
-#define COMPONENTS_ZUCCHINI_RELOC_UTILS_H_
+#ifndef COMPONENTS_ZUCCHINI_RELOC_WIN32_H_
+#define COMPONENTS_ZUCCHINI_RELOC_WIN32_H_
 
 #include <stddef.h>
 #include <stdint.h>
@@ -137,4 +137,4 @@
 
 }  // namespace zucchini
 
-#endif  // COMPONENTS_ZUCCHINI_RELOC_UTILS_H_
+#endif  // COMPONENTS_ZUCCHINI_RELOC_WIN32_H_
diff --git a/components/zucchini/reloc_utils_unittest.cc b/components/zucchini/reloc_win32_unittest.cc
similarity index 99%
rename from components/zucchini/reloc_utils_unittest.cc
rename to components/zucchini/reloc_win32_unittest.cc
index 65acf5b..ca9bbe6 100644
--- a/components/zucchini/reloc_utils_unittest.cc
+++ b/components/zucchini/reloc_win32_unittest.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 "components/zucchini/reloc_utils.h"
+#include "components/zucchini/reloc_win32.h"
 
 #include <stdint.h>
 
diff --git a/content/browser/background_fetch/background_fetch_service_impl.cc b/content/browser/background_fetch/background_fetch_service_impl.cc
index 596cfaa..864a1704c 100644
--- a/content/browser/background_fetch/background_fetch_service_impl.cc
+++ b/content/browser/background_fetch/background_fetch_service_impl.cc
@@ -106,10 +106,14 @@
     int64_t service_worker_registration_id,
     const std::string& developer_id,
     const std::string& unique_id,
-    const std::string& title,
+    const base::Optional<std::string>& title,
+    const SkBitmap& icon,
     UpdateUICallback callback) {
   DCHECK_CURRENTLY_ON(BrowserThread::IO);
-  if (!ValidateUniqueId(unique_id) || !ValidateTitle(title)) {
+
+  // TODO(crbug.com/865063): Remove the check for |title| when updating both the
+  // title and icons is supported.
+  if (!ValidateUniqueId(unique_id) || !title || !ValidateTitle(*title)) {
     std::move(callback).Run(
         blink::mojom::BackgroundFetchError::INVALID_ARGUMENT);
     return;
@@ -117,7 +121,7 @@
 
   BackgroundFetchRegistrationId registration_id(
       service_worker_registration_id, origin_, developer_id, unique_id);
-  background_fetch_context_->UpdateUI(registration_id, title,
+  background_fetch_context_->UpdateUI(registration_id, *title,
                                       std::move(callback));
 }
 
diff --git a/content/browser/background_fetch/background_fetch_service_impl.h b/content/browser/background_fetch/background_fetch_service_impl.h
index c61095e1..d59fbe16 100644
--- a/content/browser/background_fetch/background_fetch_service_impl.h
+++ b/content/browser/background_fetch/background_fetch_service_impl.h
@@ -47,7 +47,8 @@
   void UpdateUI(int64_t service_worker_registration_id,
                 const std::string& developer_id,
                 const std::string& unique_id,
-                const std::string& title,
+                const base::Optional<std::string>& title,
+                const SkBitmap& icon,
                 UpdateUICallback callback) override;
   void Abort(int64_t service_worker_registration_id,
              const std::string& developer_id,
diff --git a/content/browser/background_fetch/background_fetch_service_unittest.cc b/content/browser/background_fetch/background_fetch_service_unittest.cc
index 6e2a9c6..36f5702 100644
--- a/content/browser/background_fetch/background_fetch_service_unittest.cc
+++ b/content/browser/background_fetch/background_fetch_service_unittest.cc
@@ -183,7 +183,7 @@
 
     base::RunLoop run_loop;
     service_->UpdateUI(service_worker_registration_id, unique_id, developer_id,
-                       title,
+                       title, SkBitmap(),
                        base::BindOnce(&BackgroundFetchServiceTest::DidGetError,
                                       base::Unretained(this),
                                       run_loop.QuitClosure(), out_error));
diff --git a/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.cc b/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.cc
index eb9c846..bb71805 100644
--- a/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.cc
+++ b/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.cc
@@ -58,8 +58,7 @@
 
   media_web_contents_observer_ = initiator_->media_web_contents_observer();
 
-  window_ =
-      GetContentClient()->browser()->CreateWindowForPictureInPicture(this);
+  EnsureWindow();
   DCHECK(window_) << "Picture in Picture requires a valid window.";
 }
 
@@ -84,23 +83,25 @@
 }
 
 void PictureInPictureWindowControllerImpl::Close(bool should_pause_video) {
-  DCHECK(window_);
-
-  if (!window_->IsVisible())
+  if (!window_ || !window_->IsVisible())
     return;
 
   window_->Hide();
-  initiator_->SetHasPictureInPictureVideo(false);
+  CloseInternal(should_pause_video);
+}
 
-  surface_id_ = viz::SurfaceId();
-
-  OnLeavingPictureInPicture(should_pause_video);
+void PictureInPictureWindowControllerImpl::OnWindowDestroyed() {
+  window_ = nullptr;
+  embedder_ = nullptr;
+  CloseInternal(true /* should_pause_video */);
 }
 
 void PictureInPictureWindowControllerImpl::EmbedSurface(
     const viz::SurfaceId& surface_id,
     const gfx::Size& natural_size) {
+  EnsureWindow();
   DCHECK(window_);
+
   DCHECK(surface_id.is_valid());
   surface_id_ = surface_id;
 
@@ -195,4 +196,21 @@
   }
 }
 
+void PictureInPictureWindowControllerImpl::CloseInternal(
+    bool should_pause_video) {
+  initiator_->SetHasPictureInPictureVideo(false);
+
+  surface_id_ = viz::SurfaceId();
+
+  OnLeavingPictureInPicture(should_pause_video);
+}
+
+void PictureInPictureWindowControllerImpl::EnsureWindow() {
+  if (window_)
+    return;
+
+  window_ =
+      GetContentClient()->browser()->CreateWindowForPictureInPicture(this);
+}
+
 }  // namespace content
diff --git a/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.h b/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.h
index 8cd191f..7ce6d85 100644
--- a/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.h
+++ b/content/browser/picture_in_picture/picture_in_picture_window_controller_impl.h
@@ -36,6 +36,7 @@
   // PictureInPictureWindowController:
   CONTENT_EXPORT gfx::Size Show() override;
   CONTENT_EXPORT void Close(bool should_pause_video) override;
+  CONTENT_EXPORT void OnWindowDestroyed() override;
   CONTENT_EXPORT void ClickCustomControl(
       const std::string& control_id) override;
   CONTENT_EXPORT void EmbedSurface(const viz::SurfaceId& surface_id,
@@ -59,6 +60,14 @@
   // Signal to the media player that |this| is leaving Picture-in-Picture mode.
   void OnLeavingPictureInPicture(bool should_pause_video);
 
+  // Internal method to set the states after the window was closed, whether via
+  // the system or Chromium.
+  void CloseInternal(bool should_pause_video);
+
+  // Creates a new window if the previous one was destroyed. It can happen
+  // because of the system control of the window.
+  void EnsureWindow();
+
   std::unique_ptr<OverlayWindow> window_;
   std::unique_ptr<OverlaySurfaceEmbedder> embedder_;
   WebContentsImpl* const initiator_;
diff --git a/content/browser/renderer_host/input/input_router_impl_unittest.cc b/content/browser/renderer_host/input/input_router_impl_unittest.cc
index faa3314..a4d9ed9 100644
--- a/content/browser/renderer_host/input/input_router_impl_unittest.cc
+++ b/content/browser/renderer_host/input/input_router_impl_unittest.cc
@@ -767,7 +767,13 @@
   EXPECT_FALSE(HasPendingEvents());
 }
 
-TEST_F(InputRouterImplTest, GestureTypesIgnoringAck) {
+// TODO(https://crbug.com/866946): Test is flaky on Fuchsia.
+#if defined(OS_FUCHSIA)
+#define MAYBE_GestureTypesIgnoringAck DISABLED_GestureTypesIgnoringAck
+#else
+#define MAYBE_GestureTypesIgnoringAck GestureTypesIgnoringAck
+#endif
+TEST_F(InputRouterImplTest, MAYBE_GestureTypesIgnoringAck) {
   // We test every gesture type, ensuring that the stream of gestures is valid.
   const WebInputEvent::Type eventTypes[] = {
       WebInputEvent::kGestureTapDown,     WebInputEvent::kGestureShowPress,
diff --git a/content/browser/speech/speech_recognition_dispatcher_host.cc b/content/browser/speech/speech_recognition_dispatcher_host.cc
index 4e8664e..87617e6 100644
--- a/content/browser/speech/speech_recognition_dispatcher_host.cc
+++ b/content/browser/speech/speech_recognition_dispatcher_host.cc
@@ -201,7 +201,11 @@
     : session_id_(SpeechRecognitionManager::kSessionIDInvalid),
       client_(std::move(client_ptr_info)),
       stopped_(false),
-      weak_factory_(this) {}
+      weak_factory_(this) {
+  client_.set_connection_error_handler(
+      base::BindOnce(&SpeechRecognitionSession::ConnectionErrorHandler,
+                     base::Unretained(this)));
+}
 
 SpeechRecognitionSession::~SpeechRecognitionSession() {
   // If a connection error happens and the session hasn't been stopped yet,
@@ -273,4 +277,9 @@
 void SpeechRecognitionSession::OnEnvironmentEstimationComplete(int session_id) {
 }
 
+void SpeechRecognitionSession::ConnectionErrorHandler() {
+  if (!stopped_)
+    Abort();
+}
+
 }  // namespace content
diff --git a/content/browser/speech/speech_recognition_dispatcher_host.h b/content/browser/speech/speech_recognition_dispatcher_host.h
index ffa9bd51..9eeeb44 100644
--- a/content/browser/speech/speech_recognition_dispatcher_host.h
+++ b/content/browser/speech/speech_recognition_dispatcher_host.h
@@ -109,6 +109,8 @@
                            float noise_volume) override;
 
  private:
+  void ConnectionErrorHandler();
+
   int session_id_;
   blink::mojom::SpeechRecognitionSessionClientPtr client_;
   bool stopped_;
diff --git a/content/child/runtime_features.cc b/content/child/runtime_features.cc
index 717e332..279b13f 100644
--- a/content/child/runtime_features.cc
+++ b/content/child/runtime_features.cc
@@ -406,6 +406,9 @@
   WebRuntimeFeatures::EnableWorkStealingInScriptRunner(
       base::FeatureList::IsEnabled(features::kWorkStealingInScriptRunner));
 
+  WebRuntimeFeatures::EnableScheduledScriptStreaming(
+      base::FeatureList::IsEnabled(features::kScheduledScriptStreaming));
+
   WebRuntimeFeatures::EnableFeatureFromString(
       "FeaturePolicyForPermissions",
       base::FeatureList::IsEnabled(features::kUseFeaturePolicyForPermissions));
diff --git a/content/public/browser/picture_in_picture_window_controller.h b/content/public/browser/picture_in_picture_window_controller.h
index 64f01043..6df6f22 100644
--- a/content/public/browser/picture_in_picture_window_controller.h
+++ b/content/public/browser/picture_in_picture_window_controller.h
@@ -38,7 +38,14 @@
   // Returns the size of the window in pixels.
   virtual gfx::Size Show() = 0;
 
+  // Called to notify the controller that the window was requested to be closed
+  // by the user or the content.
   virtual void Close(bool should_pause_video) = 0;
+
+  // Called by the window implementation to notify the controller that the
+  // window was requested to be closed and destroyed by the system.
+  virtual void OnWindowDestroyed() = 0;
+
   virtual void ClickCustomControl(const std::string& control_id) = 0;
   virtual void EmbedSurface(const viz::SurfaceId& surface_id,
                             const gfx::Size& natural_size) = 0;
diff --git a/content/public/common/content_features.cc b/content/public/common/content_features.cc
index 12feb84..04d6bae 100644
--- a/content/public/common/content_features.cc
+++ b/content/public/common/content_features.cc
@@ -590,6 +590,10 @@
 const base::Feature kWorkStealingInScriptRunner{
     "WorkStealingInScriptRunner", base::FEATURE_DISABLED_BY_DEFAULT};
 
+// Enabled scheduler use for script streaming.
+const base::Feature kScheduledScriptStreaming{
+    "ScheduledScriptStreaming", base::FEATURE_DISABLED_BY_DEFAULT};
+
 #if defined(OS_ANDROID)
 // Autofill Accessibility in Android.
 // crbug.com/627860
diff --git a/content/public/common/content_features.h b/content/public/common/content_features.h
index 8f0e58ea..d6249a7 100644
--- a/content/public/common/content_features.h
+++ b/content/public/common/content_features.h
@@ -135,6 +135,7 @@
 CONTENT_EXPORT extern const base::Feature kWebXrOrientationSensorDevice;
 CONTENT_EXPORT extern const base::Feature kWipeCorruptV2IDBDatabases;
 CONTENT_EXPORT extern const base::Feature kWorkStealingInScriptRunner;
+CONTENT_EXPORT extern const base::Feature kScheduledScriptStreaming;
 
 #if defined(OS_ANDROID)
 CONTENT_EXPORT extern const base::Feature kAndroidAutofillAccessibility;
diff --git a/content/renderer/render_process_impl.cc b/content/renderer/render_process_impl.cc
index 5326845..80559a1 100644
--- a/content/renderer/render_process_impl.cc
+++ b/content/renderer/render_process_impl.cc
@@ -144,8 +144,10 @@
   SetV8FlagIfFeature(features::kV8VmFuture, "--future");
   SetV8FlagIfNotFeature(features::kV8VmFuture, "--no-future");
 
-  SetV8FlagIfFeature(features::kWebAssemblyBaseline, "--wasm-tier-up");
-  SetV8FlagIfNotFeature(features::kWebAssemblyBaseline, "--no-wasm-tier-up");
+  SetV8FlagIfFeature(features::kWebAssemblyBaseline,
+                     "--liftoff --wasm-tier-up");
+  SetV8FlagIfNotFeature(features::kWebAssemblyBaseline,
+                        "--no-liftoff --no-wasm-tier-up");
 
   if (base::FeatureList::IsEnabled(features::kWebAssemblyThreads)) {
     constexpr char kFlags[] =
diff --git a/content/shell/app/shell_main_delegate.cc b/content/shell/app/shell_main_delegate.cc
index d573edc2..0ec1525 100644
--- a/content/shell/app/shell_main_delegate.cc
+++ b/content/shell/app/shell_main_delegate.cc
@@ -248,6 +248,7 @@
     command_line.AppendSwitch(switches::kEnablePreciseMemoryInfo);
 
     command_line.AppendSwitchASCII(network::switches::kHostResolverRules,
+                                   "MAP nonexistent.*.test ~NOTFOUND,"
                                    "MAP *.test 127.0.0.1");
 
     command_line.AppendSwitch(switches::kEnablePartialRaster);
diff --git a/device/usb/BUILD.gn b/device/usb/BUILD.gn
index 3248fe6b4..5b183fcc 100644
--- a/device/usb/BUILD.gn
+++ b/device/usb/BUILD.gn
@@ -81,9 +81,6 @@
 
   if (is_win || is_mac) {
     sources += [
-      "scoped_libusb_device_handle.cc",
-      "scoped_libusb_device_handle.h",
-      "scoped_libusb_device_ref.cc",
       "scoped_libusb_device_ref.h",
       "usb_context.cc",
       "usb_context.h",
diff --git a/device/usb/scoped_libusb_device_handle.cc b/device/usb/scoped_libusb_device_handle.cc
deleted file mode 100644
index 4223a81..0000000
--- a/device/usb/scoped_libusb_device_handle.cc
+++ /dev/null
@@ -1,37 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "device/usb/scoped_libusb_device_handle.h"
-
-#include "device/usb/usb_context.h"
-#include "third_party/libusb/src/libusb/libusb.h"
-
-namespace device {
-
-ScopedLibusbDeviceHandle::ScopedLibusbDeviceHandle(
-    libusb_device_handle* handle,
-    scoped_refptr<UsbContext> context)
-    : handle_(handle), context_(std::move(context)) {}
-
-ScopedLibusbDeviceHandle::ScopedLibusbDeviceHandle(
-    ScopedLibusbDeviceHandle&& other)
-    : handle_(other.handle_), context_(std::move(other.context_)) {
-  other.handle_ = nullptr;
-}
-
-ScopedLibusbDeviceHandle::~ScopedLibusbDeviceHandle() {
-  Reset();
-}
-
-void ScopedLibusbDeviceHandle::Reset() {
-  libusb_close(handle_);
-  handle_ = nullptr;
-  context_.reset();
-}
-
-bool ScopedLibusbDeviceHandle::IsValid() const {
-  return handle_ != nullptr;
-}
-
-}  // namespace device
diff --git a/device/usb/scoped_libusb_device_handle.h b/device/usb/scoped_libusb_device_handle.h
deleted file mode 100644
index a4eb378..0000000
--- a/device/usb/scoped_libusb_device_handle.h
+++ /dev/null
@@ -1,41 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef DEVICE_USB_SCOPED_LIBUSB_DEVICE_HANDLE_H_
-#define DEVICE_USB_SCOPED_LIBUSB_DEVICE_HANDLE_H_
-
-#include "base/macros.h"
-#include "base/memory/scoped_refptr.h"
-
-struct libusb_device_handle;
-
-namespace device {
-
-class UsbContext;
-
-// This class owns a reference to a libusb_device_handle as well as a reference
-// to the libusb_context. The libusb_context must outlive any
-// libusb_device_handle instances created from it.
-class ScopedLibusbDeviceHandle {
- public:
-  ScopedLibusbDeviceHandle(libusb_device_handle* handle,
-                           scoped_refptr<UsbContext> context);
-  ScopedLibusbDeviceHandle(ScopedLibusbDeviceHandle&& other);
-  ~ScopedLibusbDeviceHandle();
-
-  libusb_device_handle* get() const { return handle_; }
-
-  void Reset();
-  bool IsValid() const;
-
- private:
-  libusb_device_handle* handle_;
-  scoped_refptr<UsbContext> context_;
-
-  DISALLOW_COPY_AND_ASSIGN(ScopedLibusbDeviceHandle);
-};
-
-}  // namespace device
-
-#endif  // DEVICE_USB_SCOPED_LIBUSB_DEVICE_HANDLE_H_
diff --git a/device/usb/scoped_libusb_device_ref.cc b/device/usb/scoped_libusb_device_ref.cc
deleted file mode 100644
index b3ac5e8..0000000
--- a/device/usb/scoped_libusb_device_ref.cc
+++ /dev/null
@@ -1,39 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "device/usb/scoped_libusb_device_ref.h"
-
-#include "device/usb/usb_context.h"
-#include "third_party/libusb/src/libusb/libusb.h"
-
-namespace device {
-
-ScopedLibusbDeviceRef::ScopedLibusbDeviceRef(libusb_device* device,
-                                             scoped_refptr<UsbContext> context)
-    : device_(device), context_(std::move(context)) {}
-
-ScopedLibusbDeviceRef::ScopedLibusbDeviceRef(ScopedLibusbDeviceRef&& other)
-    : device_(other.device_), context_(std::move(other.context_)) {
-  other.device_ = nullptr;
-}
-
-ScopedLibusbDeviceRef::~ScopedLibusbDeviceRef() {
-  Reset();
-}
-
-void ScopedLibusbDeviceRef::Reset() {
-  libusb_unref_device(device_);
-  device_ = nullptr;
-  context_.reset();
-}
-
-bool ScopedLibusbDeviceRef::IsValid() const {
-  return device_ != nullptr;
-}
-
-bool operator==(const ScopedLibusbDeviceRef& ref, libusb_device* device) {
-  return ref.get() == device;
-}
-
-}  // namespace device
diff --git a/device/usb/scoped_libusb_device_ref.h b/device/usb/scoped_libusb_device_ref.h
index 73a3731..05a21b2 100644
--- a/device/usb/scoped_libusb_device_ref.h
+++ b/device/usb/scoped_libusb_device_ref.h
@@ -5,40 +5,19 @@
 #ifndef DEVICE_USB_SCOPED_LIBUSB_DEVICE_REF_H_
 #define DEVICE_USB_SCOPED_LIBUSB_DEVICE_REF_H_
 
-#include "base/macros.h"
-#include "base/memory/scoped_refptr.h"
-
-struct libusb_device;
+#include "base/scoped_generic.h"
+#include "third_party/libusb/src/libusb/libusb.h"
 
 namespace device {
 
-class UsbContext;
+struct LibusbDeviceRefTraits {
+  static libusb_device* InvalidValue() { return nullptr; }
 
-// This class owns a reference to a libusb_device as well as a reference to
-// the libusb_context. The libusb_context must outlive any libusb_device
-// instances created from it.
-class ScopedLibusbDeviceRef {
- public:
-  ScopedLibusbDeviceRef(libusb_device* device,
-                        scoped_refptr<UsbContext> context);
-  ScopedLibusbDeviceRef(ScopedLibusbDeviceRef&& other);
-  ~ScopedLibusbDeviceRef();
-
-  libusb_device* get() const { return device_; }
-
-  scoped_refptr<UsbContext> GetContext() const { return context_; }
-
-  void Reset();
-  bool IsValid() const;
-
- private:
-  libusb_device* device_;
-  scoped_refptr<UsbContext> context_;
-
-  DISALLOW_COPY_AND_ASSIGN(ScopedLibusbDeviceRef);
+  static void Free(libusb_device* device) { libusb_unref_device(device); }
 };
 
-bool operator==(const ScopedLibusbDeviceRef& ref, libusb_device* device);
+using ScopedLibusbDeviceRef =
+    base::ScopedGeneric<libusb_device*, LibusbDeviceRefTraits>;
 
 }  // namespace device
 
diff --git a/device/usb/usb_device_handle_impl.cc b/device/usb/usb_device_handle_impl.cc
index b8c2307a..cce0691 100644
--- a/device/usb/usb_device_handle_impl.cc
+++ b/device/usb/usb_device_handle_impl.cc
@@ -311,7 +311,7 @@
   libusb_fill_control_setup(buffer->front(), type, request, value, index,
                             length);
   libusb_fill_control_transfer(transfer->platform_transfer_,
-                               device_handle->handle(), buffer->front(),
+                               device_handle->handle_, buffer->front(),
                                &UsbDeviceHandleImpl::Transfer::PlatformCallback,
                                transfer.get(), timeout);
 
@@ -341,7 +341,7 @@
   }
 
   libusb_fill_bulk_transfer(
-      transfer->platform_transfer_, device_handle->handle(), endpoint,
+      transfer->platform_transfer_, device_handle->handle_, endpoint,
       buffer->front(), length, &UsbDeviceHandleImpl::Transfer::PlatformCallback,
       transfer.get(), timeout);
 
@@ -371,7 +371,7 @@
   }
 
   libusb_fill_interrupt_transfer(
-      transfer->platform_transfer_, device_handle->handle(), endpoint,
+      transfer->platform_transfer_, device_handle->handle_, endpoint,
       buffer->front(), length, &UsbDeviceHandleImpl::Transfer::PlatformCallback,
       transfer.get(), timeout);
 
@@ -401,10 +401,10 @@
     return nullptr;
   }
 
-  libusb_fill_iso_transfer(
-      transfer->platform_transfer_, device_handle->handle(), endpoint,
-      buffer->front(), static_cast<int>(length), num_packets,
-      &Transfer::PlatformCallback, transfer.get(), timeout);
+  libusb_fill_iso_transfer(transfer->platform_transfer_, device_handle->handle_,
+                           endpoint, buffer->front(), static_cast<int>(length),
+                           num_packets, &Transfer::PlatformCallback,
+                           transfer.get(), timeout);
 
   for (size_t i = 0; i < packet_lengths.size(); ++i)
     transfer->platform_transfer_->iso_packet_desc[i].length = packet_lengths[i];
@@ -798,14 +798,16 @@
 }
 
 UsbDeviceHandleImpl::UsbDeviceHandleImpl(
+    scoped_refptr<UsbContext> context,
     scoped_refptr<UsbDeviceImpl> device,
-    ScopedLibusbDeviceHandle handle,
+    PlatformUsbDeviceHandle handle,
     scoped_refptr<base::SequencedTaskRunner> blocking_task_runner)
-    : device_(std::move(device)),
-      handle_(std::move(handle)),
+    : device_(device),
+      handle_(handle),
+      context_(context),
       task_runner_(base::ThreadTaskRunnerHandle::Get()),
       blocking_task_runner_(blocking_task_runner) {
-  DCHECK(handle_.IsValid()) << "Cannot create device with an invalid handle.";
+  DCHECK(handle) << "Cannot create device with NULL handle.";
 }
 
 UsbDeviceHandleImpl::~UsbDeviceHandleImpl() {
@@ -815,19 +817,17 @@
   // any thread. libusb is not safe to reentrancy so be sure not to try to close
   // the device from inside a transfer completion callback.
   if (blocking_task_runner_->RunsTasksInCurrentSequence()) {
-    handle_.Reset();
+    libusb_close(handle_);
   } else {
-    blocking_task_runner_->PostTask(
-        FROM_HERE,
-        base::BindOnce(base::DoNothing::Once<ScopedLibusbDeviceHandle>(),
-                       std::move(handle_)));
+    blocking_task_runner_->PostTask(FROM_HERE,
+                                    base::BindOnce(&libusb_close, handle_));
   }
 }
 
 void UsbDeviceHandleImpl::SetConfigurationOnBlockingThread(
     int configuration_value,
     ResultCallback callback) {
-  int rv = libusb_set_configuration(handle(), configuration_value);
+  int rv = libusb_set_configuration(handle_, configuration_value);
   if (rv != LIBUSB_SUCCESS) {
     USB_LOG(EVENT) << "Failed to set configuration " << configuration_value
                    << ": " << ConvertPlatformUsbErrorToString(rv);
@@ -855,7 +855,7 @@
 void UsbDeviceHandleImpl::ClaimInterfaceOnBlockingThread(
     int interface_number,
     ResultCallback callback) {
-  int rv = libusb_claim_interface(handle(), interface_number);
+  int rv = libusb_claim_interface(handle_, interface_number);
   scoped_refptr<InterfaceClaimer> interface_claimer;
   if (rv == LIBUSB_SUCCESS) {
     interface_claimer =
@@ -897,7 +897,7 @@
     int interface_number,
     int alternate_setting,
     ResultCallback callback) {
-  int rv = libusb_set_interface_alt_setting(handle(), interface_number,
+  int rv = libusb_set_interface_alt_setting(handle_, interface_number,
                                             alternate_setting);
   if (rv != LIBUSB_SUCCESS) {
     USB_LOG(EVENT) << "Failed to set interface " << interface_number
@@ -930,7 +930,7 @@
 }
 
 void UsbDeviceHandleImpl::ResetDeviceOnBlockingThread(ResultCallback callback) {
-  int rv = libusb_reset_device(handle());
+  int rv = libusb_reset_device(handle_);
   if (rv != LIBUSB_SUCCESS) {
     USB_LOG(EVENT) << "Failed to reset device: "
                    << ConvertPlatformUsbErrorToString(rv);
@@ -941,7 +941,7 @@
 
 void UsbDeviceHandleImpl::ClearHaltOnBlockingThread(uint8_t endpoint,
                                                     ResultCallback callback) {
-  int rv = libusb_clear_halt(handle(), endpoint);
+  int rv = libusb_clear_halt(handle_, endpoint);
   if (rv != LIBUSB_SUCCESS) {
     USB_LOG(EVENT) << "Failed to clear halt: "
                    << ConvertPlatformUsbErrorToString(rv);
diff --git a/device/usb/usb_device_handle_impl.h b/device/usb/usb_device_handle_impl.h
index 32dbe04..acf0332d 100644
--- a/device/usb/usb_device_handle_impl.h
+++ b/device/usb/usb_device_handle_impl.h
@@ -17,7 +17,6 @@
 #include "base/macros.h"
 #include "base/memory/ref_counted.h"
 #include "base/threading/thread_checker.h"
-#include "device/usb/scoped_libusb_device_handle.h"
 #include "device/usb/usb_device_handle.h"
 #include "third_party/libusb/src/libusb/libusb.h"
 
@@ -35,8 +34,10 @@
   const UsbEndpointDescriptor* endpoint;
 };
 
+class UsbContext;
 class UsbDeviceImpl;
 
+typedef libusb_device_handle* PlatformUsbDeviceHandle;
 typedef libusb_iso_packet_descriptor* PlatformUsbIsoPacketDescriptor;
 typedef libusb_transfer* PlatformUsbTransferHandle;
 
@@ -89,13 +90,14 @@
 
   // This constructor is called by UsbDeviceImpl.
   UsbDeviceHandleImpl(
+      scoped_refptr<UsbContext> context,
       scoped_refptr<UsbDeviceImpl> device,
-      ScopedLibusbDeviceHandle handle,
+      PlatformUsbDeviceHandle handle,
       scoped_refptr<base::SequencedTaskRunner> blocking_task_runner);
 
   ~UsbDeviceHandleImpl() override;
 
-  libusb_device_handle* handle() const { return handle_.get(); }
+  PlatformUsbDeviceHandle handle() const { return handle_; }
 
  private:
   class InterfaceClaimer;
@@ -172,7 +174,7 @@
 
   scoped_refptr<UsbDeviceImpl> device_;
 
-  ScopedLibusbDeviceHandle handle_;
+  PlatformUsbDeviceHandle handle_;
 
   typedef std::map<int, scoped_refptr<InterfaceClaimer>> ClaimedInterfaceMap;
   ClaimedInterfaceMap claimed_interfaces_;
@@ -184,6 +186,10 @@
   typedef std::map<int, EndpointMapValue> EndpointMap;
   EndpointMap endpoint_map_;
 
+  // Retain the UsbContext so that the platform context will not be destroyed
+  // before this handle.
+  scoped_refptr<UsbContext> context_;
+
   scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
   scoped_refptr<base::SequencedTaskRunner> blocking_task_runner_;
 
diff --git a/device/usb/usb_device_impl.cc b/device/usb/usb_device_impl.cc
index d6eb68d..056c6c3 100644
--- a/device/usb/usb_device_impl.cc
+++ b/device/usb/usb_device_impl.cc
@@ -28,7 +28,8 @@
 
 namespace device {
 
-UsbDeviceImpl::UsbDeviceImpl(ScopedLibusbDeviceRef platform_device,
+UsbDeviceImpl::UsbDeviceImpl(scoped_refptr<UsbContext> context,
+                             ScopedLibusbDeviceRef platform_device,
                              const libusb_device_descriptor& descriptor)
     : UsbDevice(descriptor.bcdUSB,
                 descriptor.bDeviceClass,
@@ -40,8 +41,9 @@
                 base::string16(),
                 base::string16(),
                 base::string16()),
+      context_(std::move(context)),
       platform_device_(std::move(platform_device)) {
-  CHECK(platform_device_.IsValid()) << "platform_device must be valid";
+  CHECK(platform_device_.is_valid()) << "platform_device must be valid";
   ReadAllConfigurations();
   RefreshActiveConfiguration();
 }
@@ -102,14 +104,12 @@
     scoped_refptr<base::TaskRunner> task_runner,
     scoped_refptr<base::SequencedTaskRunner> blocking_task_runner) {
   base::AssertBlockingAllowed();
-  libusb_device_handle* handle;
+  PlatformUsbDeviceHandle handle;
   const int rv = libusb_open(platform_device(), &handle);
-  ScopedLibusbDeviceHandle scoped_handle(handle, platform_device_.GetContext());
   if (LIBUSB_SUCCESS == rv) {
     task_runner->PostTask(
-        FROM_HERE,
-        base::BindOnce(&UsbDeviceImpl::Opened, this, std::move(scoped_handle),
-                       std::move(callback), blocking_task_runner));
+        FROM_HERE, base::BindOnce(&UsbDeviceImpl::Opened, this, handle,
+                                  std::move(callback), blocking_task_runner));
   } else {
     USB_LOG(EVENT) << "Failed to open device: "
                    << ConvertPlatformUsbErrorToString(rv);
@@ -119,12 +119,12 @@
 }
 
 void UsbDeviceImpl::Opened(
-    ScopedLibusbDeviceHandle platform_handle,
+    PlatformUsbDeviceHandle platform_handle,
     OpenCallback callback,
     scoped_refptr<base::SequencedTaskRunner> blocking_task_runner) {
   DCHECK(thread_checker_.CalledOnValidThread());
   scoped_refptr<UsbDeviceHandle> device_handle = new UsbDeviceHandleImpl(
-      this, std::move(platform_handle), blocking_task_runner);
+      context_, this, platform_handle, blocking_task_runner);
   handles().push_back(device_handle.get());
   std::move(callback).Run(device_handle);
 }
diff --git a/device/usb/usb_device_impl.h b/device/usb/usb_device_impl.h
index 9e3d2c7..16ab891 100644
--- a/device/usb/usb_device_impl.h
+++ b/device/usb/usb_device_impl.h
@@ -21,7 +21,9 @@
 #include "device/usb/usb_descriptors.h"
 #include "device/usb/usb_device.h"
 
+struct libusb_device;
 struct libusb_device_descriptor;
+struct libusb_device_handle;
 
 namespace base {
 class SequencedTaskRunner;
@@ -29,12 +31,15 @@
 
 namespace device {
 
-class ScopedLibusbDeviceHandle;
 class UsbDeviceHandleImpl;
+class UsbContext;
+
+typedef struct libusb_device_handle* PlatformUsbDeviceHandle;
 
 class UsbDeviceImpl : public UsbDevice {
  public:
-  UsbDeviceImpl(ScopedLibusbDeviceRef platform_device,
+  UsbDeviceImpl(scoped_refptr<UsbContext> context,
+                ScopedLibusbDeviceRef platform_device,
                 const libusb_device_descriptor& descriptor);
 
   // UsbDevice implementation:
@@ -74,13 +79,15 @@
       OpenCallback callback,
       scoped_refptr<base::TaskRunner> task_runner,
       scoped_refptr<base::SequencedTaskRunner> blocking_task_runner);
-  void Opened(ScopedLibusbDeviceHandle platform_handle,
+  void Opened(PlatformUsbDeviceHandle platform_handle,
               OpenCallback callback,
               scoped_refptr<base::SequencedTaskRunner> blocking_task_runner);
 
   base::ThreadChecker thread_checker_;
   bool visited_ = false;
 
+  // The libusb_context must not be released before the libusb_device.
+  const scoped_refptr<UsbContext> context_;
   const ScopedLibusbDeviceRef platform_device_;
 
   DISALLOW_COPY_AND_ASSIGN(UsbDeviceImpl);
diff --git a/device/usb/usb_service_impl.cc b/device/usb/usb_service_impl.cc
index 626ece05..83e14c6f0 100644
--- a/device/usb/usb_service_impl.cc
+++ b/device/usb/usb_service_impl.cc
@@ -131,7 +131,7 @@
   std::vector<ScopedLibusbDeviceRef> scoped_devices;
   scoped_devices.reserve(device_count);
   for (ssize_t i = 0; i < device_count; ++i)
-    scoped_devices.emplace_back(platform_devices[i], usb_context);
+    scoped_devices.emplace_back(platform_devices[i]);
 
   // Free the list but don't unref the devices because ownership has been
   // been transfered to the elements of |scoped_devices|.
@@ -231,7 +231,6 @@
       device_observer_(this),
 #endif
       weak_factory_(this) {
-  weak_self_ = weak_factory_.GetWeakPtr();
   base::PostTaskWithTraits(
       FROM_HERE, kBlockingTaskTraits,
       base::Bind(&InitializeUsbContextOnBlockingThread, task_runner(),
@@ -362,7 +361,7 @@
       // Mark the existing device object visited and remove it from the list so
       // it will not be ignored.
       it->second->set_visited(true);
-      device.Reset();
+      device.reset();
     }
   }
 
@@ -383,7 +382,7 @@
   // that have been removed don't remain in |ignored_devices_| indefinitely.
   ignored_devices_.clear();
   for (auto& device : *devices) {
-    if (device.IsValid())
+    if (device.is_valid())
       ignored_devices_.push_back(std::move(device));
   }
 
@@ -442,8 +441,8 @@
 
   devices_being_enumerated_.insert(platform_device.get());
 
-  auto device = base::MakeRefCounted<UsbDeviceImpl>(std::move(platform_device),
-                                                    descriptor);
+  auto device = base::MakeRefCounted<UsbDeviceImpl>(
+      context_, std::move(platform_device), descriptor);
   base::OnceClosure add_device =
       base::BindOnce(&UsbServiceImpl::AddDevice, weak_factory_.GetWeakPtr(),
                      refresh_complete, device);
@@ -459,8 +458,7 @@
     libusb_ref_device(device->platform_device());
     base::OnceClosure enumeration_failed = base::BindOnce(
         &UsbServiceImpl::EnumerationFailed, weak_factory_.GetWeakPtr(),
-        ScopedLibusbDeviceRef(device->platform_device(), context_),
-        refresh_complete);
+        ScopedLibusbDeviceRef(device->platform_device()), refresh_complete);
 
     device->Open(base::BindOnce(
         &OnDeviceOpenedReadDescriptors, descriptor.iManufacturer,
@@ -520,18 +518,18 @@
   // libusb does not transfer ownership of |device_raw| to this function so a
   // reference must be taken here.
   libusb_ref_device(device_raw);
-  ScopedLibusbDeviceRef device(device_raw, self->context_);
+  ScopedLibusbDeviceRef device(device_raw);
 
   switch (event) {
     case LIBUSB_HOTPLUG_EVENT_DEVICE_ARRIVED:
       self->task_runner()->PostTask(
           FROM_HERE, base::BindOnce(&UsbServiceImpl::OnPlatformDeviceAdded,
-                                    self->weak_self_, std::move(device)));
+                                    base::Unretained(self), std::move(device)));
       break;
     case LIBUSB_HOTPLUG_EVENT_DEVICE_LEFT:
       self->task_runner()->PostTask(
           FROM_HERE, base::BindOnce(&UsbServiceImpl::OnPlatformDeviceRemoved,
-                                    self->weak_self_, std::move(device)));
+                                    base::Unretained(self), std::move(device)));
       break;
     default:
       NOTREACHED();
diff --git a/device/usb/usb_service_impl.h b/device/usb/usb_service_impl.h
index 36f7115..e7e89d0 100644
--- a/device/usb/usb_service_impl.h
+++ b/device/usb/usb_service_impl.h
@@ -122,10 +122,6 @@
   ScopedObserver<DeviceMonitorWin, DeviceMonitorWin::Observer> device_observer_;
 #endif  // OS_WIN
 
-  // This WeakPtr is used to safely post hotplug events back to the thread this
-  // object lives on.
-  base::WeakPtr<UsbServiceImpl> weak_self_;
-
   base::WeakPtrFactory<UsbServiceImpl> weak_factory_;
 
   DISALLOW_COPY_AND_ASSIGN(UsbServiceImpl);
diff --git a/device/vr/BUILD.gn b/device/vr/BUILD.gn
index f0a7cb1..9a11ad0d 100644
--- a/device/vr/BUILD.gn
+++ b/device/vr/BUILD.gn
@@ -160,8 +160,6 @@
       "test/fake_vr_display_impl_client.h",
       "test/fake_vr_service_client.cc",
       "test/fake_vr_service_client.h",
-      "test/mock_vr_display_impl.cc",
-      "test/mock_vr_display_impl.h",
       "vr_export.h",
     ]
 
diff --git a/device/vr/android/gvr/gvr_device.cc b/device/vr/android/gvr/gvr_device.cc
index 20166c65..9d4c9b17 100644
--- a/device/vr/android/gvr/gvr_device.cc
+++ b/device/vr/android/gvr/gvr_device.cc
@@ -179,32 +179,31 @@
 void GvrDevice::RequestSession(
     mojom::XRDeviceRuntimeSessionOptionsPtr options,
     mojom::XRRuntime::RequestSessionCallback callback) {
+  if (!options->immersive) {
+    // TODO(https://crbug.com/695937): This should be NOTREACHED() once we no
+    // longer need the hacked GRV non-immersive mode.
+    ReturnNonImmersiveSession(std::move(callback));
+    return;
+  }
+
   GvrDelegateProvider* delegate_provider = GetGvrDelegateProvider();
   if (!delegate_provider) {
     std::move(callback).Run(nullptr, nullptr);
     return;
   }
 
-  if (options->immersive) {
-    // StartWebXRPresentation is async as we may trigger a DON (Device ON) flow
-    // that pauses Chrome.
-    delegate_provider->StartWebXRPresentation(
-        GetVRDisplayInfo(), std::move(options),
-        base::BindOnce(&GvrDevice::OnStartPresentResult,
-                       weak_ptr_factory_.GetWeakPtr(), std::move(callback)));
-  } else {
-    // TODO(https://crbug.com/695937): This should be NOTREACHED() once
-    // orientation device handles non-immersive VR sessions.
-    // TODO(https://crbug.com/842025): Handle this when RequestSession is called
-    // for non-immersive sessions.
-    NOTREACHED();
-  }
+  // StartWebXRPresentation is async as we may trigger a DON (Device ON) flow
+  // that pauses Chrome.
+  delegate_provider->StartWebXRPresentation(
+      GetVRDisplayInfo(), std::move(options),
+      base::BindOnce(&GvrDevice::OnStartPresentResult,
+                     weak_ptr_factory_.GetWeakPtr(), std::move(callback)));
 }
 
 void GvrDevice::OnStartPresentResult(
     mojom::XRRuntime::RequestSessionCallback callback,
     mojom::XRSessionPtr session) {
-  if (!session || !session->connection) {
+  if (!session) {
     std::move(callback).Run(nullptr, nullptr);
     return;
   }
@@ -223,8 +222,7 @@
       base::BindOnce(&GvrDevice::OnPresentingControllerMojoConnectionError,
                      base::Unretained(this)));
 
-  std::move(callback).Run(std::move(session->connection),
-                          std::move(session_controller));
+  std::move(callback).Run(std::move(session), std::move(session_controller));
 }
 
 // XRSessionController
@@ -246,7 +244,7 @@
 }
 
 void GvrDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   mojom::XRFrameDataPtr frame_data = mojom::XRFrameData::New();
   frame_data->pose =
       GvrDelegate::GetVRPosePtrWithNeckModel(gvr_api_.get(), nullptr);
diff --git a/device/vr/android/gvr/gvr_device.h b/device/vr/android/gvr/gvr_device.h
index 6061fcbe..e401efc0 100644
--- a/device/vr/android/gvr/gvr_device.h
+++ b/device/vr/android/gvr/gvr_device.h
@@ -42,7 +42,7 @@
   // VRDeviceBase
   void OnListeningForActivate(bool listening) override;
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
 
   void OnStartPresentResult(mojom::XRRuntime::RequestSessionCallback callback,
                             mojom::XRSessionPtr session);
diff --git a/device/vr/oculus/oculus_device.cc b/device/vr/oculus/oculus_device.cc
index 51b2627..eeb68ee 100644
--- a/device/vr/oculus/oculus_device.cc
+++ b/device/vr/oculus/oculus_device.cc
@@ -113,6 +113,11 @@
 void OculusDevice::RequestSession(
     mojom::XRDeviceRuntimeSessionOptionsPtr options,
     mojom::XRRuntime::RequestSessionCallback callback) {
+  if (!options->immersive) {
+    ReturnNonImmersiveSession(std::move(callback));
+    return;
+  }
+
   StopOvrSession();
 
   if (!render_loop_->IsRunning())
@@ -137,9 +142,7 @@
 void OculusDevice::OnRequestSessionResult(
     mojom::XRRuntime::RequestSessionCallback callback,
     bool result,
-    mojom::VRSubmitFrameClientRequest request,
-    mojom::VRPresentationProviderPtrInfo provider_info,
-    mojom::VRDisplayFrameTransportOptionsPtr transport_options) {
+    mojom::XRSessionPtr session) {
   if (!result) {
     std::move(callback).Run(nullptr, nullptr);
 
@@ -150,11 +153,6 @@
 
   OnStartPresenting();
 
-  auto connection = mojom::XRPresentationConnection::New();
-  connection->client_request = std::move(request);
-  connection->provider = std::move(provider_info);
-  connection->transport_options = std::move(transport_options);
-
   mojom::XRSessionControllerPtr session_controller;
   exclusive_controller_binding_.Bind(mojo::MakeRequest(&session_controller));
 
@@ -164,7 +162,7 @@
       base::BindOnce(&OculusDevice::OnPresentingControllerMojoConnectionError,
                      base::Unretained(this)));
 
-  std::move(callback).Run(std::move(connection), std::move(session_controller));
+  std::move(callback).Run(std::move(session), std::move(session_controller));
 
   if (!oculus_gamepad_factory_) {
     oculus_gamepad_factory_ =
@@ -227,7 +225,7 @@
 }
 
 void OculusDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   if (!session_) {
     std::move(callback).Run(nullptr);
     return;
diff --git a/device/vr/oculus/oculus_device.h b/device/vr/oculus/oculus_device.h
index cff84ccd..9002ac33 100644
--- a/device/vr/oculus/oculus_device.h
+++ b/device/vr/oculus/oculus_device.h
@@ -31,13 +31,10 @@
       mojom::XRDeviceRuntimeSessionOptionsPtr options,
       mojom::XRRuntime::RequestSessionCallback callback) override;
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
-  void OnRequestSessionResult(
-      mojom::XRRuntime::RequestSessionCallback callback,
-      bool result,
-      mojom::VRSubmitFrameClientRequest request,
-      mojom::VRPresentationProviderPtrInfo provider_info,
-      mojom::VRDisplayFrameTransportOptionsPtr transport_options);
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
+  void OnRequestSessionResult(mojom::XRRuntime::RequestSessionCallback callback,
+                              bool result,
+                              mojom::XRSessionPtr session);
 
   bool IsInitialized() { return !!session_; }
 
diff --git a/device/vr/oculus/oculus_render_loop.cc b/device/vr/oculus/oculus_render_loop.cc
index 80a37b6e..f914b14 100644
--- a/device/vr/oculus/oculus_render_loop.cc
+++ b/device/vr/oculus/oculus_render_loop.cc
@@ -48,7 +48,8 @@
         on_controller_updated)
     : base::Thread("OculusRenderLoop"),
       main_thread_task_runner_(base::ThreadTaskRunnerHandle::Get()),
-      binding_(this),
+      presentation_binding_(this),
+      frame_data_binding_(this),
       on_presentation_ended_(on_presentation_ended),
       on_controller_updated_(on_controller_updated),
       weak_ptr_factory_(this) {
@@ -69,7 +70,8 @@
 void OculusRenderLoop::CleanUp() {
   submit_client_ = nullptr;
   StopOvrSession();
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
 }
 
 void OculusRenderLoop::SubmitFrameMissing(int16_t frame_index,
@@ -241,28 +243,36 @@
 #endif
           ) {
     main_thread_task_runner_->PostTask(
-        FROM_HERE,
-        base::BindOnce(std::move(callback), false, nullptr, nullptr, nullptr));
+        FROM_HERE, base::BindOnce(std::move(callback), false, nullptr));
     return;
   }
 
-  binding_.Close();
-  device::mojom::VRPresentationProviderPtr provider;
-  binding_.Bind(mojo::MakeRequest(&provider));
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
+  device::mojom::XRPresentationProviderPtr presentation_provider;
+  device::mojom::XRFrameDataProviderPtr frame_data_provider;
+  presentation_binding_.Bind(mojo::MakeRequest(&presentation_provider));
+  frame_data_binding_.Bind(mojo::MakeRequest(&frame_data_provider));
 
-  device::mojom::VRDisplayFrameTransportOptionsPtr transport_options =
-      device::mojom::VRDisplayFrameTransportOptions::New();
+  device::mojom::XRPresentationTransportOptionsPtr transport_options =
+      device::mojom::XRPresentationTransportOptions::New();
   transport_options->transport_method =
-      device::mojom::VRDisplayFrameTransportMethod::SUBMIT_AS_TEXTURE_HANDLE;
+      device::mojom::XRPresentationTransportMethod::SUBMIT_AS_TEXTURE_HANDLE;
   // Only set boolean options that we need. Default is false, and we should be
   // able to safely ignore ones that our implementation doesn't care about.
   transport_options->wait_for_transfer_notification = true;
 
+  auto submit_frame_sink = device::mojom::XRPresentationConnection::New();
+  submit_frame_sink->provider = presentation_provider.PassInterface();
+  submit_frame_sink->client_request = mojo::MakeRequest(&submit_client_);
+  submit_frame_sink->transport_options = std::move(transport_options);
+
+  auto session = device::mojom::XRSession::New();
+  session->data_provider = frame_data_provider.PassInterface();
+  session->submit_frame_sink = std::move(submit_frame_sink);
+
   main_thread_task_runner_->PostTask(
-      FROM_HERE,
-      base::BindOnce(std::move(callback), true,
-                     mojo::MakeRequest(&submit_client_),
-                     provider.PassInterface(), std::move(transport_options)));
+      FROM_HERE, base::BindOnce(std::move(callback), true, std::move(session)));
   is_presenting_ = true;
 }
 
@@ -296,7 +306,8 @@
 
 void OculusRenderLoop::ExitPresent() {
   is_presenting_ = false;
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
   submit_client_ = nullptr;
   ClearPendingFrame();
 
@@ -312,7 +323,7 @@
 }
 
 void OculusRenderLoop::GetFrameData(
-    mojom::VRPresentationProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   DCHECK(is_presenting_);
 
   if (has_outstanding_frame_) {
diff --git a/device/vr/oculus/oculus_render_loop.h b/device/vr/oculus/oculus_render_loop.h
index c8186b7..26cfdd5 100644
--- a/device/vr/oculus/oculus_render_loop.h
+++ b/device/vr/oculus/oculus_render_loop.h
@@ -23,13 +23,12 @@
 
 const int kMaxOculusRenderLoopInputId = (ovrControllerType_Remote + 1);
 
-class OculusRenderLoop : public base::Thread, mojom::VRPresentationProvider {
+class OculusRenderLoop : public base::Thread,
+                         mojom::XRPresentationProvider,
+                         mojom::XRFrameDataProvider {
  public:
   using RequestSessionCallback =
-      base::OnceCallback<void(bool result,
-                              mojom::VRSubmitFrameClientRequest,
-                              mojom::VRPresentationProviderPtrInfo,
-                              mojom::VRDisplayFrameTransportOptionsPtr)>;
+      base::OnceCallback<void(bool result, mojom::XRSessionPtr)>;
 
   OculusRenderLoop(
       base::RepeatingCallback<void()> on_presentation_ended,
@@ -43,7 +42,7 @@
   void ExitPresent();
   base::WeakPtr<OculusRenderLoop> GetWeakPtr();
 
-  // VRPresentationProvider overrides:
+  // XRPresentationProvider overrides:
   void SubmitFrameMissing(int16_t frame_index, const gpu::SyncToken&) override;
   void SubmitFrame(int16_t frame_index,
                    const gpu::MailboxHolder& mailbox,
@@ -58,7 +57,7 @@
                          const gfx::RectF& right_bounds,
                          const gfx::Size& source_size) override;
   void GetFrameData(
-      mojom::VRPresentationProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
 
  private:
   // base::Thread overrides:
@@ -98,13 +97,14 @@
   gfx::RectF right_bounds_;
   gfx::Size source_size_;
   scoped_refptr<base::SingleThreadTaskRunner> main_thread_task_runner_;
-  mojom::VRSubmitFrameClientPtr submit_client_;
+  mojom::XRPresentationClientPtr submit_client_;
   ovrSession session_ = nullptr;
   ovrGraphicsLuid luid_ = {};
   ovrPosef last_render_pose_;
   ovrTextureSwapChain texture_swap_chain_ = 0;
   double sensor_time_;
-  mojo::Binding<mojom::VRPresentationProvider> binding_;
+  mojo::Binding<mojom::XRPresentationProvider> presentation_binding_;
+  mojo::Binding<mojom::XRFrameDataProvider> frame_data_binding_;
   bool primary_input_pressed[kMaxOculusRenderLoopInputId];
   base::RepeatingCallback<void()> on_presentation_ended_;
   base::RepeatingCallback<
diff --git a/device/vr/openvr/openvr_device.cc b/device/vr/openvr/openvr_device.cc
index 09ab3ef..e3d4aa2 100644
--- a/device/vr/openvr/openvr_device.cc
+++ b/device/vr/openvr/openvr_device.cc
@@ -173,6 +173,11 @@
 void OpenVRDevice::RequestSession(
     mojom::XRDeviceRuntimeSessionOptionsPtr options,
     mojom::XRRuntime::RequestSessionCallback callback) {
+  if (!options->immersive) {
+    ReturnNonImmersiveSession(std::move(callback));
+    return;
+  }
+
   if (!render_loop_->IsRunning())
     render_loop_->Start();
 
@@ -211,9 +216,7 @@
 void OpenVRDevice::OnRequestSessionResult(
     mojom::XRRuntime::RequestSessionCallback callback,
     bool result,
-    mojom::VRSubmitFrameClientRequest request,
-    mojom::VRPresentationProviderPtrInfo provider_info,
-    mojom::VRDisplayFrameTransportOptionsPtr transport_options) {
+    mojom::XRSessionPtr session) {
   if (!result) {
     OnPresentationEnded();
     std::move(callback).Run(nullptr, nullptr);
@@ -222,11 +225,6 @@
 
   OnStartPresenting();
 
-  auto connection = mojom::XRPresentationConnection::New();
-  connection->client_request = std::move(request);
-  connection->provider = std::move(provider_info);
-  connection->transport_options = std::move(transport_options);
-
   mojom::XRSessionControllerPtr session_controller;
   exclusive_controller_binding_.Bind(mojo::MakeRequest(&session_controller));
 
@@ -236,7 +234,7 @@
       base::BindOnce(&OpenVRDevice::OnPresentingControllerMojoConnectionError,
                      base::Unretained(this)));
 
-  std::move(callback).Run(std::move(connection), std::move(session_controller));
+  std::move(callback).Run(std::move(session), std::move(session_controller));
 }
 
 // XRSessionController
@@ -255,7 +253,7 @@
 }
 
 void OpenVRDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRPresentationProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   if (!openvr_) {
     std::move(callback).Run(nullptr);
     return;
diff --git a/device/vr/openvr/openvr_device.h b/device/vr/openvr/openvr_device.h
index c5af92f..95f4794 100644
--- a/device/vr/openvr/openvr_device.h
+++ b/device/vr/openvr/openvr_device.h
@@ -36,19 +36,16 @@
 
   void OnPollingEvents();
 
-  void OnRequestSessionResult(
-      mojom::XRRuntime::RequestSessionCallback callback,
-      bool result,
-      mojom::VRSubmitFrameClientRequest request,
-      mojom::VRPresentationProviderPtrInfo provider_info,
-      mojom::VRDisplayFrameTransportOptionsPtr transport_options);
+  void OnRequestSessionResult(mojom::XRRuntime::RequestSessionCallback callback,
+                              bool result,
+                              mojom::XRSessionPtr session);
 
   bool IsInitialized() { return !!openvr_; }
 
  private:
   // VRDeviceBase
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
 
   // XRSessionController
   void SetFrameDataRestricted(bool restricted) override;
diff --git a/device/vr/openvr/openvr_render_loop.cc b/device/vr/openvr/openvr_render_loop.cc
index fa37e4d..ce598d1 100644
--- a/device/vr/openvr/openvr_render_loop.cc
+++ b/device/vr/openvr/openvr_render_loop.cc
@@ -50,7 +50,8 @@
     : base::Thread("OpenVRRenderLoop"),
       main_thread_task_runner_(base::ThreadTaskRunnerHandle::Get()),
       update_gamepad_(std::move(update_gamepad)),
-      binding_(this),
+      presentation_binding_(this),
+      frame_data_binding_(this),
       weak_ptr_factory_(this) {
   DCHECK(main_thread_task_runner_);
 }
@@ -148,7 +149,8 @@
 
 void OpenVRRenderLoop::CleanUp() {
   submit_client_ = nullptr;
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
 }
 
 void OpenVRRenderLoop::UpdateLayerBounds(int16_t frame_id,
@@ -169,15 +171,15 @@
     mojom::XRDeviceRuntimeSessionOptionsPtr options,
     RequestSessionCallback callback) {
   DCHECK(options->immersive);
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
 
   if (!openvr_) {
     openvr_ = std::make_unique<OpenVRWrapper>(true);
     if (!openvr_->IsInitialized()) {
       openvr_ = nullptr;
       main_thread_task_runner_->PostTask(
-          FROM_HERE, base::BindOnce(std::move(callback), false, nullptr,
-                                    nullptr, nullptr));
+          FROM_HERE, base::BindOnce(std::move(callback), false, nullptr));
       return;
     }
 
@@ -193,20 +195,22 @@
       !texture_helper_.EnsureInitialized()) {
     openvr_ = nullptr;
     main_thread_task_runner_->PostTask(
-        FROM_HERE,
-        base::BindOnce(std::move(callback), false, nullptr, nullptr, nullptr));
+        FROM_HERE, base::BindOnce(std::move(callback), false, nullptr));
     return;
   }
 #endif
   DCHECK(!on_presentation_ended_);
   on_presentation_ended_ = std::move(on_presentation_ended);
-  device::mojom::VRPresentationProviderPtr provider;
-  binding_.Bind(mojo::MakeRequest(&provider));
 
-  device::mojom::VRDisplayFrameTransportOptionsPtr transport_options =
-      device::mojom::VRDisplayFrameTransportOptions::New();
+  device::mojom::XRPresentationProviderPtr presentation_provider;
+  device::mojom::XRFrameDataProviderPtr frame_data_provider;
+  presentation_binding_.Bind(mojo::MakeRequest(&presentation_provider));
+  frame_data_binding_.Bind(mojo::MakeRequest(&frame_data_provider));
+
+  device::mojom::XRPresentationTransportOptionsPtr transport_options =
+      device::mojom::XRPresentationTransportOptions::New();
   transport_options->transport_method =
-      device::mojom::VRDisplayFrameTransportMethod::SUBMIT_AS_TEXTURE_HANDLE;
+      device::mojom::XRPresentationTransportMethod::SUBMIT_AS_TEXTURE_HANDLE;
   // Only set boolean options that we need. Default is false, and we should be
   // able to safely ignore ones that our implementation doesn't care about.
   transport_options->wait_for_transfer_notification = true;
@@ -220,11 +224,17 @@
     input_active_state.controller_role = vr::TrackedControllerRole_Invalid;
   }
 
+  auto submit_frame_sink = device::mojom::XRPresentationConnection::New();
+  submit_frame_sink->provider = presentation_provider.PassInterface();
+  submit_frame_sink->client_request = mojo::MakeRequest(&submit_client_);
+  submit_frame_sink->transport_options = std::move(transport_options);
+
+  auto session = device::mojom::XRSession::New();
+  session->data_provider = frame_data_provider.PassInterface();
+  session->submit_frame_sink = std::move(submit_frame_sink);
+
   main_thread_task_runner_->PostTask(
-      FROM_HERE,
-      base::BindOnce(std::move(callback), true,
-                     mojo::MakeRequest(&submit_client_),
-                     provider.PassInterface(), std::move(transport_options)));
+      FROM_HERE, base::BindOnce(std::move(callback), true, std::move(session)));
   is_presenting_ = true;
   openvr_->GetCompositor()->SuspendRendering(false);
 
@@ -247,7 +257,8 @@
 
 void OpenVRRenderLoop::ExitPresent() {
   is_presenting_ = false;
-  binding_.Close();
+  presentation_binding_.Close();
+  frame_data_binding_.Close();
   submit_client_ = nullptr;
   if (openvr_)
     openvr_->GetCompositor()->SuspendRendering(true);
@@ -325,7 +336,7 @@
 }
 
 void OpenVRRenderLoop::GetFrameData(
-    mojom::VRPresentationProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   DCHECK(is_presenting_);
 
   if (has_outstanding_frame_) {
diff --git a/device/vr/openvr/openvr_render_loop.h b/device/vr/openvr/openvr_render_loop.h
index df7fdc3..a27f7f4 100644
--- a/device/vr/openvr/openvr_render_loop.h
+++ b/device/vr/openvr/openvr_render_loop.h
@@ -25,13 +25,12 @@
 class OpenVRWrapper;
 struct OpenVRGamepadState;
 
-class OpenVRRenderLoop : public base::Thread, mojom::VRPresentationProvider {
+class OpenVRRenderLoop : public base::Thread,
+                         mojom::XRPresentationProvider,
+                         mojom::XRFrameDataProvider {
  public:
   using RequestSessionCallback =
-      base::OnceCallback<void(bool result,
-                              mojom::VRSubmitFrameClientRequest,
-                              mojom::VRPresentationProviderPtrInfo,
-                              mojom::VRDisplayFrameTransportOptionsPtr)>;
+      base::OnceCallback<void(bool result, mojom::XRSessionPtr)>;
 
   OpenVRRenderLoop(
       base::RepeatingCallback<void(OpenVRGamepadState)> update_gamepad);
@@ -43,7 +42,7 @@
   void ExitPresent();
   base::WeakPtr<OpenVRRenderLoop> GetWeakPtr();
 
-  // VRPresentationProvider overrides:
+  // XRPresentationProvider overrides:
   void SubmitFrameMissing(int16_t frame_index, const gpu::SyncToken&) override;
   void SubmitFrame(int16_t frame_index,
                    const gpu::MailboxHolder& mailbox,
@@ -58,7 +57,7 @@
                          const gfx::RectF& right_bounds,
                          const gfx::Size& source_size) override;
   void GetFrameData(
-      VRPresentationProvider::GetFrameDataCallback callback) override;
+      XRFrameDataProvider::GetFrameDataCallback callback) override;
 
  private:
   // base::Thread overrides:
@@ -94,11 +93,12 @@
   gfx::RectF right_bounds_;
   gfx::Size source_size_;
   scoped_refptr<base::SingleThreadTaskRunner> main_thread_task_runner_;
-  mojom::VRSubmitFrameClientPtr submit_client_;
+  mojom::XRPresentationClientPtr submit_client_;
   base::RepeatingCallback<void(OpenVRGamepadState)> update_gamepad_;
   base::OnceCallback<void()> on_presentation_ended_;
   std::unique_ptr<OpenVRWrapper> openvr_;
-  mojo::Binding<mojom::VRPresentationProvider> binding_;
+  mojo::Binding<mojom::XRPresentationProvider> presentation_binding_;
+  mojo::Binding<mojom::XRFrameDataProvider> frame_data_binding_;
   base::WeakPtrFactory<OpenVRRenderLoop> weak_ptr_factory_;
 
   DISALLOW_COPY_AND_ASSIGN(OpenVRRenderLoop);
diff --git a/device/vr/orientation/orientation_device.cc b/device/vr/orientation/orientation_device.cc
index ee322dd..2de5b21 100644
--- a/device/vr/orientation/orientation_device.cc
+++ b/device/vr/orientation/orientation_device.cc
@@ -144,11 +144,11 @@
   DCHECK(!options->immersive);
   // TODO(offenwanger): Perform a check to see if sensors are available when
   // RequestSession is called for non-immersive sessions.
-  std::move(callback).Run(nullptr, nullptr);
+  ReturnNonImmersiveSession(std::move(callback));
 }
 
 void VROrientationDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   mojom::VRPosePtr pose = mojom::VRPose::New();
   pose->orientation.emplace(4);
 
diff --git a/device/vr/orientation/orientation_device.h b/device/vr/orientation/orientation_device.h
index f9f3b19..0ac5337 100644
--- a/device/vr/orientation/orientation_device.h
+++ b/device/vr/orientation/orientation_device.h
@@ -49,7 +49,7 @@
 
   // VRDeviceBase
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
 
   // Indicates whether the device was able to connect to orientation events.
   bool IsAvailable() const { return available_; }
diff --git a/device/vr/public/mojom/README.md b/device/vr/public/mojom/README.md
index 273b0d5e..dab1c92 100644
--- a/device/vr/public/mojom/README.md
+++ b/device/vr/public/mojom/README.md
@@ -6,6 +6,24 @@
 the browser process.  The ones that cannot live in the browser, are hosted in a
 service.
 
+# Supported Session types
+
+## Magic-window:
+Magic window sessions are requested by sites that request poses, but render
+through the normal Chrome compositor pipeline.
+It serves as a basic mode that requires only some way to get orientation poses.
+
+## Immersive:
+Immersive sessions are where the site wishes to request poses, then render
+content back to a display other than chrome. The common case for this is Head
+Mounted Displays (HMD), like Vive, Oculus, or Daydream.
+
+## Enviroment Integration
+This type of session allows for enviroment integration by providing functions
+that allow the site to query the enviroment, such as HitTest. A Enviroment
+Integration session may also supply data in addition to the pose, such as a
+camera frame.
+
 # Renderer <-> Browser interfaces (defined in vr_service.mojom)
 VRService - lives in the browser process, corresponds to a single frame.  Root
 object to obtain other XR objects.
@@ -24,23 +42,23 @@
 process.  They may live in the browser process or may live in the isolated
 service.
 
-## Presentation-related:
-Presentation is exclusive access to a headset where a site may display
-a stereo view to the user.
 
-VRPresentationProvider - lives in the XRDevice process.  Implements the details
+## Data related:
+All sessions need to be able to get data from a XR device.
+
+XRFrameDataProvider - lives in the XRDevice process.  Provides a way to obtain
+poses and other forms of data needed to render frames.
+
+## Presentation related:
+Presentation is exclusive access to a device, where the experience takes over
+the device's display, such as presenting a stereo view in an HMD.
+
+XRPresentationProvider - lives in the XRDevice process.  Implements the details
 for a presentation session, such as submitting frames to the underlying VR API.
 
-VRSubmitFrameClient - lives in the renderer process.  Is notified when various
+XRPresentationClient - lives in the renderer process.  Is notified when various
 rendering events occur, so it can reclaim/reuse textures.
 
-## Magic-window related:
-Magic window is a mode where a site may request poses, but renders through the
-normal Chrome compositor pipeline.
-
-VRMagicWindowProvider - lives in the XRDevice process.  Provides a way to obtain
-poses.
-
 # Browser <-> Device interfaces (defined in isolated_xr_service.mojom)
 The XRDevice process may be the browser process or an isolated service for
 different devices implementations.  A device provider in the browser will choose
diff --git a/device/vr/public/mojom/isolated_xr_service.mojom b/device/vr/public/mojom/isolated_xr_service.mojom
index a1bcacda..83320de 100644
--- a/device/vr/public/mojom/isolated_xr_service.mojom
+++ b/device/vr/public/mojom/isolated_xr_service.mojom
@@ -6,10 +6,10 @@
 
 import "device/vr/public/mojom/vr_service.mojom";
 
-// The XRSessionController lives in the vr device service, and corresponds to a
-// VRPresentationProvider or a VRMagicWindowProvider.  The client is the browser
-// process, which will pause or stop sessions depending events/state such as
-// focus or other tabs requesting presentation.
+// The XRSessionController lives in the vr device service, and corresponds to
+// a set of the XRSession bindings.  The client is the browser process, which
+// will pause or stop sessions depending events/state such as focus or other
+// tabs requesting immersive sessions.
 // Sessions are stopped by closing the mojo connection.
 interface XRSessionController {
   // A session may be paused temporarily for example when a non-presenting
@@ -58,22 +58,14 @@
 // browser process is the client, and may in turn expose device information to
 // render processes using vr_service interfaces, such as VRDisplayHost.
 interface XRRuntime {
-  // Attempt to start a presentation session.  Clients may submit graphics to be
-  // displayed in the headset.  Called by the browser process, but the
-  // VRPresentationProvider may be passed to the renderer process to allow
-  // submitting graphics without going through an extra IPC hop through the
-  // browser process.
+  // Attempt to start a session. Called by the browser process, but the result
+  // will probably be passed to the renderer process to allow getting data and
+  // possibly submitting graphics without going through an extra IPC hop through
+  // the browser process.
   RequestSession(XRDeviceRuntimeSessionOptions options) => (
-                     device.mojom.XRPresentationConnection? connection,
+                     XRSession? session,
                      XRSessionController? controller);
 
-  // Attempt to start a "magic window" session.  Magic window sessions allow
-  // Clients to obtain poses (device position and orientation), but rendering
-  // goes through the standard Chrome compositor.
-  RequestMagicWindowSession() =>
-      (device.mojom.VRMagicWindowProvider? session,
-       device.mojom.XRSessionController? controller);
-
   // The browser may register for changes to a device. Initial VRDisplayInfo
   // will immediately be returned to the listener to prevent races.
   ListenToDeviceChanges(XRRuntimeEventListener listener) =>
diff --git a/device/vr/public/mojom/vr_service.mojom b/device/vr/public/mojom/vr_service.mojom
index 108c4291..a00fd3c 100644
--- a/device/vr/public/mojom/vr_service.mojom
+++ b/device/vr/public/mojom/vr_service.mojom
@@ -41,17 +41,25 @@
   bool use_legacy_webvr_render_path;
 };
 
-// TODO(offenwanger) Rearrange these two interfaces to merge duplicate
-// functionality.
+// This structure contains all the mojo interfaces for different kinds of
+// XRSession. The only interface required by all sessions is the
+// XRFrameDataProvider. It must always be present. Other interfaces are set as
+// apropriate based on the session creation options. (for example, an immersive
+// session ought to have a XRPresentationConnection to submit the frames to the
+// immersive enviroment).
 struct XRSession {
-  VRMagicWindowProvider? magic_window_provider;
-  XRPresentationConnection? connection;
+  XRFrameDataProvider data_provider;
+  XRPresentationConnection? submit_frame_sink;
+  XREnviromentIntegrationProvider? enviroment_provider;
 };
 
+// This structure contains the infomation and interfaces needed to create a two
+// way connection between the renderer and a device to synchronize and submit
+// frames to a sink outside of Chrome.
 struct XRPresentationConnection {
-  VRSubmitFrameClient& client_request;
-  VRPresentationProvider provider;
-  VRDisplayFrameTransportOptions transport_options;
+  XRPresentationProvider provider;
+  XRPresentationClient& client_request;
+  XRPresentationTransportOptions transport_options;
 };
 
 struct XRInputSourceDescription {
@@ -171,7 +179,7 @@
 };
 
 // Frame transport method from the Renderer's point of view.
-enum VRDisplayFrameTransportMethod {
+enum XRPresentationTransportMethod {
   NONE = 0,
 
   // Renderer should create a new texture handle (Windows) or
@@ -185,10 +193,10 @@
   DRAW_INTO_TEXTURE_MAILBOX = 3,
 };
 
-struct VRDisplayFrameTransportOptions {
-  VRDisplayFrameTransportMethod transport_method;
+struct XRPresentationTransportOptions {
+  XRPresentationTransportMethod transport_method;
 
-  // Booleans indicating which of the VRSubmitFrameClient callbacks
+  // Booleans indicating which of the XRPresentationClient callbacks
   // are in use. Default is false, the device implementation should set
   // the ones to true that it needs and can ignore the rest.
   bool wait_for_transfer_notification;
@@ -249,40 +257,6 @@
                      VRDisplayInfo display_info);
 };
 
-// After submitting a frame, the VRPresentationProvider will notify the client
-// about several stages of the render pipeline.  This allows pipelining
-// efficiency.  Following VRPresentationProvider::Submit*, the submitted frame
-// will be transferred (read from, perhaps copied to another texture), and then
-// rendered (submitted to the underlying VR API).
-// The client lives in the render process, implemented by VRDisplay.
-//
-// See VRDisplayFrameTransportConfiguration which configures which of these
-// callbacks are in use.
-interface VRSubmitFrameClient {
-  // The VRPresentationProvider calls this to indicate that the submitted frame
-  // has been transferred, so the backing data (mailbox or GpuMemoryBuffer) can
-  // be reused or discarded.  Note that this is a convenience/optimization
-  // feature, not a security feature - if a site discards the data early we may
-  // drop a frame, but nothing will otherwise misbehave.
-  // When the frame wasn't successfully transferred, the client should create a
-  // new mailbox/GpuMemoryBuffer rather than reusing an existing one to recover
-  // for subsequent frames.
-  OnSubmitFrameTransferred(bool success);
-
-  // The VRPresentationProvider calls this after the frame was handed off to the
-  // underlying VR API. This allows some pipelining of CPU/GPU work, while
-  // delaying expensive tasks for a subsequent frame until the current frame has
-  // completed.
-  OnSubmitFrameRendered();
-
-  // This callback provides a GpuFence corresponding to the previous frame's
-  // rendering completion, intended for use with a server wait issued before
-  // the following wait to prevent its rendering work from competing with
-  // the previous frame.
-  OnSubmitFrameGpuFence(gfx.mojom.GpuFenceHandle gpu_fence_handle);
-};
-
-
 // Provides a communication channel from the renderer to the browser-side host
 // of a (device/) VrDisplayImpl.
 interface VRDisplayHost {
@@ -298,47 +272,44 @@
   ExitPresent();
 };
 
-// Provides the necessary functionality for a non-presenting WebXR session to
-// draw magic window content.
+// Provides the necessary functionality for a WebXR session to get data for
+// drawing frames. The kind of data it gets depends on what kind of session was
+// requested.
 // This interface is hosted in the Browser process, but will move to a sandboxed
 // utility process on Windows.  The render process communicates with it.
-// For AR displays (VRDisplayCapabilities.can_provide_pass_through_images
-// is true), clients can use GetFrameData to get images.
-// TODO(836478): rename VRMagicWindowProvider to NonImmersiveWindowProvider or
-// similar.
-interface VRMagicWindowProvider {
+interface XRFrameDataProvider {
   // frame_data is optional and will not be set if and only if the call fails
   // for some reason, such as device disconnection.
   GetFrameData() => (XRFrameData? frame_data);
+};
 
-  // Different devices can have different native orientations - 0
-  // is the native orientation, and then increments of 90 degrees
-  // from there.
+// Provides functionality for integrating enviroment information into an
+// XRSession. For example, some AR sessions would implement hit test to allow
+// developers to get the information about the world that its sensors supply.
+interface XREnviromentIntegrationProvider {
+  // Different devices can have different native orientations - 0 is the native
+  // orientation, and then increments of 90 degrees from there. Session geometry
+  // is needed by the device when integrating enviroment image data, i.e. camera
+  // feeds, into a session.
   UpdateSessionGeometry(
       gfx.mojom.Size frame_size,
       display.mojom.Rotation display_rotation);
 
-  // Performs a raycast into the magic window scene and returns a list of
-  // XRHitResults sorted from closest to furthest hit from the ray. Each
-  // hit result contains a hit_matrix containing the transform of the hit
-  // where the rotation represents the normal of the surface hit.
+  // Performs a raycast into the scene and returns a list of XRHitResults sorted
+  // from closest to furthest hit from the ray. Each hit result contains a
+  // hit_matrix containing the transform of the hit where the rotation
+  // represents the normal of the surface hit.
   // An empty result list means there were no hits. If a nullopt is returned,
   // there was an error.
-  // TODO(https://crbug.com/842025): have one "session" type, merging
-  // VRMagicWindowProvider and VRPresentationProvider because RequestHitTest
-  // makes sense for both types of sessions.
   RequestHitTest(XRRay ray) => (array<XRHitResult>? results);
 };
 
-// Provides the necessary functionality for a presenting WebVR page to draw
-// frames for a VrDisplay.
+// Provides the necessary functionality for sending frames to a headset.
 // This interface is hosted in the Browser process, but will move to a sandboxed
 // utility process on Windows.  The render process communicates with it.
-interface VRPresentationProvider {
-  // frame_data is optional and will not be set if and only if the call fails
-  // for some reason, such as device disconnection.
-  GetFrameData() => (XRFrameData? frame_data);
-
+interface XRPresentationProvider {
+  // This function tells the device which parts of the canvas should be rendered
+  // to which view.
   UpdateLayerBounds(int16 frame_id, gfx.mojom.RectF left_bounds,
                     gfx.mojom.RectF right_bounds, gfx.mojom.Size source_size);
 
@@ -346,26 +317,59 @@
   // ensure that every GetFrameData has a matching Submit call. This happens for
   // WebXR if there were no drawing operations to the opaque framebuffer, and
   // for WebVR 1.1 if the application didn't call SubmitFrame. Usable with any
-  // VRDisplayFrameTransportMethod. This path does *not* call the
+  // XRPresentationTransportMethod. This path does *not* call the
   // SubmitFrameClient methods such as OnSubmitFrameTransferred. This is
   // intended to help separate frames while presenting, it may or may not
   // be called for the last animating frame when presentation ends.
   SubmitFrameMissing(int16 frame_id, gpu.mojom.SyncToken sync_token);
 
-  // VRDisplayFrameTransportMethod SUBMIT_AS_MAILBOX_HOLDER
+  // XRPresentationTransportMethod SUBMIT_AS_MAILBOX_HOLDER
   SubmitFrame(int16 frame_id, gpu.mojom.MailboxHolder mailbox_holder,
               mojo_base.mojom.TimeDelta time_waited);
 
-  // VRDisplayFrameTransportMethod SUBMIT_AS_TEXTURE_HANDLE
+  // XRPresentationTransportMethod SUBMIT_AS_TEXTURE_HANDLE
   // TODO(https://crbug.com/676224): Support preprocessing of mojom files, since
   // this is Windows only.
   SubmitFrameWithTextureHandle(int16 frameId, handle texture);
 
-  // VRDisplayFrameTransportMethod DRAW_INTO_TEXTURE_MAILBOX
+  // XRPresentationTransportMethod DRAW_INTO_TEXTURE_MAILBOX
   SubmitFrameDrawnIntoTexture(int16 frameId, gpu.mojom.SyncToken sync_token,
                               mojo_base.mojom.TimeDelta time_waited);
 };
 
+// After submitting a frame, the XRPresentationProvider will notify the client
+// about several stages of the render pipeline.  This allows pipelining
+// efficiency.  Following XRPresentationProvider::Submit*, the submitted frame
+// will be transferred (read from, perhaps copied to another texture), and then
+// rendered (submitted to the underlying VR API).
+// The client lives in the render process.
+//
+// See XRPresentationTransportOptions which configures which of these
+// callbacks are in use.
+interface XRPresentationClient {
+  // The XRPresentationProvider calls this to indicate that the submitted frame
+  // has been transferred, so the backing data (mailbox or GpuMemoryBuffer) can
+  // be reused or discarded.  Note that this is a convenience/optimization
+  // feature, not a security feature - if a site discards the data early we may
+  // drop a frame, but nothing will otherwise misbehave.
+  // When the frame wasn't successfully transferred, the client should create a
+  // new mailbox/GpuMemoryBuffer rather than reusing an existing one to recover
+  // for subsequent frames.
+  OnSubmitFrameTransferred(bool success);
+
+  // The XRPresentationProvider calls this after the frame was handed off to the
+  // underlying VR API. This allows some pipelining of CPU/GPU work, while
+  // delaying expensive tasks for a subsequent frame until the current frame has
+  // completed.
+  OnSubmitFrameRendered();
+
+  // This callback provides a GpuFence corresponding to the previous frame's
+  // rendering completion, intended for use with a server wait issued before
+  // the following wait to prevent its rendering work from competing with
+  // the previous frame.
+  OnSubmitFrameGpuFence(gfx.mojom.GpuFenceHandle gpu_fence_handle);
+};
+
 interface VRDisplayClient {
   OnChanged(VRDisplayInfo display);
   OnExitPresent();
diff --git a/device/vr/test/fake_vr_device.cc b/device/vr/test/fake_vr_device.cc
index 4a5740d..626a2953 100644
--- a/device/vr/test/fake_vr_device.cc
+++ b/device/vr/test/fake_vr_device.cc
@@ -54,18 +54,9 @@
     mojom::XRDeviceRuntimeSessionOptionsPtr options,
     mojom::XRRuntime::RequestSessionCallback callback) {
   OnStartPresenting();
-
-  mojom::XRSessionControllerPtr exclusive_session_controller;
-  controller_binding_.Bind(mojo::MakeRequest(&exclusive_session_controller));
-
-  // Unretained is safe because the error handler won't be called after
-  // controller_binding_ is destroyed.
-  controller_binding_.set_connection_error_handler(
-      base::BindOnce(&FakeVRDevice::OnPresentingControllerMojoConnectionError,
-                     base::Unretained(this)));
-
-  std::move(callback).Run(mojom::XRPresentationConnection::New(),
-                          std::move(exclusive_session_controller));
+  // The current tests never use the return values, so it's fine to return
+  // invalid data here.
+  std::move(callback).Run(nullptr, nullptr);
 }
 
 void FakeVRDevice::OnPresentingControllerMojoConnectionError() {
@@ -74,7 +65,7 @@
 }
 
 void FakeVRDevice::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   mojom::XRFrameDataPtr frame_data = mojom::XRFrameData::New();
   frame_data->pose = pose_.Clone();
   std::move(callback).Run(std::move(frame_data));
diff --git a/device/vr/test/fake_vr_device.h b/device/vr/test/fake_vr_device.h
index cfe8601..4e7c70a 100644
--- a/device/vr/test/fake_vr_device.h
+++ b/device/vr/test/fake_vr_device.h
@@ -35,7 +35,7 @@
   void OnPresentingControllerMojoConnectionError();
 
   void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback) override;
 
   mojom::VRDisplayInfoPtr InitBasicDevice();
   mojom::VREyeParametersPtr InitEye(float fov, float offset, uint32_t size);
diff --git a/device/vr/test/mock_vr_display_impl.cc b/device/vr/test/mock_vr_display_impl.cc
deleted file mode 100644
index 4c63aeb..0000000
--- a/device/vr/test/mock_vr_display_impl.cc
+++ /dev/null
@@ -1,20 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "device/vr/test/mock_vr_display_impl.h"
-
-namespace device {
-
-MockVRDisplayImpl::MockVRDisplayImpl(
-    device::VRDeviceBase* device,
-    mojom::VRMagicWindowProviderRequest session,
-    mojom::XRSessionControllerRequest controller,
-    bool is_frame_focused)
-    : VRDisplayImpl(device, std::move(session), std::move(controller)) {
-  SetFrameDataRestricted(!is_frame_focused);
-}
-
-MockVRDisplayImpl::~MockVRDisplayImpl() = default;
-
-}  // namespace device
diff --git a/device/vr/test/mock_vr_display_impl.h b/device/vr/test/mock_vr_display_impl.h
deleted file mode 100644
index 32d0e1b8..0000000
--- a/device/vr/test/mock_vr_display_impl.h
+++ /dev/null
@@ -1,39 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef DEVICE_VR_TEST_MOCK_VR_DISPLAY_IMPL_H
-#define DEVICE_VR_TEST_MOCK_VR_DISPLAY_IMPL_H
-
-#include "base/macros.h"
-#include "device/vr/vr_display_impl.h"
-#include "testing/gmock/include/gmock/gmock.h"
-
-namespace device {
-
-class MockVRDisplayImpl : public VRDisplayImpl {
- public:
-  MockVRDisplayImpl(device::VRDeviceBase* device,
-                    mojom::VRMagicWindowProviderRequest session,
-                    mojom::XRSessionControllerRequest controller,
-                    bool in_frame_focused);
-  ~MockVRDisplayImpl() override;
-
-  MOCK_METHOD1(DoOnChanged, void(mojom::VRDisplayInfo* vr_device_info));
-  void OnChanged(mojom::VRDisplayInfoPtr vr_device_info) {
-    DoOnChanged(vr_device_info.get());
-  }
-
-  MOCK_METHOD2(OnActivate,
-               void(mojom::VRDisplayEventReason, base::Callback<void(bool)>));
-
-  MOCK_METHOD0(ListeningForActivate, bool());
-  MOCK_METHOD0(InFocusedFrame, bool());
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockVRDisplayImpl);
-};
-
-}  // namespace device
-
-#endif  // DEVICE_VR_TEST_MOCK_VR_DISPLAY_IMPL_H
diff --git a/device/vr/vr_device_base.cc b/device/vr/vr_device_base.cc
index 1c7703a..d5f3e5f 100644
--- a/device/vr/vr_device_base.cc
+++ b/device/vr/vr_device_base.cc
@@ -53,7 +53,7 @@
 }
 
 void VRDeviceBase::GetFrameData(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   if (!magic_window_enabled_) {
     std::move(callback).Run(nullptr);
     return;
@@ -95,7 +95,7 @@
 void VRDeviceBase::OnListeningForActivate(bool listening) {}
 
 void VRDeviceBase::OnMagicWindowFrameDataRequest(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   std::move(callback).Run(nullptr);
 }
 
@@ -105,18 +105,27 @@
 
 void VRDeviceBase::RequestHitTest(
     mojom::XRRayPtr ray,
-    mojom::VRMagicWindowProvider::RequestHitTestCallback callback) {
+    mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback) {
   NOTREACHED() << "Unexpected call to a device without hit-test support";
   std::move(callback).Run(base::nullopt);
 }
 
-void VRDeviceBase::RequestMagicWindowSession(
-    mojom::XRRuntime::RequestMagicWindowSessionCallback callback) {
-  mojom::VRMagicWindowProviderPtr provider;
+void VRDeviceBase::ReturnNonImmersiveSession(
+    mojom::XRRuntime::RequestSessionCallback callback) {
+  mojom::XRFrameDataProviderPtr data_provider;
+  mojom::XREnviromentIntegrationProviderPtr enviroment_provider;
   mojom::XRSessionControllerPtr controller;
   magic_window_sessions_.push_back(std::make_unique<VRDisplayImpl>(
-      this, mojo::MakeRequest(&provider), mojo::MakeRequest(&controller)));
-  std::move(callback).Run(std::move(provider), std::move(controller));
+      this, mojo::MakeRequest(&data_provider),
+      mojo::MakeRequest(&enviroment_provider), mojo::MakeRequest(&controller)));
+
+  auto session = mojom::XRSession::New();
+  session->data_provider = data_provider.PassInterface();
+  // TODO(offenwanger) Not all session will want the enviroment provider. This
+  // should be refactored so it's only passed when it's requested.
+  session->enviroment_provider = enviroment_provider.PassInterface();
+
+  std::move(callback).Run(std::move(session), std::move(controller));
 }
 
 void VRDeviceBase::EndMagicWindowSession(VRDisplayImpl* session) {
diff --git a/device/vr/vr_device_base.h b/device/vr/vr_device_base.h
index 04f6c506..230a0acf 100644
--- a/device/vr/vr_device_base.h
+++ b/device/vr/vr_device_base.h
@@ -31,12 +31,11 @@
       mojom::XRRuntime::ListenToDeviceChangesCallback callback) final;
   void SetListeningForActivate(bool is_listening) override;
 
-  void GetFrameData(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback);
+  void GetFrameData(mojom::XRFrameDataProvider::GetFrameDataCallback callback);
 
   virtual void RequestHitTest(
       mojom::XRRayPtr ray,
-      mojom::VRMagicWindowProvider::RequestHitTestCallback callback);
+      mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback);
   unsigned int GetId() const;
 
   bool HasExclusiveSession();
@@ -75,17 +74,16 @@
   void OnActivate(mojom::VRDisplayEventReason reason,
                   base::Callback<void(bool)> on_handled);
 
+  void ReturnNonImmersiveSession(
+      mojom::XRRuntime::RequestSessionCallback callback);
+
   std::vector<std::unique_ptr<VRDisplayImpl>> magic_window_sessions_;
 
  private:
   // TODO(https://crbug.com/842227): Rename methods to HandleOnXXX
   virtual void OnListeningForActivate(bool listening);
   virtual void OnMagicWindowFrameDataRequest(
-      mojom::VRMagicWindowProvider::GetFrameDataCallback callback);
-
-  // XRRuntime
-  void RequestMagicWindowSession(
-      mojom::XRRuntime::RequestMagicWindowSessionCallback callback) override;
+      mojom::XRFrameDataProvider::GetFrameDataCallback callback);
 
   mojom::XRRuntimeEventListenerPtr listener_;
 
diff --git a/device/vr/vr_device_base_unittest.cc b/device/vr/vr_device_base_unittest.cc
index 5025d3ed..10cdc517 100644
--- a/device/vr/vr_device_base_unittest.cc
+++ b/device/vr/vr_device_base_unittest.cc
@@ -11,8 +11,8 @@
 #include "device/vr/public/mojom/vr_service.mojom.h"
 #include "device/vr/test/fake_vr_device.h"
 #include "device/vr/test/fake_vr_service_client.h"
-#include "device/vr/test/mock_vr_display_impl.h"
 #include "device/vr/vr_device_base.h"
+#include "testing/gmock/include/gmock/gmock.h"
 #include "testing/gtest/include/gtest/gtest.h"
 
 namespace device {
@@ -95,14 +95,6 @@
     client_ = std::make_unique<FakeVRServiceClient>(mojo::MakeRequest(&proxy));
   }
 
-  std::unique_ptr<MockVRDisplayImpl> MakeMockDisplay(VRDeviceBase* device) {
-    mojom::VRMagicWindowProviderPtr session;
-    mojom::XRSessionControllerPtr controller;
-    return std::make_unique<testing::NiceMock<MockVRDisplayImpl>>(
-        device, mojo::MakeRequest(&session), mojo::MakeRequest(&controller),
-        false);
-  }
-
   std::unique_ptr<VRDeviceBaseForTesting> MakeVRDevice() {
     std::unique_ptr<VRDeviceBaseForTesting> device =
         std::make_unique<VRDeviceBaseForTesting>();
diff --git a/device/vr/vr_display_impl.cc b/device/vr/vr_display_impl.cc
index b4e482b..d6b0ce5 100644
--- a/device/vr/vr_display_impl.cc
+++ b/device/vr/vr_display_impl.cc
@@ -17,14 +17,13 @@
 
 VRDisplayImpl::VRDisplayImpl(
     VRDeviceBase* device,
-    mojom::VRMagicWindowProviderRequest magic_window_request,
+    mojom::XRFrameDataProviderRequest magic_window_request,
+    mojom::XREnviromentIntegrationProviderRequest enviroment_request,
     mojom::XRSessionControllerRequest session_request)
-    : magic_window_binding_(this),
-      session_controller_binding_(this),
+    : magic_window_binding_(this, std::move(magic_window_request)),
+      enviroment_binding_(this, std::move(enviroment_request)),
+      session_controller_binding_(this, std::move(session_request)),
       device_(device) {
-  magic_window_binding_.Bind(std::move(magic_window_request));
-  session_controller_binding_.Bind(std::move(session_request));
-
   // Unretained is safe because the binding will close when we are destroyed,
   // so we won't receive any more callbacks after that.
   session_controller_binding_.set_connection_error_handler(base::BindOnce(
@@ -35,7 +34,7 @@
 
 // Gets frame data for sessions.
 void VRDisplayImpl::GetFrameData(
-    mojom::VRMagicWindowProvider::GetFrameDataCallback callback) {
+    mojom::XRFrameDataProvider::GetFrameDataCallback callback) {
   if (device_->HasExclusiveSession() || restrict_frame_data_) {
     std::move(callback).Run(nullptr);
     return;
@@ -63,7 +62,7 @@
 
 void VRDisplayImpl::RequestHitTest(
     mojom::XRRayPtr ray,
-    mojom::VRMagicWindowProvider::RequestHitTestCallback callback) {
+    mojom::XREnviromentIntegrationProvider::RequestHitTestCallback callback) {
   if (restrict_frame_data_) {
     std::move(callback).Run(base::nullopt);
     return;
diff --git a/device/vr/vr_display_impl.h b/device/vr/vr_display_impl.h
index 0d0bd8f..980798e 100644
--- a/device/vr/vr_display_impl.h
+++ b/device/vr/vr_display_impl.h
@@ -20,15 +20,18 @@
 
 class VRDeviceBase;
 
-// VR device process implementation of a VRMagicWindowProvider within a WebVR
+// VR device process implementation of a XRFrameDataProvider within a WebVR
 // or WebXR site session.
 // VRDisplayImpl objects are owned by their respective XRRuntime instances.
 // TODO(offenwanger): Rename this.
-class DEVICE_VR_EXPORT VRDisplayImpl : public mojom::VRMagicWindowProvider,
-                                       public mojom::XRSessionController {
+class DEVICE_VR_EXPORT VRDisplayImpl
+    : public mojom::XRFrameDataProvider,
+      public mojom::XREnviromentIntegrationProvider,
+      public mojom::XRSessionController {
  public:
   VRDisplayImpl(VRDeviceBase* device,
-                mojom::VRMagicWindowProviderRequest,
+                mojom::XRFrameDataProviderRequest,
+                mojom::XREnviromentIntegrationProviderRequest,
                 mojom::XRSessionControllerRequest);
   ~VRDisplayImpl() override;
 
@@ -39,7 +42,7 @@
 
   // Accessible to tests.
  protected:
-  // mojom::VRMagicWindowProvider
+  // mojom::XRFrameDataProvider
   void GetFrameData(GetFrameDataCallback callback) override;
   void UpdateSessionGeometry(const gfx::Size& frame_size,
                              display::Display::Rotation rotation) override;
@@ -51,7 +54,8 @@
 
   void OnMojoConnectionError();
 
-  mojo::Binding<mojom::VRMagicWindowProvider> magic_window_binding_;
+  mojo::Binding<mojom::XRFrameDataProvider> magic_window_binding_;
+  mojo::Binding<mojom::XREnviromentIntegrationProvider> enviroment_binding_;
   mojo::Binding<mojom::XRSessionController> session_controller_binding_;
   device::VRDeviceBase* device_;
   bool restrict_frame_data_ = true;
diff --git a/device/vr/vr_display_impl_unittest.cc b/device/vr/vr_display_impl_unittest.cc
index 6735be75..d4de7de 100644
--- a/device/vr/vr_display_impl_unittest.cc
+++ b/device/vr/vr_display_impl_unittest.cc
@@ -21,13 +21,6 @@
   VRDisplayImplTest() {}
   ~VRDisplayImplTest() override {}
   void onDisplaySynced() {}
-  void onPresentComplete(
-      device::mojom::XRPresentationConnectionPtr connection,
-      mojom::XRSessionControllerPtr immersive_session_controller) {
-    is_request_presenting_success_ = connection ? true : false;
-
-    immersive_session_controller_ = std::move(immersive_session_controller);
-  }
 
  protected:
   void SetUp() override {
@@ -39,9 +32,11 @@
 
   std::unique_ptr<VRDisplayImpl> MakeDisplay(
       mojom::XRSessionControllerPtr* controller) {
-    mojom::VRMagicWindowProviderPtr session;
+    mojom::XRFrameDataProviderPtr data_provider;
+    mojom::XREnviromentIntegrationProviderPtr enviroment_provider;
     auto display = std::make_unique<VRDisplayImpl>(
-        device(), mojo::MakeRequest(&session), mojo::MakeRequest(controller));
+        device(), mojo::MakeRequest(&data_provider),
+        mojo::MakeRequest(&enviroment_provider), mojo::MakeRequest(controller));
     static_cast<mojom::XRSessionController*>(display.get())
         ->SetFrameDataRestricted(true);
     return display;
@@ -50,13 +45,13 @@
   void RequestSession(VRDisplayImpl* display_impl) {
     device_->RequestSession(
         mojom::XRDeviceRuntimeSessionOptionsPtr(),
-        base::BindOnce(&VRDisplayImplTest::onPresentComplete,
-                       base::Unretained(this)));
+        base::BindOnce(
+            [](device::mojom::XRSessionPtr session,
+               mojom::XRSessionControllerPtr immersive_session_controller) {}));
   }
 
   void ExitPresent() {
     device_->StopSession();
-    immersive_session_controller_ = nullptr;
   }
 
   bool presenting() { return device_->IsPresenting(); }
@@ -64,10 +59,8 @@
   FakeVRServiceClient* client() { return client_.get(); }
 
   base::MessageLoop message_loop_;
-  bool is_request_presenting_success_ = false;
   std::unique_ptr<FakeVRDevice> device_;
   std::unique_ptr<FakeVRServiceClient> client_;
-  mojom::XRSessionControllerPtr immersive_session_controller_;
 
   DISALLOW_COPY_AND_ASSIGN(VRDisplayImplTest);
 };
@@ -77,6 +70,7 @@
   std::unique_ptr<VRDisplayImpl> display_1 = MakeDisplay(&controller1);
   static_cast<mojom::XRSessionController*>(display_1.get())
       ->SetFrameDataRestricted(false);
+
   mojom::XRSessionControllerPtr controller2;
   std::unique_ptr<VRDisplayImpl> display_2 = MakeDisplay(&controller2);
   static_cast<mojom::XRSessionController*>(display_2.get())
@@ -92,31 +86,36 @@
     EXPECT_EQ(expect_null, !data);
   };
 
-  static_cast<mojom::VRMagicWindowProvider*>(display_1.get())
+  static_cast<mojom::XRFrameDataProvider*>(display_1.get())
       ->GetFrameData(base::BindOnce(callback, false, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
-  static_cast<mojom::VRMagicWindowProvider*>(display_2.get())
+
+  static_cast<mojom::XRFrameDataProvider*>(display_2.get())
       ->GetFrameData(base::BindOnce(callback, false, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
 
   // Attempt to present.
   RequestSession(display_1.get());
-  EXPECT_TRUE(is_request_presenting_success_);
   EXPECT_TRUE(presenting());
   EXPECT_TRUE(device()->HasExclusiveSession());
 
-  // While a device is presenting, noone should have access to magic window.
-  static_cast<mojom::VRMagicWindowProvider*>(display_1.get())
+  // While a device is presenting, no one should have access to magic window.
+  static_cast<mojom::XRFrameDataProvider*>(display_1.get())
       ->GetFrameData(base::BindOnce(callback, true, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
-  static_cast<mojom::VRMagicWindowProvider*>(display_2.get())
+
+  static_cast<mojom::XRFrameDataProvider*>(display_2.get())
       ->GetFrameData(base::BindOnce(callback, true, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
@@ -128,13 +127,16 @@
 
   // Once presentation had ended both services should be able to access the
   // device.
-  static_cast<mojom::VRMagicWindowProvider*>(display_1.get())
+  static_cast<mojom::XRFrameDataProvider*>(display_1.get())
       ->GetFrameData(base::BindOnce(callback, false, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
-  static_cast<mojom::VRMagicWindowProvider*>(display_2.get())
+
+  static_cast<mojom::XRFrameDataProvider*>(display_2.get())
       ->GetFrameData(base::BindOnce(callback, false, &was_called));
+
   base::RunLoop().RunUntilIdle();
   EXPECT_TRUE(was_called);
   was_called = false;
diff --git a/extensions/browser/api/socket/BUILD.gn b/extensions/browser/api/socket/BUILD.gn
index a707906e..e20046a05 100644
--- a/extensions/browser/api/socket/BUILD.gn
+++ b/extensions/browser/api/socket/BUILD.gn
@@ -9,6 +9,8 @@
 
 source_set("socket") {
   sources = [
+    "mojo_data_pump.cc",
+    "mojo_data_pump.h",
     "socket.cc",
     "socket.h",
     "socket_api.cc",
diff --git a/extensions/browser/api/socket/mojo_data_pump.cc b/extensions/browser/api/socket/mojo_data_pump.cc
new file mode 100644
index 0000000..fdc22f73
--- /dev/null
+++ b/extensions/browser/api/socket/mojo_data_pump.cc
@@ -0,0 +1,117 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "extensions/browser/api/socket/mojo_data_pump.h"
+
+#include <utility>
+
+#include "base/memory/ptr_util.h"
+#include "net/base/completion_callback.h"
+#include "net/base/io_buffer.h"
+#include "net/base/net_errors.h"
+
+namespace extensions {
+
+MojoDataPump::MojoDataPump(mojo::ScopedDataPipeConsumerHandle receive_stream,
+                           mojo::ScopedDataPipeProducerHandle send_stream)
+    : receive_stream_(std::move(receive_stream)),
+      receive_stream_watcher_(FROM_HERE,
+                              mojo::SimpleWatcher::ArmingPolicy::MANUAL),
+      send_stream_(std::move(send_stream)),
+      send_stream_watcher_(FROM_HERE,
+                           mojo::SimpleWatcher::ArmingPolicy::MANUAL) {
+  DCHECK(receive_stream_.is_valid());
+  DCHECK(send_stream_.is_valid());
+  receive_stream_watcher_.Watch(
+      receive_stream_.get(),
+      MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED,
+      MOJO_TRIGGER_CONDITION_SIGNALS_SATISFIED,
+      base::BindRepeating(&MojoDataPump::ReceiveMore, base::Unretained(this)));
+  send_stream_watcher_.Watch(
+      send_stream_.get(),
+      MOJO_HANDLE_SIGNAL_WRITABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED,
+      MOJO_TRIGGER_CONDITION_SIGNALS_SATISFIED,
+      base::BindRepeating(&MojoDataPump::SendMore, base::Unretained(this)));
+}
+
+MojoDataPump::~MojoDataPump() {}
+
+void MojoDataPump::Read(int count, ReadCallback callback) {
+  DCHECK(callback);
+  DCHECK(!read_callback_);
+
+  if (count <= 0) {
+    std::move(callback).Run(net::ERR_INVALID_ARGUMENT, nullptr);
+    return;
+  }
+
+  read_size_ = count;
+  read_callback_ = std::move(callback);
+  receive_stream_watcher_.ArmOrNotify();
+}
+
+void MojoDataPump::Write(net::IOBuffer* io_buffer,
+                         int io_buffer_size,
+                         net::CompletionOnceCallback callback) {
+  DCHECK(callback);
+  DCHECK(!write_callback_);
+
+  write_callback_ = std::move(callback);
+  pending_write_buffer_ = io_buffer;
+  pending_write_buffer_size_ = io_buffer_size;
+
+  send_stream_watcher_.ArmOrNotify();
+}
+
+void MojoDataPump::ReceiveMore(MojoResult result,
+                               const mojo::HandleSignalsState& /* ignored */) {
+  DCHECK(read_callback_);
+  DCHECK_NE(0u, read_size_);
+
+  if (result != MOJO_RESULT_OK) {
+    read_size_ = 0;
+    std::move(read_callback_).Run(net::ERR_FAILED, nullptr);
+    return;
+  }
+
+  uint32_t num_bytes = read_size_;
+  auto io_buffer =
+      base::MakeRefCounted<net::IOBuffer>(static_cast<size_t>(num_bytes));
+  result = receive_stream_->ReadData(io_buffer->data(), &num_bytes,
+                                     MOJO_READ_DATA_FLAG_NONE);
+  if (result != MOJO_RESULT_OK) {
+    read_size_ = 0;
+    std::move(read_callback_).Run(net::ERR_FAILED, nullptr);
+    return;
+  }
+
+  if (result == MOJO_RESULT_SHOULD_WAIT) {
+    receive_stream_watcher_.ArmOrNotify();
+    return;
+  }
+  read_size_ = 0;
+  std::move(read_callback_).Run(num_bytes, io_buffer);
+}
+
+void MojoDataPump::SendMore(MojoResult result,
+                            const mojo::HandleSignalsState& state) {
+  DCHECK(write_callback_);
+
+  uint32_t num_bytes = pending_write_buffer_size_;
+  result = send_stream_->WriteData(pending_write_buffer_->data(), &num_bytes,
+                                   MOJO_WRITE_DATA_FLAG_NONE);
+  if (result == MOJO_RESULT_SHOULD_WAIT) {
+    send_stream_watcher_.ArmOrNotify();
+    return;
+  }
+  pending_write_buffer_ = nullptr;
+  pending_write_buffer_size_ = 0;
+  if (result != MOJO_RESULT_OK) {
+    std::move(write_callback_).Run(net::ERR_FAILED);
+    return;
+  }
+  std::move(write_callback_).Run(num_bytes);
+}
+
+}  // namespace extensions
diff --git a/extensions/browser/api/socket/mojo_data_pump.h b/extensions/browser/api/socket/mojo_data_pump.h
new file mode 100644
index 0000000..f4f6876
--- /dev/null
+++ b/extensions/browser/api/socket/mojo_data_pump.h
@@ -0,0 +1,71 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef EXTENSIONS_BROWSER_API_SOCKET_MOJO_DATA_PUMP_H_
+#define EXTENSIONS_BROWSER_API_SOCKET_MOJO_DATA_PUMP_H_
+
+#include "base/macros.h"
+#include "base/memory/ref_counted.h"
+#include "extensions/browser/api/socket/socket.h"
+#include "mojo/public/cpp/system/data_pipe.h"
+#include "mojo/public/cpp/system/simple_watcher.h"
+#include "net/base/completion_once_callback.h"
+
+namespace net {
+class IOBuffer;
+};
+
+namespace extensions {
+
+// Helper class to read from a mojo consumer handle and write to mojo producer
+// handle.
+class MojoDataPump {
+ public:
+  using ReadCallback =
+      base::OnceCallback<void(int, scoped_refptr<net::IOBuffer> io_buffer)>;
+
+  MojoDataPump(mojo::ScopedDataPipeConsumerHandle receive_stream,
+               mojo::ScopedDataPipeProducerHandle send_stream);
+
+  ~MojoDataPump();
+
+  // Reads from |receive_stream|. It's illegal to call Read() when a previous
+  // one hasn't completed yet.
+  void Read(int count, ReadCallback callback);
+
+  // Writes to |send_stream|. It's illegal to call Write() when a previous one
+  // hasn't completed yet.
+  void Write(net::IOBuffer* io_buffer,
+             int io_buffer_size,
+             net::CompletionOnceCallback callback);
+
+  // Returns whether a read is pending.
+  bool HasPendingRead() const { return !read_callback_.is_null(); }
+
+  // Returns whether a write is pending.
+  bool HasPendingWrite() const { return !write_callback_.is_null(); }
+
+ private:
+  void OnReadComplete(scoped_refptr<net::IOBuffer> io_buffer, int result);
+  void StartWatching();
+  void ReceiveMore(MojoResult result, const mojo::HandleSignalsState& state);
+  void SendMore(MojoResult result, const mojo::HandleSignalsState& state);
+
+  mojo::ScopedDataPipeConsumerHandle receive_stream_;
+  mojo::SimpleWatcher receive_stream_watcher_;
+  mojo::ScopedDataPipeProducerHandle send_stream_;
+  mojo::SimpleWatcher send_stream_watcher_;
+
+  ReadCallback read_callback_;
+  net::CompletionOnceCallback write_callback_;
+  scoped_refptr<net::IOBuffer> pending_write_buffer_;
+  int pending_write_buffer_size_ = 0;
+  uint32_t read_size_ = 0;
+
+  DISALLOW_COPY_AND_ASSIGN(MojoDataPump);
+};
+
+}  //  namespace extensions
+
+#endif  // EXTENSIONS_BROWSER_API_SOCKET_MOJO_DATA_PUMP_H_
diff --git a/extensions/browser/api/socket/socket.cc b/extensions/browser/api/socket/socket.cc
index 74f1132d..23dac927 100644
--- a/extensions/browser/api/socket/socket.cc
+++ b/extensions/browser/api/socket/socket.cc
@@ -85,20 +85,27 @@
     WriteData();
 }
 
-bool Socket::SetKeepAlive(bool enable, int delay) { return false; }
-
-bool Socket::SetNoDelay(bool no_delay) { return false; }
-
-int Socket::Listen(const std::string& address,
-                   uint16_t port,
-                   int backlog,
-                   std::string* error_msg) {
-  *error_msg = kSocketTypeNotSupported;
-  return net::ERR_FAILED;
+void Socket::SetKeepAlive(bool enable,
+                          int delay,
+                          SetKeepAliveCallback callback) {
+  std::move(callback).Run(false);
 }
 
-void Socket::Accept(const AcceptCompletionCallback& callback) {
-  callback.Run(net::ERR_FAILED, NULL);
+void Socket::SetNoDelay(bool no_delay, SetNoDelayCallback callback) {
+  std::move(callback).Run(false);
+}
+
+void Socket::Listen(const std::string& address,
+                    uint16_t port,
+                    int backlog,
+                    ListenCallback callback) {
+  std::move(callback).Run(net::ERR_FAILED, kSocketTypeNotSupported);
+}
+
+void Socket::Accept(AcceptCompletionCallback callback) {
+  std::move(callback).Run(net::ERR_FAILED, nullptr /* socket */, base::nullopt,
+                          mojo::ScopedDataPipeConsumerHandle(),
+                          mojo::ScopedDataPipeProducerHandle());
 }
 
 // static
diff --git a/extensions/browser/api/socket/socket.h b/extensions/browser/api/socket/socket.h
index d83b94c..a55eede 100644
--- a/extensions/browser/api/socket/socket.h
+++ b/extensions/browser/api/socket/socket.h
@@ -7,6 +7,7 @@
 
 #include <stdint.h>
 
+#include <memory>
 #include <string>
 #include <utility>
 
@@ -17,11 +18,13 @@
 #include "content/public/browser/browser_thread.h"
 #include "extensions/browser/api/api_resource.h"
 #include "extensions/browser/api/api_resource_manager.h"
-#include "net/base/completion_callback.h"
+#include "net/base/completion_once_callback.h"
 #include "net/base/io_buffer.h"
 #include "net/base/ip_endpoint.h"
 #include "net/socket/tcp_client_socket.h"
 #include "net/traffic_annotation/network_traffic_annotation.h"
+#include "services/network/public/mojom/tcp_socket.mojom.h"
+#include "services/network/public/mojom/tls_socket.mojom.h"
 
 #if defined(OS_CHROMEOS)
 #include "extensions/browser/api/socket/app_firewall_hole_manager.h"
@@ -36,7 +39,9 @@
 namespace extensions {
 
 using CompletionCallback = base::Callback<void(int)>;
-using ReadCompletionCallback = base::Callback<
+using SetNoDelayCallback = base::OnceCallback<void(bool)>;
+using SetKeepAliveCallback = base::OnceCallback<void(bool)>;
+using ReadCompletionCallback = base::OnceCallback<
     void(int, scoped_refptr<net::IOBuffer> io_buffer, bool socket_destroying)>;
 using RecvFromCompletionCallback =
     base::Callback<void(int,
@@ -44,8 +49,15 @@
                         bool socket_destroying,
                         const std::string&,
                         uint16_t)>;
+using ListenCallback =
+    base::OnceCallback<void(int, const std::string& error_msg)>;
+
 using AcceptCompletionCallback =
-    base::Callback<void(int, std::unique_ptr<net::TCPClientSocket>)>;
+    base::OnceCallback<void(int,
+                            network::mojom::TCPConnectedSocketPtr,
+                            const base::Optional<net::IPEndPoint>&,
+                            mojo::ScopedDataPipeConsumerHandle,
+                            mojo::ScopedDataPipeProducerHandle)>;
 
 // A Socket wraps a low-level socket and includes housekeeping information that
 // we need to manage it in the context of an extension.
@@ -78,7 +90,7 @@
   // the remote endpoint. In order to upgrade this socket to TLS, callers
   // must also supply the hostname of the endpoint via set_hostname().
   virtual void Connect(const net::AddressList& address,
-                       const CompletionCallback& callback) = 0;
+                       net::CompletionOnceCallback callback) = 0;
   // |socket_destroying| is true if disconnect is due to destruction of the
   // socket.
   virtual void Disconnect(bool socket_destroying) = 0;
@@ -88,7 +100,7 @@
 
   // The |callback| will be called with the number of bytes read into the
   // buffer, or a negative number if an error occurred.
-  virtual void Read(int count, const ReadCompletionCallback& callback) = 0;
+  virtual void Read(int count, ReadCompletionCallback callback) = 0;
 
   // The |callback| will be called with |byte_count| or a negative number if an
   // error occurred.
@@ -103,13 +115,15 @@
                       const net::IPEndPoint& address,
                       const CompletionCallback& callback) = 0;
 
-  virtual bool SetKeepAlive(bool enable, int delay);
-  virtual bool SetNoDelay(bool no_delay);
-  virtual int Listen(const std::string& address,
-                     uint16_t port,
-                     int backlog,
-                     std::string* error_msg);
-  virtual void Accept(const AcceptCompletionCallback& callback);
+  virtual void SetKeepAlive(bool enable,
+                            int delay,
+                            SetKeepAliveCallback callback);
+  virtual void SetNoDelay(bool no_delay, SetNoDelayCallback callback);
+  virtual void Listen(const std::string& address,
+                      uint16_t port,
+                      int backlog,
+                      ListenCallback callback);
+  virtual void Accept(AcceptCompletionCallback callback);
 
   virtual bool IsConnected() = 0;
 
diff --git a/extensions/browser/api/socket/socket_api.cc b/extensions/browser/api/socket/socket_api.cc
index 8f7c8cf..3d33656 100644
--- a/extensions/browser/api/socket/socket_api.cc
+++ b/extensions/browser/api/socket/socket_api.cc
@@ -4,7 +4,6 @@
 
 #include "extensions/browser/api/socket/socket_api.h"
 
-#include <memory>
 #include <utility>
 #include <vector>
 
@@ -254,7 +253,7 @@
 void SocketCreateFunction::Work() {
   Socket* socket = NULL;
   if (socket_type_ == kSocketTypeTCP) {
-    socket = new TCPSocket(extension_->id());
+    socket = new TCPSocket(browser_context(), extension_->id());
   } else if (socket_type_ == kSocketTypeUDP) {
     socket =
         new UDPSocket(std::move(socket_), std::move(socket_receiver_request_),
@@ -463,15 +462,27 @@
     return;
   }
 
-  int result =
-      socket->Listen(params_->address, params_->port,
-                     params_->backlog.get() ? *params_->backlog : 5, &error_);
-  SetResult(std::make_unique<base::Value>(result));
-  if (result != net::OK) {
+  socket->Listen(params_->address, params_->port,
+                 params_->backlog.get() ? *params_->backlog : 5,
+                 base::BindOnce(&SocketListenFunction::OnCompleted, this));
+}
+
+void SocketListenFunction::OnCompleted(int result,
+                                       const std::string& error_msg) {
+  DCHECK_NE(net::ERR_IO_PENDING, result);
+  Socket* socket = GetSocket(params_->socket_id);
+  if (!socket) {
+    error_ = kSocketNotFoundError;
+    SetResult(std::make_unique<base::Value>(-1));
     AsyncWorkCompleted();
     return;
   }
-
+  SetResult(std::make_unique<base::Value>(result));
+  if (result != net::OK) {
+    error_ = error_msg;
+    AsyncWorkCompleted();
+    return;
+  }
   OpenFirewallHole(params_->address, params_->socket_id, socket);
 }
 
@@ -488,21 +499,27 @@
 void SocketAcceptFunction::AsyncWorkStart() {
   Socket* socket = GetSocket(params_->socket_id);
   if (socket) {
-    socket->Accept(base::Bind(&SocketAcceptFunction::OnAccept, this));
+    socket->Accept(base::BindOnce(&SocketAcceptFunction::OnAccept, this));
   } else {
     error_ = kSocketNotFoundError;
-    OnAccept(-1, NULL);
+    OnAccept(net::ERR_FAILED, nullptr, base::nullopt,
+             mojo::ScopedDataPipeConsumerHandle(),
+             mojo::ScopedDataPipeProducerHandle());
   }
 }
 
 void SocketAcceptFunction::OnAccept(
     int result_code,
-    std::unique_ptr<net::TCPClientSocket> socket) {
+    network::mojom::TCPConnectedSocketPtr socket,
+    const base::Optional<net::IPEndPoint>& remote_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+    mojo::ScopedDataPipeProducerHandle send_pipe_handle) {
   std::unique_ptr<base::DictionaryValue> result(new base::DictionaryValue());
   result->SetInteger(kResultCodeKey, result_code);
-  if (socket) {
+  if (result_code == net::OK) {
     Socket* client_socket =
-        new TCPSocket(std::move(socket), extension_id(), true);
+        new TCPSocket(std::move(socket), std::move(receive_pipe_handle),
+                      std::move(send_pipe_handle), remote_addr, extension_id());
     result->SetInteger(kSocketIdKey, AddSocket(client_socket));
   }
   SetResult(std::move(result));
@@ -529,7 +546,7 @@
   }
 
   socket->Read(params_->buffer_size.get() ? *params_->buffer_size : 4096,
-               base::BindRepeating(&SocketReadFunction::OnCompleted, this));
+               base::BindOnce(&SocketReadFunction::OnCompleted, this));
 }
 
 void SocketReadFunction::OnCompleted(int bytes_read,
@@ -731,18 +748,25 @@
   return true;
 }
 
-void SocketSetKeepAliveFunction::Work() {
-  bool result = false;
+void SocketSetKeepAliveFunction::AsyncWorkStart() {
   Socket* socket = GetSocket(params_->socket_id);
-  if (socket) {
-    int delay = 0;
-    if (params_->delay.get())
-      delay = *params_->delay;
-    result = socket->SetKeepAlive(params_->enable, delay);
-  } else {
+  if (!socket) {
+    SetResult(std::make_unique<base::Value>(false));
     error_ = kSocketNotFoundError;
+    AsyncWorkCompleted();
+    return;
   }
-  SetResult(std::make_unique<base::Value>(result));
+  int delay = 0;
+  if (params_->delay.get())
+    delay = *params_->delay;
+  socket->SetKeepAlive(
+      params_->enable, delay,
+      base::BindOnce(&SocketSetKeepAliveFunction::OnCompleted, this));
+}
+
+void SocketSetKeepAliveFunction::OnCompleted(bool success) {
+  SetResult(std::make_unique<base::Value>(success));
+  AsyncWorkCompleted();
 }
 
 SocketSetNoDelayFunction::SocketSetNoDelayFunction() {}
@@ -755,14 +779,22 @@
   return true;
 }
 
-void SocketSetNoDelayFunction::Work() {
-  bool result = false;
+void SocketSetNoDelayFunction::AsyncWorkStart() {
   Socket* socket = GetSocket(params_->socket_id);
-  if (socket)
-    result = socket->SetNoDelay(params_->no_delay);
-  else
+  if (!socket) {
     error_ = kSocketNotFoundError;
-  SetResult(std::make_unique<base::Value>(result));
+    SetResult(std::make_unique<base::Value>(false));
+    AsyncWorkCompleted();
+    return;
+  }
+  socket->SetNoDelay(
+      params_->no_delay,
+      base::BindOnce(&SocketSetNoDelayFunction::OnCompleted, this));
+}
+
+void SocketSetNoDelayFunction::OnCompleted(bool success) {
+  SetResult(std::make_unique<base::Value>(success));
+  AsyncWorkCompleted();
 }
 
 SocketGetInfoFunction::SocketGetInfoFunction() {}
@@ -1084,8 +1116,6 @@
   DCHECK_CURRENTLY_ON(BrowserThread::UI);
   params_ = api::socket::Secure::Params::Create(*args_);
   EXTENSION_FUNCTION_VALIDATE(params_.get());
-  url_request_getter_ = content::BrowserContext::GetDefaultStoragePartition(
-      browser_context())->GetURLRequestContext();
   return true;
 }
 
@@ -1103,8 +1133,7 @@
   }
 
   // Make sure that the socket is a TCP client socket.
-  if (socket->GetSocketType() != Socket::TYPE_TCP ||
-      static_cast<TCPSocket*>(socket)->ClientStream() == NULL) {
+  if (socket->GetSocketType() != Socket::TYPE_TCP) {
     SetResult(std::make_unique<base::Value>(net::ERR_INVALID_ARGUMENT));
     error_ = kSecureSocketTypeError;
     AsyncWorkCompleted();
@@ -1118,31 +1147,32 @@
     return;
   }
 
-  net::URLRequestContext* url_request_context =
-      url_request_getter_->GetURLRequestContext();
-
-  TLSSocket::UpgradeSocketToTLS(
-      socket, url_request_context->ssl_config_service(),
-      url_request_context->cert_verifier(),
-      url_request_context->transport_security_state(),
-      url_request_context->cert_transparency_verifier(),
-      url_request_context->ct_policy_enforcer(), extension_id(),
+  TCPSocket* tcp_socket = static_cast<TCPSocket*>(socket);
+  tcp_socket->UpgradeToTLS(
       params_->options.get(),
-      base::Bind(&SocketSecureFunction::TlsConnectDone, this));
+      base::BindOnce(&SocketSecureFunction::TlsConnectDone, this));
 }
 
-void SocketSecureFunction::TlsConnectDone(std::unique_ptr<TLSSocket> socket,
-                                          int result) {
-  // if an error occurred, socket MUST be NULL.
-  DCHECK(result == net::OK || socket == NULL);
-
-  if (socket && result == net::OK) {
-    ReplaceSocket(params_->socket_id, socket.release());
-  } else {
+void SocketSecureFunction::TlsConnectDone(
+    int result,
+    network::mojom::TLSClientSocketPtr tls_socket,
+    const net::IPEndPoint& local_addr,
+    const net::IPEndPoint& peer_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+    mojo::ScopedDataPipeProducerHandle send_pipe_handle) {
+  if (result != net::OK) {
     RemoveSocket(params_->socket_id);
     error_ = net::ErrorToString(result);
+    results_ = api::socket::Secure::Results::Create(result);
+    AsyncWorkCompleted();
+    return;
   }
 
+  auto socket =
+      std::make_unique<TLSSocket>(std::move(tls_socket), local_addr, peer_addr,
+                                  std::move(receive_pipe_handle),
+                                  std::move(send_pipe_handle), extension_id());
+  ReplaceSocket(params_->socket_id, socket.release());
   results_ = api::socket::Secure::Results::Create(result);
   AsyncWorkCompleted();
 }
diff --git a/extensions/browser/api/socket/socket_api.h b/extensions/browser/api/socket/socket_api.h
index cf52938..eae243f 100644
--- a/extensions/browser/api/socket/socket_api.h
+++ b/extensions/browser/api/socket/socket_api.h
@@ -8,6 +8,7 @@
 #include <stddef.h>
 #include <stdint.h>
 
+#include <memory>
 #include <string>
 
 #include "base/gtest_prod_util.h"
@@ -41,7 +42,6 @@
 
 namespace extensions {
 class Socket;
-class TLSSocket;
 
 // A simple interface to ApiResourceManager<Socket> or derived class. The goal
 // of this interface is to allow Socket API functions to use distinct instances
@@ -184,8 +184,10 @@
   FRIEND_TEST_ALL_PREFIXES(SocketUnitTest, Create);
   enum SocketType { kSocketTypeInvalid = -1, kSocketTypeTCP, kSocketTypeUDP };
 
+  // These two fields are only applicable if |socket_type_| is UDP.
   network::mojom::UDPSocketPtrInfo socket_;
   network::mojom::UDPSocketReceiverRequest socket_receiver_request_;
+
   std::unique_ptr<api::socket::Create::Params> params_;
   SocketType socket_type_;
 };
@@ -278,6 +280,7 @@
   void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(int result, const std::string& error_msg);
   std::unique_ptr<api::socket::Listen::Params> params_;
 };
 
@@ -295,7 +298,11 @@
   void AsyncWorkStart() override;
 
  private:
-  void OnAccept(int result_code, std::unique_ptr<net::TCPClientSocket> socket);
+  void OnAccept(int result_code,
+                network::mojom::TCPConnectedSocketPtr socket,
+                const base::Optional<net::IPEndPoint>& remote_addr,
+                mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+                mojo::ScopedDataPipeProducerHandle send_pipe_handle);
 
   std::unique_ptr<api::socket::Accept::Params> params_;
 };
@@ -400,9 +407,11 @@
 
   // AsyncApiFunction:
   bool Prepare() override;
-  void Work() override;
+  void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(bool success);
+
   std::unique_ptr<api::socket::SetKeepAlive::Params> params_;
 };
 
@@ -417,9 +426,11 @@
 
   // AsyncApiFunction:
   bool Prepare() override;
-  void Work() override;
+  void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(bool success);
+
   std::unique_ptr<api::socket::SetNoDelay::Params> params_;
 };
 
@@ -561,11 +572,14 @@
   void AsyncWorkStart() override;
 
  private:
-  // Callback from TLSSocket::UpgradeSocketToTLS().
-  void TlsConnectDone(std::unique_ptr<TLSSocket> socket, int result);
+  void TlsConnectDone(int result,
+                      network::mojom::TLSClientSocketPtr tls_socket,
+                      const net::IPEndPoint& local_addr,
+                      const net::IPEndPoint& peer_addr,
+                      mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+                      mojo::ScopedDataPipeProducerHandle send_pipe_handle);
 
   std::unique_ptr<api::socket::Secure::Params> params_;
-  scoped_refptr<net::URLRequestContextGetter> url_request_getter_;
 
   DISALLOW_COPY_AND_ASSIGN(SocketSecureFunction);
 };
diff --git a/extensions/browser/api/socket/tcp_socket.cc b/extensions/browser/api/socket/tcp_socket.cc
index a1ee755..e80bd771 100644
--- a/extensions/browser/api/socket/tcp_socket.cc
+++ b/extensions/browser/api/socket/tcp_socket.cc
@@ -4,21 +4,49 @@
 
 #include "extensions/browser/api/socket/tcp_socket.h"
 
+#include <utility>
+
 #include "base/callback_helpers.h"
 #include "base/lazy_instance.h"
 #include "base/logging.h"
 #include "base/macros.h"
 #include "base/memory/ptr_util.h"
+#include "content/public/browser/browser_context.h"
+#include "content/public/browser/browser_thread.h"
+#include "content/public/browser/storage_partition.h"
 #include "extensions/browser/api/api_resource.h"
+#include "extensions/browser/api/socket/mojo_data_pump.h"
 #include "net/base/address_list.h"
 #include "net/base/ip_endpoint.h"
 #include "net/base/net_errors.h"
-#include "net/base/rand_callback.h"
-#include "net/log/net_log_source.h"
-#include "net/socket/tcp_client_socket.h"
+#include "net/base/url_util.h"
+#include "services/network/public/mojom/ssl_config.mojom.h"
 
 namespace extensions {
 
+namespace {
+
+// Returns true if successfully parsed the SSL protocol version that is
+// represented by a string. Returns false if |version_str| is invalid.
+bool SSLProtocolVersionFromString(const std::string& version_str,
+                                  network::mojom::SSLVersion* version_out) {
+  if (version_str == "tls1") {
+    *version_out = network::mojom::SSLVersion::kTLS1;
+    return true;
+  }
+  if (version_str == "tls1.1") {
+    *version_out = network::mojom::SSLVersion::kTLS11;
+    return true;
+  }
+  if (version_str == "tls1.2") {
+    *version_out = network::mojom::SSLVersion::kTLS12;
+    return true;
+  }
+  return false;
+}
+
+}  // namespace
+
 const char kTCPSocketTypeInvalidError[] =
     "Cannot call both connect and listen on the same socket.";
 const char kSocketListenError[] = "Could not listen on the specified port.";
@@ -45,38 +73,31 @@
   return g_server_factory.Pointer();
 }
 
-TCPSocket::TCPSocket(const std::string& owner_extension_id)
-    : Socket(owner_extension_id), socket_mode_(UNKNOWN) {}
-
-TCPSocket::TCPSocket(std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-                     const std::string& owner_extension_id,
-                     bool is_connected)
-    : Socket(owner_extension_id),
-      socket_(std::move(tcp_client_socket)),
-      socket_mode_(CLIENT) {
-  this->is_connected_ = is_connected;
-}
-
-TCPSocket::TCPSocket(std::unique_ptr<net::TCPServerSocket> tcp_server_socket,
+TCPSocket::TCPSocket(content::BrowserContext* browser_context,
                      const std::string& owner_extension_id)
     : Socket(owner_extension_id),
-      server_socket_(std::move(tcp_server_socket)),
-      socket_mode_(SERVER) {}
+      browser_context_(browser_context),
+      socket_mode_(UNKNOWN),
+      mojo_data_pump_(nullptr),
+      task_runner_(base::SequencedTaskRunnerHandle::Get()),
+      weak_factory_(this) {}
 
-// static
-TCPSocket* TCPSocket::CreateSocketForTesting(
-    std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-    const std::string& owner_extension_id,
-    bool is_connected) {
-  return new TCPSocket(std::move(tcp_client_socket), owner_extension_id,
-                       is_connected);
-}
+TCPSocket::TCPSocket(network::mojom::TCPConnectedSocketPtr socket,
+                     mojo::ScopedDataPipeConsumerHandle receive_stream,
+                     mojo::ScopedDataPipeProducerHandle send_stream,
+                     const base::Optional<net::IPEndPoint>& remote_addr,
+                     const std::string& owner_extension_id)
+    : Socket(owner_extension_id),
+      browser_context_(nullptr),
+      socket_mode_(CLIENT),
+      client_socket_(std::move(socket)),
+      mojo_data_pump_(std::make_unique<MojoDataPump>(std::move(receive_stream),
+                                                     std::move(send_stream))),
+      task_runner_(base::SequencedTaskRunnerHandle::Get()),
+      peer_addr_(remote_addr),
 
-// static
-TCPSocket* TCPSocket::CreateServerSocketForTesting(
-    std::unique_ptr<net::TCPServerSocket> tcp_server_socket,
-    const std::string& owner_extension_id) {
-  return new TCPSocket(std::move(tcp_server_socket), owner_extension_id);
+      weak_factory_(this) {
+  is_connected_ = true;
 }
 
 TCPSocket::~TCPSocket() {
@@ -84,48 +105,57 @@
 }
 
 void TCPSocket::Connect(const net::AddressList& address,
-                        const CompletionCallback& callback) {
-  DCHECK(!callback.is_null());
+                        net::CompletionOnceCallback callback) {
+  DCHECK(callback);
 
-  if (socket_mode_ == SERVER || !connect_callback_.is_null()) {
-    callback.Run(net::ERR_CONNECTION_FAILED);
+  if (socket_mode_ == SERVER || connect_callback_) {
+    std::move(callback).Run(net::ERR_CONNECTION_FAILED);
     return;
   }
 
   if (is_connected_) {
-    callback.Run(net::ERR_SOCKET_IS_CONNECTED);
+    std::move(callback).Run(net::ERR_SOCKET_IS_CONNECTED);
     return;
   }
 
-  DCHECK(!server_socket_.get());
+  DCHECK(!server_socket_);
   socket_mode_ = CLIENT;
-  connect_callback_ = callback;
+  connect_callback_ = std::move(callback);
 
-  int result = net::ERR_CONNECTION_FAILED;
-  if (!is_connected_) {
-    socket_.reset(
-        new net::TCPClientSocket(address, NULL, NULL, net::NetLogSource()));
-    result = socket_->Connect(
-        base::Bind(&TCPSocket::OnConnectComplete, base::Unretained(this)));
-  }
+  // |completion_callback| is called on current thread.
+  network::mojom::NetworkContext::CreateTCPConnectedSocketCallback
+      completion_callback = base::BindOnce(&TCPSocket::OnConnectComplete,
+                                           weak_factory_.GetWeakPtr());
 
-  if (result != net::ERR_IO_PENDING)
-    OnConnectComplete(result);
+  // |completion_callback_ui| is called on the UI thread.
+  network::mojom::NetworkContext::CreateTCPConnectedSocketCallback
+      completion_callback_ui =
+          base::BindOnce(&TCPSocket::OnConnectCompleteOnUIThread, task_runner_,
+                         std::move(completion_callback));
+
+  content::BrowserThread::PostTask(
+      content::BrowserThread::UI, FROM_HERE,
+      base::BindOnce(&TCPSocket::ConnectOnUIThread, storage_partition_,
+                     browser_context_, address,
+                     mojo::MakeRequest(&client_socket_),
+                     std::move(completion_callback_ui)));
 }
 
 void TCPSocket::Disconnect(bool socket_destroying) {
   is_connected_ = false;
-  if (socket_.get())
-    socket_->Disconnect();
-  server_socket_.reset(NULL);
+  local_addr_ = base::nullopt;
+  peer_addr_ = base::nullopt;
+  mojo_data_pump_ = nullptr;
+  client_socket_.reset();
+  server_socket_.reset();
+  listen_callback_.Reset();
   connect_callback_.Reset();
+  accept_callback_.Reset();
   // TODO(devlin): Should we do this for all callbacks?
-  if (!read_callback_.is_null()) {
-    base::ResetAndReturn(&read_callback_)
+  if (read_callback_) {
+    std::move(read_callback_)
         .Run(net::ERR_CONNECTION_CLOSED, nullptr, socket_destroying);
   }
-  accept_callback_.Reset();
-  accept_socket_.reset(NULL);
 }
 
 void TCPSocket::Bind(const std::string& address,
@@ -134,42 +164,30 @@
   callback.Run(net::ERR_FAILED);
 }
 
-void TCPSocket::Read(int count, const ReadCompletionCallback& callback) {
-  DCHECK(!callback.is_null());
+void TCPSocket::Read(int count, ReadCompletionCallback callback) {
+  DCHECK(callback);
 
   const bool socket_destroying = false;
   if (socket_mode_ != CLIENT) {
-    callback.Run(net::ERR_FAILED, nullptr, socket_destroying);
+    std::move(callback).Run(net::ERR_FAILED, nullptr, socket_destroying);
     return;
   }
 
-  if (!read_callback_.is_null() || !connect_callback_.is_null()) {
+  if (!mojo_data_pump_) {
+    std::move(callback).Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr,
+                            socket_destroying);
+    return;
+  }
+  if (mojo_data_pump_->HasPendingRead() || connect_callback_) {
     // It's illegal to read a net::TCPSocket while a pending Connect or Read is
     // already in progress.
-    callback.Run(net::ERR_IO_PENDING, nullptr, socket_destroying);
+    std::move(callback).Run(net::ERR_IO_PENDING, nullptr, socket_destroying);
     return;
   }
 
-  if (count < 0) {
-    callback.Run(net::ERR_INVALID_ARGUMENT, nullptr, socket_destroying);
-    return;
-  }
-
-  if (!socket_.get() || !is_connected_) {
-    callback.Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr, socket_destroying);
-    return;
-  }
-
-  read_callback_ = callback;
-  scoped_refptr<net::IOBuffer> io_buffer = new net::IOBuffer(count);
-  int result = socket_->Read(
-      io_buffer.get(),
-      count,
-      base::Bind(
-          &TCPSocket::OnReadComplete, base::Unretained(this), io_buffer));
-
-  if (result != net::ERR_IO_PENDING)
-    OnReadComplete(io_buffer, result);
+  read_callback_ = std::move(callback);
+  mojo_data_pump_->Read(count, base::BindOnce(&TCPSocket::OnReadComplete,
+                                              base::Unretained(this)));
 }
 
 void TCPSocket::RecvFrom(int count,
@@ -185,85 +203,105 @@
   callback.Run(net::ERR_FAILED);
 }
 
-bool TCPSocket::SetKeepAlive(bool enable, int delay) {
-  if (!socket_.get())
-    return false;
-  return socket_->SetKeepAlive(enable, delay);
-}
-
-bool TCPSocket::SetNoDelay(bool no_delay) {
-  if (!socket_.get())
-    return false;
-  return socket_->SetNoDelay(no_delay);
-}
-
-int TCPSocket::Listen(const std::string& address,
-                      uint16_t port,
-                      int backlog,
-                      std::string* error_msg) {
-  if (socket_mode_ == CLIENT) {
-    *error_msg = kTCPSocketTypeInvalidError;
-    return net::ERR_NOT_IMPLEMENTED;
+void TCPSocket::SetKeepAlive(bool enable,
+                             int delay,
+                             SetKeepAliveCallback callback) {
+  if (!client_socket_) {
+    std::move(callback).Run(net::ERR_FAILED);
+    return;
   }
-  DCHECK(!socket_.get());
+  client_socket_->SetKeepAlive(enable, delay, std::move(callback));
+}
+
+void TCPSocket::SetNoDelay(bool no_delay, SetNoDelayCallback callback) {
+  if (!client_socket_) {
+    std::move(callback).Run(net::ERR_FAILED);
+    return;
+  }
+  client_socket_->SetNoDelay(no_delay, std::move(callback));
+}
+
+void TCPSocket::Listen(const std::string& address,
+                       uint16_t port,
+                       int backlog,
+                       ListenCallback callback) {
+  DCHECK(!server_socket_);
+  DCHECK(!client_socket_);
+  DCHECK(!listen_callback_);
+
+  if (socket_mode_ == CLIENT) {
+    std::move(callback).Run(net::ERR_NOT_IMPLEMENTED,
+                            kTCPSocketTypeInvalidError);
+    return;
+  }
+
+  net::IPEndPoint ip_end_point;
+  if (!StringAndPortToIPEndPoint(address, port, &ip_end_point)) {
+    std::move(callback).Run(net::ERR_INVALID_ARGUMENT, "");
+    return;
+  }
+
   socket_mode_ = SERVER;
 
-  if (!server_socket_.get()) {
-    server_socket_.reset(new net::TCPServerSocket(NULL, net::NetLogSource()));
-  }
+  listen_callback_ = std::move(callback);
 
-  int result = server_socket_->ListenWithAddressAndPort(address, port, backlog);
-  if (result) {
-    server_socket_.reset();
-    *error_msg = kSocketListenError;
-  }
-  return result;
+  // |completion_callback| is called on current thread.
+  network::mojom::NetworkContext::CreateTCPServerSocketCallback
+      completion_callback = base::BindOnce(&TCPSocket::OnListenComplete,
+                                           weak_factory_.GetWeakPtr());
+
+  // |completion_callback_ui| is called on the UI thread.
+  network::mojom::NetworkContext::CreateTCPServerSocketCallback
+      completion_callback_ui =
+          base::BindOnce(&TCPSocket::OnListenCompleteOnUIThread, task_runner_,
+                         std::move(completion_callback));
+
+  content::BrowserThread::PostTask(
+      content::BrowserThread::UI, FROM_HERE,
+      base::BindOnce(&TCPSocket::ListenOnUIThread, storage_partition_,
+                     browser_context_, ip_end_point, backlog,
+                     mojo::MakeRequest(&server_socket_),
+                     std::move(completion_callback_ui)));
 }
 
-void TCPSocket::Accept(const AcceptCompletionCallback& callback) {
+void TCPSocket::Accept(AcceptCompletionCallback callback) {
   if (socket_mode_ != SERVER || !server_socket_.get()) {
-    callback.Run(net::ERR_FAILED, NULL);
+    std::move(callback).Run(net::ERR_FAILED, nullptr, base::nullopt,
+                            mojo::ScopedDataPipeConsumerHandle(),
+                            mojo::ScopedDataPipeProducerHandle());
     return;
   }
 
   // Limits to only 1 blocked accept call.
-  if (!accept_callback_.is_null()) {
-    callback.Run(net::ERR_FAILED, NULL);
+  if (accept_callback_) {
+    std::move(callback).Run(net::ERR_FAILED, nullptr, base::nullopt,
+                            mojo::ScopedDataPipeConsumerHandle(),
+                            mojo::ScopedDataPipeProducerHandle());
     return;
   }
 
-  int result = server_socket_->Accept(
-      &accept_socket_,
-      base::Bind(&TCPSocket::OnAccept, base::Unretained(this)));
-  if (result == net::ERR_IO_PENDING) {
-    accept_callback_ = callback;
-  } else if (result == net::OK) {
-    accept_callback_ = callback;
-    this->OnAccept(result);
-  } else {
-    callback.Run(result, NULL);
-  }
+  accept_callback_ = std::move(callback);
+  server_socket_->Accept(
+      nullptr /* observer */,
+      base::BindOnce(&TCPSocket::OnAccept, base::Unretained(this)));
 }
 
 bool TCPSocket::IsConnected() {
-  RefreshConnectionStatus();
   return is_connected_;
 }
 
 bool TCPSocket::GetPeerAddress(net::IPEndPoint* address) {
-  if (!socket_.get())
+  if (!peer_addr_)
     return false;
-  return !socket_->GetPeerAddress(address);
+  *address = peer_addr_.value();
+  return true;
 }
 
 bool TCPSocket::GetLocalAddress(net::IPEndPoint* address) {
-  if (socket_.get()) {
-    return !socket_->GetLocalAddress(address);
-  } else if (server_socket_.get()) {
-    return !server_socket_->GetLocalAddress(address);
-  } else {
+  if (!local_addr_)
     return false;
-  }
+  *address = local_addr_.value();
+  return true;
 }
 
 Socket::SocketType TCPSocket::GetSocketType() const { return Socket::TYPE_TCP; }
@@ -271,97 +309,265 @@
 int TCPSocket::WriteImpl(net::IOBuffer* io_buffer,
                          int io_buffer_size,
                          const net::CompletionCallback& callback) {
-  if (socket_mode_ != CLIENT)
-    return net::ERR_FAILED;
-  else if (!socket_.get() || !IsConnected())
+  if (!mojo_data_pump_)
     return net::ERR_SOCKET_NOT_CONNECTED;
-  else
-    return socket_->Write(io_buffer, io_buffer_size, callback,
-                          Socket::GetNetworkTrafficAnnotationTag());
+
+  mojo_data_pump_->Write(io_buffer, io_buffer_size,
+                         base::BindOnce(&TCPSocket::OnWriteComplete,
+                                        base::Unretained(this), callback));
+  return net::ERR_IO_PENDING;
 }
 
-void TCPSocket::RefreshConnectionStatus() {
-  if (!is_connected_)
+// static
+void TCPSocket::ConnectOnUIThread(
+    content::StoragePartition* storage_partition,
+    content::BrowserContext* browser_context,
+    const net::AddressList& remote_addr_list,
+    network::mojom::TCPConnectedSocketRequest request,
+    network::mojom::NetworkContext::CreateTCPConnectedSocketCallback
+        completion_callback) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+  if (!storage_partition) {
+    storage_partition =
+        content::BrowserContext::GetDefaultStoragePartition(browser_context);
+  }
+  storage_partition->GetNetworkContext()->CreateTCPConnectedSocket(
+      base::nullopt, remote_addr_list,
+      net::MutableNetworkTrafficAnnotationTag(
+          Socket::GetNetworkTrafficAnnotationTag()),
+      std::move(request), nullptr /* observer */,
+      std::move(completion_callback));
+}
+
+// static
+void TCPSocket::OnConnectCompleteOnUIThread(
+    scoped_refptr<base::SequencedTaskRunner> original_task_runner,
+    network::mojom::NetworkContext::CreateTCPConnectedSocketCallback callback,
+    int result,
+    const base::Optional<net::IPEndPoint>& local_addr,
+    const base::Optional<net::IPEndPoint>& peer_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_stream,
+    mojo::ScopedDataPipeProducerHandle send_stream) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+  original_task_runner->PostTask(
+      FROM_HERE,
+      base::BindOnce(std::move(callback), result, local_addr, peer_addr,
+                     std::move(receive_stream), std::move(send_stream)));
+}
+
+void TCPSocket::OnConnectComplete(
+    int result,
+    const base::Optional<net::IPEndPoint>& local_addr,
+    const base::Optional<net::IPEndPoint>& peer_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_stream,
+    mojo::ScopedDataPipeProducerHandle send_stream) {
+  DCHECK(!is_connected_);
+  DCHECK(connect_callback_);
+  DCHECK(task_runner_->RunsTasksInCurrentSequence());
+
+  if (result == net::OK) {
+    is_connected_ = true;
+    local_addr_ = local_addr;
+    peer_addr_ = peer_addr;
+    mojo_data_pump_ = std::make_unique<MojoDataPump>(std::move(receive_stream),
+                                                     std::move(send_stream));
+  }
+  std::move(connect_callback_).Run(result);
+}
+
+// static
+void TCPSocket::ListenOnUIThread(
+    content::StoragePartition* storage_partition,
+    content::BrowserContext* browser_context,
+    const net::IPEndPoint& local_addr,
+    int backlog,
+    network::mojom::TCPServerSocketRequest request,
+    network::mojom::NetworkContext::CreateTCPServerSocketCallback callback) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+  if (!storage_partition) {
+    storage_partition =
+        content::BrowserContext::GetDefaultStoragePartition(browser_context);
+  }
+  storage_partition->GetNetworkContext()->CreateTCPServerSocket(
+      local_addr, backlog,
+      net::MutableNetworkTrafficAnnotationTag(
+          Socket::GetNetworkTrafficAnnotationTag()),
+      std::move(request), std::move(callback));
+}
+
+// static
+void TCPSocket::OnListenCompleteOnUIThread(
+    const scoped_refptr<base::SequencedTaskRunner>& original_task_runner,
+    network::mojom::NetworkContext::CreateTCPServerSocketCallback callback,
+    int result,
+    const base::Optional<net::IPEndPoint>& local_addr) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+  original_task_runner->PostTask(
+      FROM_HERE, base::BindOnce(std::move(callback), result, local_addr));
+}
+
+void TCPSocket::OnListenComplete(
+    int result,
+    const base::Optional<net::IPEndPoint>& local_addr) {
+  DCHECK(task_runner_->RunsTasksInCurrentSequence());
+  DCHECK(listen_callback_);
+
+  if (result != net::OK) {
+    server_socket_.reset();
+    std::move(listen_callback_).Run(result, kSocketListenError);
     return;
-  if (server_socket_)
-    return;
-  if (!socket_->IsConnected()) {
+  }
+  local_addr_ = local_addr;
+  std::move(listen_callback_).Run(result, "");
+}
+
+content::StoragePartition* TCPSocket::GetStoragePartitionHelper() {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+  return storage_partition_
+             ? storage_partition_
+             : content::BrowserContext::GetDefaultStoragePartition(
+                   browser_context_);
+}
+
+void TCPSocket::OnAccept(int result,
+                         const base::Optional<net::IPEndPoint>& remote_addr,
+                         network::mojom::TCPConnectedSocketPtr connected_socket,
+                         mojo::ScopedDataPipeConsumerHandle receive_stream,
+                         mojo::ScopedDataPipeProducerHandle send_stream) {
+  DCHECK(accept_callback_);
+  std::move(accept_callback_)
+      .Run(result, std::move(connected_socket), remote_addr,
+           std::move(receive_stream), std::move(send_stream));
+}
+
+void TCPSocket::OnWriteComplete(const net::CompletionCallback& callback,
+                                int result) {
+  if (result < 0) {
+    // Write side has terminated. This can be an error or a graceful close.
+    // TCPSocketEventDispatcher doesn't distinguish between the two.
     Disconnect(false /* socket_destroying */);
   }
+  callback.Run(result);
 }
 
-void TCPSocket::OnConnectComplete(int result) {
-  DCHECK(!connect_callback_.is_null());
-  DCHECK(!is_connected_);
-  is_connected_ = result == net::OK;
+void TCPSocket::OnReadComplete(int result,
+                               scoped_refptr<net::IOBuffer> io_buffer) {
+  DCHECK(read_callback_);
 
-  // The completion callback may re-enter TCPSocket, e.g. to Read(); therefore
-  // we reset |connect_callback_| before calling it.
-  CompletionCallback connect_callback = connect_callback_;
-  connect_callback_.Reset();
-  connect_callback.Run(result);
-}
-
-void TCPSocket::OnReadComplete(scoped_refptr<net::IOBuffer> io_buffer,
-                               int result) {
-  DCHECK(!read_callback_.is_null());
-  read_callback_.Run(result, io_buffer, false /* socket_destroying */);
-  read_callback_.Reset();
-}
-
-void TCPSocket::OnAccept(int result) {
-  DCHECK(!accept_callback_.is_null());
-  if (result == net::OK && accept_socket_.get()) {
-    accept_callback_.Run(result,
-                         base::WrapUnique(static_cast<net::TCPClientSocket*>(
-                             accept_socket_.release())));
-  } else {
-    accept_callback_.Run(result, NULL);
+  // Use a local variable for |read_callback_|, because otherwise Disconnect()
+  // will try to invoke |read_callback_| with a hardcoded result code.
+  ReadCompletionCallback callback = std::move(read_callback_);
+  if (result < 0) {
+    // Read side has terminated. This can be an error or a graceful close.
+    // TCPSocketEventDispatcher doesn't distinguish between the two.
+    Disconnect(false /* socket_destroying */);
   }
-  accept_callback_.Reset();
+  std::move(callback).Run(result, io_buffer, false /* socket_destroying */);
 }
 
-void TCPSocket::Release() {
-  // Release() is only invoked when the underlying sockets are taken (via
-  // ClientStream()) by TLSSocket. TLSSocket only supports CLIENT-mode
-  // sockets.
-  DCHECK(!server_socket_.release() && !accept_socket_.release() &&
-         socket_mode_ == CLIENT)
-      << "Called in server mode.";
-
-  // Release() doesn't disconnect the underlying sockets, but it does
-  // disconnect them from this TCPSocket.
-  is_connected_ = false;
-
-  connect_callback_.Reset();
-  read_callback_.Reset();
-  accept_callback_.Reset();
-
-  DCHECK(socket_.get()) << "Called on null client socket.";
-  ignore_result(socket_.release());
+void TCPSocket::OnUpgradeToTLSComplete(
+    UpgradeToTLSCallback callback,
+    network::mojom::TLSClientSocketPtr tls_socket,
+    const net::IPEndPoint& local_addr,
+    const net::IPEndPoint& peer_addr,
+    int result,
+    mojo::ScopedDataPipeConsumerHandle receive_stream,
+    mojo::ScopedDataPipeProducerHandle send_stream) {
+  std::move(callback).Run(result, std::move(tls_socket), local_addr, peer_addr,
+                          std::move(receive_stream), std::move(send_stream));
 }
 
-net::TCPClientSocket* TCPSocket::ClientStream() {
-  if (socket_mode_ != CLIENT || GetSocketType() != TYPE_TCP)
-    return NULL;
-  return socket_.get();
+void TCPSocket::UpgradeToTLS(api::socket::SecureOptions* options,
+                             UpgradeToTLSCallback callback) {
+  if (!client_socket_ || !mojo_data_pump_ ||
+      mojo_data_pump_->HasPendingRead() || mojo_data_pump_->HasPendingWrite()) {
+    std::move(callback).Run(net::ERR_FAILED, nullptr, net::IPEndPoint(),
+                            net::IPEndPoint(),
+                            mojo::ScopedDataPipeConsumerHandle(),
+                            mojo::ScopedDataPipeProducerHandle());
+    return;
+  }
+  if (!local_addr_ || !peer_addr_) {
+    DVLOG(1) << "Could not get local address or peer address.";
+    std::move(callback).Run(net::ERR_FAILED, nullptr, net::IPEndPoint(),
+                            net::IPEndPoint(),
+                            mojo::ScopedDataPipeConsumerHandle(),
+                            mojo::ScopedDataPipeProducerHandle());
+    return;
+  }
+
+  // Convert any U-LABELs to A-LABELs.
+  url::CanonHostInfo host_info;
+  std::string canon_host = net::CanonicalizeHost(hostname(), &host_info);
+
+  // Canonicalization shouldn't fail: the socket is already connected with a
+  // host, using this hostname.
+  if (host_info.family == url::CanonHostInfo::BROKEN) {
+    DVLOG(1) << "Could not canonicalize hostname";
+    std::move(callback).Run(net::ERR_FAILED, nullptr, net::IPEndPoint(),
+                            net::IPEndPoint(),
+                            mojo::ScopedDataPipeConsumerHandle(),
+                            mojo::ScopedDataPipeProducerHandle());
+    return;
+  }
+
+  mojo_data_pump_ = nullptr;
+  network::mojom::TLSClientSocketOptionsPtr mojo_socket_options =
+      network::mojom::TLSClientSocketOptions::New();
+  if (options && options->tls_version.get()) {
+    network::mojom::SSLVersion version_min, version_max;
+    bool has_version_min = false;
+    bool has_version_max = false;
+    api::socket::TLSVersionConstraints* versions = options->tls_version.get();
+    if (versions->min.get()) {
+      has_version_min =
+          SSLProtocolVersionFromString(*versions->min, &version_min);
+    }
+    if (versions->max.get()) {
+      has_version_max =
+          SSLProtocolVersionFromString(*versions->max, &version_max);
+    }
+    if (has_version_min)
+      mojo_socket_options->version_min = version_min;
+    if (has_version_max)
+      mojo_socket_options->version_max = version_max;
+  }
+  network::mojom::TLSClientSocketPtr tls_socket;
+  network::mojom::TLSClientSocketRequest tls_socket_request =
+      mojo::MakeRequest(&tls_socket);
+  net::HostPortPair host_port_pair(canon_host, peer_addr_.value().port());
+  client_socket_->UpgradeToTLS(
+      host_port_pair, std::move(mojo_socket_options),
+      net::MutableNetworkTrafficAnnotationTag(
+          Socket::GetNetworkTrafficAnnotationTag()),
+      std::move(tls_socket_request), nullptr /* observer */,
+      base::BindOnce(&TCPSocket::OnUpgradeToTLSComplete, base::Unretained(this),
+                     std::move(callback), std::move(tls_socket),
+                     local_addr_.value(), peer_addr_.value()));
 }
 
-bool TCPSocket::HasPendingRead() const {
-  return !read_callback_.is_null();
-}
-
-ResumableTCPSocket::ResumableTCPSocket(const std::string& owner_extension_id)
-    : TCPSocket(owner_extension_id),
+ResumableTCPSocket::ResumableTCPSocket(content::BrowserContext* browser_context,
+                                       const std::string& owner_extension_id)
+    : TCPSocket(browser_context, owner_extension_id),
       persistent_(false),
       buffer_size_(0),
       paused_(false) {}
 
 ResumableTCPSocket::ResumableTCPSocket(
-    std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-    const std::string& owner_extension_id,
-    bool is_connected)
-    : TCPSocket(std::move(tcp_client_socket), owner_extension_id, is_connected),
+    network::mojom::TCPConnectedSocketPtr socket,
+    mojo::ScopedDataPipeConsumerHandle receive_stream,
+    mojo::ScopedDataPipeProducerHandle send_stream,
+    const base::Optional<net::IPEndPoint>& remote_addr,
+    const std::string& owner_extension_id)
+    : TCPSocket(std::move(socket),
+                std::move(receive_stream),
+                std::move(send_stream),
+                remote_addr,
+                owner_extension_id),
       persistent_(false),
       buffer_size_(0),
       paused_(false) {}
@@ -377,8 +583,11 @@
 bool ResumableTCPSocket::IsPersistent() const { return persistent(); }
 
 ResumableTCPServerSocket::ResumableTCPServerSocket(
+    content::BrowserContext* browser_context,
     const std::string& owner_extension_id)
-    : TCPSocket(owner_extension_id), persistent_(false), paused_(false) {}
+    : TCPSocket(browser_context, owner_extension_id),
+      persistent_(false),
+      paused_(false) {}
 
 bool ResumableTCPServerSocket::IsPersistent() const { return persistent(); }
 
diff --git a/extensions/browser/api/socket/tcp_socket.h b/extensions/browser/api/socket/tcp_socket.h
index bdff76ed..f834301 100644
--- a/extensions/browser/api/socket/tcp_socket.h
+++ b/extensions/browser/api/socket/tcp_socket.h
@@ -7,76 +7,88 @@
 
 #include <stdint.h>
 
+#include <memory>
 #include <string>
 
+#include "base/memory/weak_ptr.h"
+#include "base/sequenced_task_runner.h"
 #include "extensions/browser/api/socket/socket.h"
+#include "extensions/browser/api/socket/tcp_socket.h"
+#include "extensions/common/api/socket.h"
+#include "net/base/completion_once_callback.h"
+#include "services/network/public/mojom/network_service.mojom.h"
+#include "services/network/public/mojom/tcp_socket.mojom.h"
+#include "services/network/public/mojom/tls_socket.mojom.h"
 
-// This looks like it should be forward-declarable, but it does some tricky
-// moves that make it easier to just include it.
-#include "net/socket/tcp_client_socket.h"
-#include "net/socket/tcp_server_socket.h"
-
-namespace net {
-class Socket;
+namespace content {
+class BrowserContext;
+class StoragePartition;
 }
 
 namespace extensions {
 
+class MojoDataPump;
+
 class TCPSocket : public Socket {
  public:
-  explicit TCPSocket(const std::string& owner_extension_id);
-  TCPSocket(std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-            const std::string& owner_extension_id,
-            bool is_connected = false);
+  using UpgradeToTLSCallback =
+      base::OnceCallback<void(int,
+                              network::mojom::TLSClientSocketPtr,
+                              const net::IPEndPoint&,
+                              const net::IPEndPoint&,
+                              mojo::ScopedDataPipeConsumerHandle,
+                              mojo::ScopedDataPipeProducerHandle)>;
+
+  // Constuctor for when |socket_mode_| is unknown. The |socket_mode_| will be
+  // filled in when the consumer calls Listen/Connect.
+  TCPSocket(content::BrowserContext* browser_context,
+            const std::string& owner_extension_id);
+
+  // Created using TCPServerSocket::Accept().
+  TCPSocket(network::mojom::TCPConnectedSocketPtr socket,
+            mojo::ScopedDataPipeConsumerHandle receive_stream,
+            mojo::ScopedDataPipeProducerHandle send_stream,
+            const base::Optional<net::IPEndPoint>& remote_addr,
+            const std::string& owner_extension_id);
 
   ~TCPSocket() override;
 
   void Connect(const net::AddressList& address,
-               const CompletionCallback& callback) override;
+               net::CompletionOnceCallback callback) override;
   void Disconnect(bool socket_destroying) override;
   void Bind(const std::string& address,
             uint16_t port,
             const CompletionCallback& callback) override;
-  void Read(int count, const ReadCompletionCallback& callback) override;
+  void Read(int count, ReadCompletionCallback callback) override;
   void RecvFrom(int count, const RecvFromCompletionCallback& callback) override;
   void SendTo(scoped_refptr<net::IOBuffer> io_buffer,
               int byte_count,
               const net::IPEndPoint& address,
               const CompletionCallback& callback) override;
-  bool SetKeepAlive(bool enable, int delay) override;
-  bool SetNoDelay(bool no_delay) override;
-  int Listen(const std::string& address,
-             uint16_t port,
-             int backlog,
-             std::string* error_msg) override;
-  void Accept(const AcceptCompletionCallback& callback) override;
+  void SetKeepAlive(bool enable,
+                    int delay,
+                    SetKeepAliveCallback callback) override;
+  void SetNoDelay(bool no_delay, SetNoDelayCallback callback) override;
+  void Listen(const std::string& address,
+              uint16_t port,
+              int backlog,
+              ListenCallback callback) override;
+  void Accept(AcceptCompletionCallback callback) override;
 
   bool IsConnected() override;
 
   bool GetPeerAddress(net::IPEndPoint* address) override;
   bool GetLocalAddress(net::IPEndPoint* address) override;
 
-  // Like Disconnect(), only Release() doesn't delete the underlying stream
-  // or attempt to close it. Useful when giving away ownership with
-  // ClientStream().
-  virtual void Release();
-
   Socket::SocketType GetSocketType() const override;
 
-  static TCPSocket* CreateSocketForTesting(
-      std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-      const std::string& owner_extension_id,
-      bool is_connected = false);
-  static TCPSocket* CreateServerSocketForTesting(
-      std::unique_ptr<net::TCPServerSocket> tcp_server_socket,
-      const std::string& owner_extension_id);
+  void UpgradeToTLS(api::socket::SecureOptions* options,
+                    UpgradeToTLSCallback callback);
 
-  // Returns NULL if GetSocketType() isn't TYPE_TCP or if the connection
-  // wasn't set up via Connect() (vs Listen()/Accept()).
-  net::TCPClientSocket* ClientStream();
-
-  // Whether a Read() has been issued, that hasn't come back yet.
-  bool HasPendingRead() const;
+  void SetStoragePartitionForTest(
+      content::StoragePartition* storage_partition) {
+    storage_partition_ = storage_partition;
+  }
 
  protected:
   int WriteImpl(net::IOBuffer* io_buffer,
@@ -84,26 +96,101 @@
                 const net::CompletionCallback& callback) override;
 
  private:
-  void RefreshConnectionStatus();
-  void OnConnectComplete(int result);
-  void OnReadComplete(scoped_refptr<net::IOBuffer> io_buffer, int result);
-  void OnAccept(int result);
+  // Connects a client TCP socket. This is done on the UI thread because
+  // StoragePartition::GetNetworkContext() needs to happen on the UI thread.
+  // The completion callback is posted back to the thread on which |this| lives.
+  static void ConnectOnUIThread(
+      content::StoragePartition* storage_partition,
+      content::BrowserContext* browser_context,
+      const net::AddressList& remote_address_list,
+      network::mojom::TCPConnectedSocketRequest request,
+      network::mojom::NetworkContext::CreateTCPConnectedSocketCallback
+          callback);
+  static void OnConnectCompleteOnUIThread(
+      scoped_refptr<base::SequencedTaskRunner> original_task_runner,
+      network::mojom::NetworkContext::CreateTCPConnectedSocketCallback callback,
+      int result,
+      const base::Optional<net::IPEndPoint>& local_addr,
+      const base::Optional<net::IPEndPoint>& peer_addr,
+      mojo::ScopedDataPipeConsumerHandle receive_stream,
+      mojo::ScopedDataPipeProducerHandle send_stream);
+  void OnConnectComplete(int result,
+                         const base::Optional<net::IPEndPoint>& local_addr,
+                         const base::Optional<net::IPEndPoint>& peer_addr,
+                         mojo::ScopedDataPipeConsumerHandle receive_stream,
+                         mojo::ScopedDataPipeProducerHandle send_stream);
 
-  TCPSocket(std::unique_ptr<net::TCPServerSocket> tcp_server_socket,
-            const std::string& owner_extension_id);
+  // Connects a server TCP socket. This is done on the UI thread because
+  // StoragePartition::GetNetworkContext() needs to happen on the UI thread.
+  // The completion callback is posted back to the thread on which |this| lives.
+  static void ListenOnUIThread(
+      content::StoragePartition* storage_partition,
+      content::BrowserContext* browser_context,
+      const net::IPEndPoint& local_addr,
+      int backlog,
+      network::mojom::TCPServerSocketRequest request,
+      network::mojom::NetworkContext::CreateTCPServerSocketCallback callback);
+  static void OnListenCompleteOnUIThread(
+      const scoped_refptr<base::SequencedTaskRunner>& original_task_runner,
+      network::mojom::NetworkContext::CreateTCPServerSocketCallback callback,
+      int result,
+      const base::Optional<net::IPEndPoint>& local_addr);
+  void OnListenComplete(int result,
+                        const base::Optional<net::IPEndPoint>& local_addr);
+  void OnAccept(int result,
+                const base::Optional<net::IPEndPoint>& remote_addr,
+                network::mojom::TCPConnectedSocketPtr connected_socket,
+                mojo::ScopedDataPipeConsumerHandle receive_stream,
+                mojo::ScopedDataPipeProducerHandle send_stream);
+  void OnWriteComplete(const net::CompletionCallback& callback, int result);
+  void OnReadComplete(int result, scoped_refptr<net::IOBuffer> io_buffer);
+  void OnUpgradeToTLSComplete(UpgradeToTLSCallback callback,
+                              network::mojom::TLSClientSocketPtr tls_socket,
+                              const net::IPEndPoint& local_addr,
+                              const net::IPEndPoint& peer_addr,
+                              int result,
+                              mojo::ScopedDataPipeConsumerHandle receive_stream,
+                              mojo::ScopedDataPipeProducerHandle send_stream);
 
-  std::unique_ptr<net::TCPClientSocket> socket_;
-  std::unique_ptr<net::TCPServerSocket> server_socket_;
+  content::StoragePartition* GetStoragePartitionHelper();
 
-  enum SocketMode { UNKNOWN = 0, CLIENT, SERVER, };
+  enum SocketMode {
+    UNKNOWN = 0,
+    CLIENT,
+    SERVER,
+  };
+
+  // |this| doesn't outlive |browser_context_| because |this| is owned by
+  // ApiResourceManager which is a BrowserContextKeyedAPI.
+  content::BrowserContext* browser_context_;
+
   SocketMode socket_mode_;
 
-  CompletionCallback connect_callback_;
+  // CLIENT mode.
+  network::mojom::TCPConnectedSocketPtr client_socket_;
+  // SERVER mode.
+  network::mojom::TCPServerSocketPtr server_socket_;
 
+  net::CompletionOnceCallback connect_callback_;
+  ListenCallback listen_callback_;
+  AcceptCompletionCallback accept_callback_;
   ReadCompletionCallback read_callback_;
 
-  std::unique_ptr<net::StreamSocket> accept_socket_;
-  AcceptCompletionCallback accept_callback_;
+  std::unique_ptr<MojoDataPump> mojo_data_pump_;
+
+  scoped_refptr<base::SequencedTaskRunner> task_runner_;
+
+  base::Optional<net::IPEndPoint> local_addr_;
+  base::Optional<net::IPEndPoint> peer_addr_;
+
+  // Only used in tests.
+  content::StoragePartition* storage_partition_ = nullptr;
+
+  // WeakPtr is used when posting tasks to |task_runner_| which might outlive
+  // |this|.
+  base::WeakPtrFactory<TCPSocket> weak_factory_;
+
+  DISALLOW_COPY_AND_ASSIGN(TCPSocket);
 };
 
 // TCP Socket instances from the "sockets.tcp" namespace. These are regular
@@ -111,11 +198,14 @@
 // the "sockets.tcp" namespace.
 class ResumableTCPSocket : public TCPSocket {
  public:
-  explicit ResumableTCPSocket(const std::string& owner_extension_id);
-  explicit ResumableTCPSocket(
-      std::unique_ptr<net::TCPClientSocket> tcp_client_socket,
-      const std::string& owner_extension_id,
-      bool is_connected);
+  ResumableTCPSocket(content::BrowserContext* browser_context,
+                     const std::string& owner_extension_id);
+  // Created using TCPServerSocket::Accept().
+  ResumableTCPSocket(network::mojom::TCPConnectedSocketPtr socket,
+                     mojo::ScopedDataPipeConsumerHandle receive_stream,
+                     mojo::ScopedDataPipeProducerHandle send_stream,
+                     const base::Optional<net::IPEndPoint>& remote_addr,
+                     const std::string& owner_extension_id);
 
   ~ResumableTCPSocket() override;
 
@@ -155,7 +245,8 @@
 // defined in the "sockets.tcpServer" namespace.
 class ResumableTCPServerSocket : public TCPSocket {
  public:
-  explicit ResumableTCPServerSocket(const std::string& owner_extension_id);
+  ResumableTCPServerSocket(content::BrowserContext* browser_context,
+                           const std::string& owner_extension_id);
 
   // Overriden from ApiResource
   bool IsPersistent() const override;
diff --git a/extensions/browser/api/socket/tls_socket.cc b/extensions/browser/api/socket/tls_socket.cc
index aa2a4a0..0c39bf0c 100644
--- a/extensions/browser/api/socket/tls_socket.cc
+++ b/extensions/browser/api/socket/tls_socket.cc
@@ -10,307 +10,142 @@
 #include "base/logging.h"
 #include "content/public/browser/browser_thread.h"
 #include "extensions/browser/api/api_resource.h"
+#include "extensions/browser/api/socket/mojo_data_pump.h"
 #include "net/base/address_list.h"
 #include "net/base/ip_endpoint.h"
 #include "net/base/net_errors.h"
-#include "net/base/rand_callback.h"
 #include "net/base/url_util.h"
-#include "net/socket/client_socket_factory.h"
-#include "net/socket/client_socket_handle.h"
-#include "net/socket/ssl_client_socket.h"
-#include "net/socket/tcp_client_socket.h"
 #include "url/url_canon.h"
 
-namespace {
-
-// Returns the SSL protocol version (as a uint16_t) represented by a string.
-// Returns 0 if the string is invalid.
-uint16_t SSLProtocolVersionFromString(const std::string& version_str) {
-  uint16_t version = 0;  // Invalid.
-  if (version_str == "tls1") {
-    version = net::SSL_PROTOCOL_VERSION_TLS1;
-  } else if (version_str == "tls1.1") {
-    version = net::SSL_PROTOCOL_VERSION_TLS1_1;
-  } else if (version_str == "tls1.2") {
-    version = net::SSL_PROTOCOL_VERSION_TLS1_2;
-  }
-  return version;
-}
-
-void TlsConnectDone(std::unique_ptr<net::SSLClientSocket> ssl_socket,
-                    const std::string& extension_id,
-                    const extensions::TLSSocket::SecureCallback& callback,
-                    int result) {
-  DVLOG(1) << "Got back result " << result << " " << net::ErrorToString(result);
-
-  // No matter how the TLS connection attempt went, the underlying socket's
-  // no longer bound to the original TCPSocket. It belongs to |ssl_socket|,
-  // which is promoted here to a new API-accessible socket (via a TLSSocket
-  // wrapper), or deleted.
-  if (result != net::OK) {
-    callback.Run(nullptr, result);
-    return;
-  };
-
-  // Wrap the StreamSocket in a TLSSocket, which matches the extension socket
-  // API. Set the handle of the socket to the new value, so that it can be
-  // used for read/write/close/etc.
-  std::unique_ptr<extensions::TLSSocket> wrapper(
-      new extensions::TLSSocket(std::move(ssl_socket), extension_id));
-
-  // Caller will end up deleting the prior TCPSocket, once it calls
-  // SetSocket(..,wrapper).
-  callback.Run(std::move(wrapper), result);
-}
-
-}  // namespace
-
 namespace extensions {
 
 const char kTLSSocketTypeInvalidError[] =
     "Cannot listen on a socket that is already connected.";
 
-TLSSocket::TLSSocket(std::unique_ptr<net::StreamSocket> tls_socket,
+TLSSocket::TLSSocket(network::mojom::TLSClientSocketPtr tls_socket,
+                     const net::IPEndPoint& local_addr,
+                     const net::IPEndPoint& peer_addr,
+                     mojo::ScopedDataPipeConsumerHandle receive_stream,
+                     mojo::ScopedDataPipeProducerHandle send_stream,
                      const std::string& owner_extension_id)
-    : ResumableTCPSocket(owner_extension_id),
-      tls_socket_(std::move(tls_socket)) {}
+    : ResumableTCPSocket(nullptr, owner_extension_id),
+      tls_socket_(std::move(tls_socket)),
+      local_addr_(local_addr),
+      peer_addr_(peer_addr),
+      mojo_data_pump_(std::make_unique<MojoDataPump>(std::move(receive_stream),
+                                                     std::move(send_stream))) {
+  DCHECK(tls_socket_);
+  is_connected_ = true;
+}
 
 TLSSocket::~TLSSocket() {
   Disconnect(true /* socket_destroying */);
 }
 
 void TLSSocket::Connect(const net::AddressList& address,
-                        const CompletionCallback& callback) {
-  callback.Run(net::ERR_CONNECTION_FAILED);
+                        net::CompletionOnceCallback callback) {
+  std::move(callback).Run(net::ERR_CONNECTION_FAILED);
 }
 
 void TLSSocket::Disconnect(bool socket_destroying) {
-  if (tls_socket_) {
-    tls_socket_->Disconnect();
-    tls_socket_.reset();
+  is_connected_ = false;
+  tls_socket_.reset();
+  local_addr_ = base::nullopt;
+  peer_addr_ = base::nullopt;
+  mojo_data_pump_ = nullptr;
+  // TODO(devlin): Should we do this for all callbacks?
+  if (read_callback_) {
+    std::move(read_callback_)
+        .Run(net::ERR_CONNECTION_CLOSED, nullptr, socket_destroying);
   }
 }
 
-void TLSSocket::Read(int count, const ReadCompletionCallback& callback) {
-  DCHECK(!callback.is_null());
+void TLSSocket::Read(int count, ReadCompletionCallback callback) {
+  DCHECK(callback);
+  DCHECK(!read_callback_);
 
   const bool socket_destroying = false;
-  if (!read_callback_.is_null()) {
-    callback.Run(net::ERR_IO_PENDING, nullptr, socket_destroying);
+  if (!mojo_data_pump_) {
+    std::move(callback).Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr,
+                            socket_destroying);
     return;
   }
-
-  if (count <= 0) {
-    callback.Run(net::ERR_INVALID_ARGUMENT, nullptr, socket_destroying);
+  if (mojo_data_pump_->HasPendingRead()) {
+    std::move(callback).Run(net::ERR_IO_PENDING, nullptr, socket_destroying);
     return;
   }
-
-  if (!tls_socket_.get()) {
-    callback.Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr, socket_destroying);
-    return;
-  }
-
-  read_callback_ = callback;
-  scoped_refptr<net::IOBuffer> io_buffer(new net::IOBuffer(count));
-  // |tls_socket_| is owned by this class and the callback won't be run once
-  // |tls_socket_| is gone (as in an a call to Disconnect()). Therefore, it is
-  // safe to use base::Unretained() here.
-  int result = tls_socket_->Read(
-      io_buffer.get(),
-      count,
-      base::Bind(
-          &TLSSocket::OnReadComplete, base::Unretained(this), io_buffer));
-
-  if (result != net::ERR_IO_PENDING) {
-    OnReadComplete(io_buffer, result);
-  }
+  read_callback_ = std::move(callback);
+  mojo_data_pump_->Read(count, base::BindOnce(&TLSSocket::OnReadComplete,
+                                              base::Unretained(this)));
 }
 
-void TLSSocket::OnReadComplete(const scoped_refptr<net::IOBuffer>& io_buffer,
-                               int result) {
-  DCHECK(!read_callback_.is_null());
-  base::ResetAndReturn(&read_callback_)
-      .Run(result, io_buffer, false /* socket_destroying */);
-}
-
-int TLSSocket::WriteImpl(net::IOBuffer* io_buffer,
-                         int io_buffer_size,
-                         const net::CompletionCallback& callback) {
-  if (!IsConnected()) {
-    return net::ERR_SOCKET_NOT_CONNECTED;
-  }
-  return tls_socket_->Write(io_buffer, io_buffer_size, callback,
-                            Socket::GetNetworkTrafficAnnotationTag());
-}
-
-bool TLSSocket::SetKeepAlive(bool enable, int delay) {
-  return false;
-}
-
-bool TLSSocket::SetNoDelay(bool no_delay) {
-  return false;
-}
-
-int TLSSocket::Listen(const std::string& address,
-                      uint16_t port,
-                      int backlog,
-                      std::string* error_msg) {
-  *error_msg = kTLSSocketTypeInvalidError;
-  return net::ERR_NOT_IMPLEMENTED;
-}
-
-void TLSSocket::Accept(const AcceptCompletionCallback& callback) {
-  callback.Run(net::ERR_FAILED, NULL);
+void TLSSocket::Listen(const std::string& address,
+                       uint16_t port,
+                       int backlog,
+                       ListenCallback callback) {
+  std::move(callback).Run(net::ERR_NOT_IMPLEMENTED, kTLSSocketTypeInvalidError);
 }
 
 bool TLSSocket::IsConnected() {
-  return tls_socket_.get() && tls_socket_->IsConnected();
+  return is_connected_;
 }
 
 bool TLSSocket::GetPeerAddress(net::IPEndPoint* address) {
-  return IsConnected() && tls_socket_->GetPeerAddress(address);
+  if (!IsConnected())
+    return false;
+  if (!peer_addr_)
+    return false;
+  *address = peer_addr_.value();
+  return true;
 }
 
 bool TLSSocket::GetLocalAddress(net::IPEndPoint* address) {
-  return IsConnected() && tls_socket_->GetLocalAddress(address);
+  if (!IsConnected())
+    return false;
+  if (!local_addr_)
+    return false;
+  *address = local_addr_.value();
+  return true;
 }
 
 Socket::SocketType TLSSocket::GetSocketType() const {
   return Socket::TYPE_TLS;
 }
 
-// static
-void TLSSocket::UpgradeSocketToTLS(
-    Socket* socket,
-    net::SSLConfigService* ssl_config_service,
-    net::CertVerifier* cert_verifier,
-    net::TransportSecurityState* transport_security_state,
-    net::CTVerifier* ct_verifier,
-    net::CTPolicyEnforcer* ct_policy_enforcer,
-    const std::string& extension_id,
-    api::socket::SecureOptions* options,
-    const TLSSocket::SecureCallback& callback) {
-  DCHECK_CURRENTLY_ON(content::BrowserThread::IO);
-  TCPSocket* tcp_socket = static_cast<TCPSocket*>(socket);
-  std::unique_ptr<net::SSLClientSocket> null_sock;
+int TLSSocket::WriteImpl(net::IOBuffer* io_buffer,
+                         int io_buffer_size,
+                         const net::CompletionCallback& callback) {
+  if (!mojo_data_pump_)
+    return net::ERR_SOCKET_NOT_CONNECTED;
+  mojo_data_pump_->Write(io_buffer, io_buffer_size,
+                         base::BindOnce(&TLSSocket::OnWriteComplete,
+                                        base::Unretained(this), callback));
+  return net::ERR_IO_PENDING;
+}
 
-  if (!tcp_socket || tcp_socket->GetSocketType() != Socket::TYPE_TCP ||
-      !tcp_socket->ClientStream() || !tcp_socket->IsConnected() ||
-      tcp_socket->HasPendingRead()) {
-    DVLOG(1) << "Failing before trying. socket is " << tcp_socket;
-    if (tcp_socket) {
-      DVLOG(1) << "type: " << tcp_socket->GetSocketType()
-               << ", ClientStream is " << tcp_socket->ClientStream()
-               << ", IsConnected: " << tcp_socket->IsConnected()
-               << ", HasPendingRead: " << tcp_socket->HasPendingRead();
-    }
-    TlsConnectDone(std::move(null_sock), extension_id, callback,
-                   net::ERR_INVALID_ARGUMENT);
-    return;
+void TLSSocket::OnWriteComplete(const net::CompletionCallback& callback,
+                                int result) {
+  if (result < 0) {
+    // Write side has terminated. This can be an error or a graceful close.
+    // TCPSocketEventDispatcher doesn't distinguish between the two.
+    Disconnect(false /* socket_destroying */);
   }
+  callback.Run(result);
+}
 
-  net::IPEndPoint dest_host_port_pair;
-  if (!tcp_socket->GetPeerAddress(&dest_host_port_pair)) {
-    DVLOG(1) << "Could not get peer address.";
-    TlsConnectDone(std::move(null_sock), extension_id, callback,
-                   net::ERR_INVALID_ARGUMENT);
-    return;
+void TLSSocket::OnReadComplete(int result,
+                               scoped_refptr<net::IOBuffer> io_buffer) {
+  DCHECK(read_callback_);
+
+  // Use a local variable for |read_callback_|, because otherwise Disconnect()
+  // will try to invoke |read_callback_| with a hardcoded result code.
+  ReadCompletionCallback callback = std::move(read_callback_);
+  if (result < 0) {
+    // Read side has terminated. This can be an error or a graceful close.
+    // TCPSocketEventDispatcher doesn't distinguish between the two.
+    Disconnect(false /* socket_destroying */);
   }
-
-  // Convert any U-LABELs to A-LABELs.
-  url::CanonHostInfo host_info;
-  std::string canon_host =
-      net::CanonicalizeHost(tcp_socket->hostname(), &host_info);
-
-  // Canonicalization shouldn't fail: the socket is already connected with a
-  // host, using this hostname.
-  if (host_info.family == url::CanonHostInfo::BROKEN) {
-    DVLOG(1) << "Could not canonicalize hostname";
-    TlsConnectDone(std::move(null_sock), extension_id, callback,
-                   net::ERR_INVALID_ARGUMENT);
-    return;
-  }
-
-  net::HostPortPair host_and_port(canon_host, dest_host_port_pair.port());
-
-  std::unique_ptr<net::ClientSocketHandle> socket_handle(
-      new net::ClientSocketHandle());
-
-  // Set the socket handle to the socket's client stream (that should be the
-  // only one active here). Then have the old socket release ownership on
-  // that client stream.
-  socket_handle->SetSocket(
-      base::WrapUnique<net::StreamSocket>(tcp_socket->ClientStream()));
-  tcp_socket->Release();
-
-  DCHECK(transport_security_state);
-  net::SSLClientSocketContext context;
-  context.cert_verifier = cert_verifier;
-  context.transport_security_state = transport_security_state;
-  context.cert_transparency_verifier = ct_verifier;
-  context.ct_policy_enforcer = ct_policy_enforcer;
-
-  // Fill in the SSL socket params.
-  net::SSLConfig ssl_config;
-  ssl_config_service->GetSSLConfig(&ssl_config);
-  if (options && options->tls_version.get()) {
-    uint16_t version_min = 0, version_max = 0;
-    api::socket::TLSVersionConstraints* versions = options->tls_version.get();
-    if (versions->min.get()) {
-      version_min = SSLProtocolVersionFromString(*versions->min);
-    }
-    if (versions->max.get()) {
-      version_max = SSLProtocolVersionFromString(*versions->max);
-    }
-    if (version_min) {
-      ssl_config.version_min = version_min;
-    }
-    if (version_max) {
-      ssl_config.version_max = version_max;
-    }
-  }
-
-  net::ClientSocketFactory* socket_factory =
-      net::ClientSocketFactory::GetDefaultFactory();
-
-  // Create the socket.
-  std::unique_ptr<net::SSLClientSocket> ssl_socket(
-      socket_factory->CreateSSLClientSocket(
-          std::move(socket_handle), host_and_port, ssl_config, context));
-
-  DVLOG(1) << "Attempting to secure a connection to " << tcp_socket->hostname()
-           << ":" << dest_host_port_pair.port();
-
-  // We need the contents of |ssl_socket| in order to invoke its Connect()
-  // method. It belongs to |ssl_socket|, and we own that until our internal
-  // callback (|connect_cb|, below) is invoked.
-  net::SSLClientSocket* saved_ssl_socket = ssl_socket.get();
-
-  // Try establish a TLS connection. Pass ownership of |ssl_socket| to
-  // TlsConnectDone, which will pass it on to |callback|. |connect_cb| below
-  // is only for UpgradeSocketToTLS use, and not be confused with the
-  // argument |callback|, which gets invoked by TlsConnectDone() after
-  // Connect() below returns.
-  base::Callback<void(int)> connect_cb(base::Bind(
-      &TlsConnectDone, base::Passed(&ssl_socket), extension_id, callback));
-  int status = saved_ssl_socket->Connect(connect_cb);
-  saved_ssl_socket = NULL;
-
-  // Connect completed synchronously, or failed.
-  if (status != net::ERR_IO_PENDING) {
-    // Note: this can't recurse -- if |socket| is already a connected
-    // TLSSocket, it will return TYPE_TLS instead of TYPE_TCP, causing
-    // UpgradeSocketToTLS() to fail with an error above. If
-    // UpgradeSocketToTLS() is called on |socket| twice, the call to
-    // Release() on |socket| above causes the additional call to
-    // fail with an error above.
-    if (status != net::OK) {
-      DVLOG(1) << "Status is not OK or IO-pending: "
-               << net::ErrorToString(status);
-    }
-    connect_cb.Run(status);
-  }
+  std::move(callback).Run(result, io_buffer, false /* socket_destroying */);
 }
 
 }  // namespace extensions
-
diff --git a/extensions/browser/api/socket/tls_socket.h b/extensions/browser/api/socket/tls_socket.h
index c3208e2..99846ea 100644
--- a/extensions/browser/api/socket/tls_socket.h
+++ b/extensions/browser/api/socket/tls_socket.h
@@ -7,24 +7,19 @@
 
 #include <stdint.h>
 
+#include <memory>
 #include <string>
 
+#include "extensions/browser/api/socket/mojo_data_pump.h"
 #include "extensions/browser/api/socket/socket.h"
 #include "extensions/browser/api/socket/socket_api.h"
 #include "extensions/browser/api/socket/tcp_socket.h"
-#include "net/ssl/ssl_config_service.h"
-
-namespace net {
-class Socket;
-class CertVerifier;
-class CTPolicyEnforcer;
-class CTVerifier;
-class TransportSecurityState;
-}
+#include "mojo/public/cpp/system/data_pipe.h"
+#include "services/network/public/mojom/tls_socket.mojom.h"
 
 namespace extensions {
 
-class TLSSocket;
+class MojoDataPump;
 
 // TLS Sockets from the chrome.socket and chrome.sockets.tcp APIs. A regular
 // TCPSocket is converted to a TLSSocket via chrome.socket.secure() or
@@ -38,84 +33,55 @@
 // touch any socket state.
 class TLSSocket : public ResumableTCPSocket {
  public:
-  typedef base::Callback<void(std::unique_ptr<TLSSocket>, int)> SecureCallback;
-
-  TLSSocket(std::unique_ptr<net::StreamSocket> tls_socket,
+  TLSSocket(network::mojom::TLSClientSocketPtr tls_socket,
+            const net::IPEndPoint& local_addr,
+            const net::IPEndPoint& peer_addr,
+            mojo::ScopedDataPipeConsumerHandle receive_stream,
+            mojo::ScopedDataPipeProducerHandle send_stream,
             const std::string& owner_extension_id);
 
   ~TLSSocket() override;
 
-  // Most of these methods either fail or forward the method call on to the
-  // inner net::StreamSocket. The remaining few do actual TLS work.
-
   // Fails.
   void Connect(const net::AddressList& address,
-               const CompletionCallback& callback) override;
+               net::CompletionOnceCallback callback) override;
   // Forwards.
   void Disconnect(bool socket_destroying) override;
 
   // Attempts to read |count| bytes of decrypted data from the TLS socket,
   // invoking |callback| with the actual number of bytes read, or a network
   // error code if an error occurred.
-  void Read(int count, const ReadCompletionCallback& callback) override;
-
-  // Fails. This should have been called on the TCP socket before secure() was
-  // invoked.
-  bool SetKeepAlive(bool enable, int delay) override;
-
-  // Fails. This should have been called on the TCP socket before secure() was
-  // invoked.
-  bool SetNoDelay(bool no_delay) override;
+  void Read(int count, ReadCompletionCallback callback) override;
 
   // Fails. TLSSocket is only a client.
-  int Listen(const std::string& address,
-             uint16_t port,
-             int backlog,
-             std::string* error_msg) override;
-
-  // Fails. TLSSocket is only a client.
-  void Accept(const AcceptCompletionCallback& callback) override;
+  void Listen(const std::string& address,
+              uint16_t port,
+              int backlog,
+              ListenCallback callback) override;
 
   // Forwards.
   bool IsConnected() override;
 
-  // Forwards.
   bool GetPeerAddress(net::IPEndPoint* address) override;
-  // Forwards.
   bool GetLocalAddress(net::IPEndPoint* address) override;
 
   // Returns TYPE_TLS.
   SocketType GetSocketType() const override;
 
-  // Convert |socket| to a TLS socket. |socket| must be an open TCP client
-  // socket. |socket| must not have a pending read. UpgradeSocketToTLS() must
-  // be invoked in the IO thread. |callback| will always be invoked. |options|
-  // may be NULL.
-  // Note: |callback| may be synchronously invoked before
-  // UpgradeSocketToTLS() returns. Currently using the older chrome.socket
-  // version of SecureOptions, to avoid having the older API implementation
-  // depend on the newer one.
-  static void UpgradeSocketToTLS(
-      Socket* socket,
-      net::SSLConfigService* config_service,
-      net::CertVerifier* cert_verifier,
-      net::TransportSecurityState* transport_security_state,
-      net::CTVerifier* ct_verifier,
-      net::CTPolicyEnforcer* ct_policy_enforcer,
-      const std::string& extension_id,
-      api::socket::SecureOptions* options,
-      const SecureCallback& callback);
-
  private:
   int WriteImpl(net::IOBuffer* io_buffer,
                 int io_buffer_size,
                 const net::CompletionCallback& callback) override;
+  void OnWriteComplete(const net::CompletionCallback& callback, int result);
+  void OnReadComplete(int result, scoped_refptr<net::IOBuffer> io_buffer);
 
-  void OnReadComplete(const scoped_refptr<net::IOBuffer>& io_buffer,
-                      int result);
-
-  std::unique_ptr<net::StreamSocket> tls_socket_;
+  network::mojom::TLSClientSocketPtr tls_socket_;
+  base::Optional<net::IPEndPoint> local_addr_;
+  base::Optional<net::IPEndPoint> peer_addr_;
+  std::unique_ptr<MojoDataPump> mojo_data_pump_;
   ReadCompletionCallback read_callback_;
+
+  DISALLOW_COPY_AND_ASSIGN(TLSSocket);
 };
 
 }  // namespace extensions
diff --git a/extensions/browser/api/socket/udp_socket.cc b/extensions/browser/api/socket/udp_socket.cc
index 094508e3..f9b8aa9 100644
--- a/extensions/browser/api/socket/udp_socket.cc
+++ b/extensions/browser/api/socket/udp_socket.cc
@@ -5,6 +5,7 @@
 #include "extensions/browser/api/socket/udp_socket.h"
 
 #include <algorithm>
+#include <utility>
 
 #include "base/callback_helpers.h"
 #include "base/lazy_instance.h"
@@ -46,9 +47,9 @@
 }
 
 void UDPSocket::Connect(const net::AddressList& address,
-                        const net::CompletionCallback& callback) {
+                        net::CompletionOnceCallback callback) {
   if (IsConnectedOrBound()) {
-    callback.Run(net::ERR_CONNECTION_FAILED);
+    std::move(callback).Run(net::ERR_CONNECTION_FAILED);
     return;
   }
   // UDP API only connects to the first address received from DNS so
@@ -57,7 +58,7 @@
   socket_->Connect(
       ip_end_point, std::move(socket_options_),
       base::BindOnce(&UDPSocket::OnConnectCompleted, base::Unretained(this),
-                     callback, ip_end_point));
+                     std::move(callback), ip_end_point));
 }
 
 void UDPSocket::Bind(const std::string& address,
@@ -94,27 +95,28 @@
   multicast_groups_.clear();
 }
 
-void UDPSocket::Read(int count, const ReadCompletionCallback& callback) {
+void UDPSocket::Read(int count, ReadCompletionCallback callback) {
   DCHECK(!callback.is_null());
 
   if (!read_callback_.is_null()) {
-    callback.Run(net::ERR_IO_PENDING, nullptr, false /* socket_destroying */);
+    std::move(callback).Run(net::ERR_IO_PENDING, nullptr,
+                            false /* socket_destroying */);
     return;
   }
 
   if (count < 0) {
-    callback.Run(net::ERR_INVALID_ARGUMENT, nullptr,
-                 false /* socket_destroying */);
+    std::move(callback).Run(net::ERR_INVALID_ARGUMENT, nullptr,
+                            false /* socket_destroying */);
     return;
   }
 
   if (!IsConnected()) {
-    callback.Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr,
-                 false /* socket_destroying */);
+    std::move(callback).Run(net::ERR_SOCKET_NOT_CONNECTED, nullptr,
+                            false /* socket_destroying */);
     return;
   }
 
-  read_callback_ = callback;
+  read_callback_ = std::move(callback);
   socket_->ReceiveMoreWithBufferSize(1, count);
   return;
 }
@@ -220,7 +222,7 @@
   uint16_t port = 0;
   if (result != net::OK) {
     if (!read_callback_.is_null()) {
-      base::ResetAndReturn(&read_callback_)
+      std::move(read_callback_)
           .Run(result, nullptr, false /* socket_destroying */);
       return;
     }
@@ -234,7 +236,7 @@
   memcpy(io_buffer->data(), data.value().data(), data.value().size());
 
   if (!read_callback_.is_null()) {
-    base::ResetAndReturn(&read_callback_)
+    std::move(read_callback_)
         .Run(data.value().size(), io_buffer, false /* socket_destroying */);
     return;
   }
@@ -246,18 +248,18 @@
 }
 
 void UDPSocket::OnConnectCompleted(
-    const net::CompletionCallback& callback,
+    net::CompletionOnceCallback callback,
     const net::IPEndPoint& remote_addr,
     int result,
     const base::Optional<net::IPEndPoint>& local_addr) {
   if (result != net::OK) {
-    callback.Run(result);
+    std::move(callback).Run(result);
     return;
   }
   local_addr_ = local_addr;
   peer_addr_ = remote_addr;
   is_connected_ = true;
-  callback.Run(result);
+  std::move(callback).Run(result);
 }
 
 void UDPSocket::OnBindCompleted(
diff --git a/extensions/browser/api/socket/udp_socket.h b/extensions/browser/api/socket/udp_socket.h
index 6f46aaa..9c4fdfc 100644
--- a/extensions/browser/api/socket/udp_socket.h
+++ b/extensions/browser/api/socket/udp_socket.h
@@ -28,12 +28,12 @@
 
   // Socket implementation.
   void Connect(const net::AddressList& address,
-               const CompletionCallback& callback) override;
+               net::CompletionOnceCallback callback) override;
   void Disconnect(bool socket_destroying) override;
   void Bind(const std::string& address,
             uint16_t port,
             const CompletionCallback& callback) override;
-  void Read(int count, const ReadCompletionCallback& callback) override;
+  void Read(int count, ReadCompletionCallback callback) override;
   void RecvFrom(int count, const RecvFromCompletionCallback& callback) override;
   void SendTo(scoped_refptr<net::IOBuffer> io_buffer,
               int byte_count,
@@ -76,7 +76,7 @@
                   const base::Optional<net::IPEndPoint>& src_addr,
                   base::Optional<base::span<const uint8_t>> data) override;
 
-  void OnConnectCompleted(const net::CompletionCallback& user_callback,
+  void OnConnectCompleted(net::CompletionOnceCallback user_callback,
                           const net::IPEndPoint& remote_addr,
                           int result,
                           const base::Optional<net::IPEndPoint>& local_addr);
diff --git a/extensions/browser/api/sockets_tcp/sockets_tcp_api.cc b/extensions/browser/api/sockets_tcp/sockets_tcp_api.cc
index dbc43166..c0aeac5 100644
--- a/extensions/browser/api/sockets_tcp/sockets_tcp_api.cc
+++ b/extensions/browser/api/sockets_tcp/sockets_tcp_api.cc
@@ -4,6 +4,10 @@
 
 #include "extensions/browser/api/sockets_tcp/sockets_tcp_api.h"
 
+#include <string>
+#include <utility>
+#include <vector>
+
 #include "content/public/browser/browser_context.h"
 #include "content/public/browser/browser_thread.h"
 #include "content/public/browser/storage_partition.h"
@@ -124,7 +128,8 @@
 }
 
 void SocketsTcpCreateFunction::Work() {
-  ResumableTCPSocket* socket = new ResumableTCPSocket(extension_->id());
+  ResumableTCPSocket* socket =
+      new ResumableTCPSocket(browser_context(), extension_->id());
 
   sockets_tcp::SocketProperties* properties = params_->properties.get();
   if (properties) {
@@ -203,20 +208,28 @@
   return true;
 }
 
-void SocketsTcpSetKeepAliveFunction::Work() {
+void SocketsTcpSetKeepAliveFunction::AsyncWorkStart() {
   ResumableTCPSocket* socket = GetTcpSocket(params_->socket_id);
   if (!socket) {
     error_ = kSocketNotFoundError;
+    results_ = sockets_tcp::SetKeepAlive::Results::Create(net::ERR_FAILED);
+    AsyncWorkCompleted();
     return;
   }
 
   int delay = params_->delay ? *params_->delay : 0;
 
-  bool success = socket->SetKeepAlive(params_->enable, delay);
+  socket->SetKeepAlive(
+      params_->enable, delay,
+      base::BindOnce(&SocketsTcpSetKeepAliveFunction::OnCompleted, this));
+}
+
+void SocketsTcpSetKeepAliveFunction::OnCompleted(bool success) {
   int net_result = (success ? net::OK : net::ERR_FAILED);
+  results_ = sockets_tcp::SetKeepAlive::Results::Create(net_result);
   if (net_result != net::OK)
     error_ = net::ErrorToString(net_result);
-  results_ = sockets_tcp::SetKeepAlive::Results::Create(net_result);
+  AsyncWorkCompleted();
 }
 
 SocketsTcpSetNoDelayFunction::SocketsTcpSetNoDelayFunction() {}
@@ -229,18 +242,25 @@
   return true;
 }
 
-void SocketsTcpSetNoDelayFunction::Work() {
+void SocketsTcpSetNoDelayFunction::AsyncWorkStart() {
   ResumableTCPSocket* socket = GetTcpSocket(params_->socket_id);
   if (!socket) {
     error_ = kSocketNotFoundError;
+    results_ = sockets_tcp::SetNoDelay::Results::Create(net::ERR_FAILED);
+    AsyncWorkCompleted();
     return;
   }
+  socket->SetNoDelay(
+      params_->no_delay,
+      base::BindOnce(&SocketsTcpSetNoDelayFunction::OnCompleted, this));
+}
 
-  bool success = socket->SetNoDelay(params_->no_delay);
+void SocketsTcpSetNoDelayFunction::OnCompleted(bool success) {
   int net_result = (success ? net::OK : net::ERR_FAILED);
+  results_ = sockets_tcp::SetNoDelay::Results::Create(net_result);
   if (net_result != net::OK)
     error_ = net::ErrorToString(net_result);
-  results_ = sockets_tcp::SetNoDelay::Results::Create(net_result);
+  AsyncWorkCompleted();
 }
 
 SocketsTcpConnectFunction::SocketsTcpConnectFunction()
@@ -458,8 +478,6 @@
   DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   params_ = api::sockets_tcp::Secure::Params::Create(*args_);
   EXTENSION_FUNCTION_VALIDATE(params_.get());
-  url_request_getter_ = content::BrowserContext::GetDefaultStoragePartition(
-      browser_context())->GetURLRequestContext();
   return true;
 }
 
@@ -481,8 +499,7 @@
 
   // Make sure it's a connected TCP client socket. Error out if it's already
   // secure()'d.
-  if (socket->GetSocketType() != Socket::TYPE_TCP ||
-      socket->ClientStream() == NULL) {
+  if (socket->GetSocketType() != Socket::TYPE_TCP) {
     SetResult(std::make_unique<base::Value>(net::ERR_INVALID_ARGUMENT));
     error_ = kInvalidSocketStateError;
     AsyncWorkCompleted();
@@ -496,9 +513,6 @@
     return;
   }
 
-  net::URLRequestContext* url_request_context =
-      url_request_getter_->GetURLRequestContext();
-
   // UpgradeSocketToTLS() uses the older API's SecureOptions. Copy over the
   // only values inside -- TLSVersionConstraints's |min| and |max|,
   api::socket::SecureOptions legacy_params;
@@ -514,29 +528,33 @@
     }
   }
 
-  TLSSocket::UpgradeSocketToTLS(
-      socket, url_request_context->ssl_config_service(),
-      url_request_context->cert_verifier(),
-      url_request_context->transport_security_state(),
-      url_request_context->cert_transparency_verifier(),
-      url_request_context->ct_policy_enforcer(), extension_id(), &legacy_params,
-      base::Bind(&SocketsTcpSecureFunction::TlsConnectDone, this));
+  network::mojom::TLSClientSocketPtr tls_socket;
+  socket->UpgradeToTLS(
+      &legacy_params,
+      base::BindOnce(&SocketsTcpSecureFunction::TlsConnectDone, this));
 }
 
-void SocketsTcpSecureFunction::TlsConnectDone(std::unique_ptr<TLSSocket> socket,
-                                              int result) {
-  // If an error occurred, socket MUST be NULL
-  DCHECK(result == net::OK || socket == NULL);
-
-  if (socket && result == net::OK) {
-    socket->set_persistent(persistent_);
-    socket->set_paused(paused_);
-    ReplaceSocket(params_->socket_id, socket.release());
-  } else {
+void SocketsTcpSecureFunction::TlsConnectDone(
+    int result,
+    network::mojom::TLSClientSocketPtr tls_socket,
+    const net::IPEndPoint& local_addr,
+    const net::IPEndPoint& peer_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+    mojo::ScopedDataPipeProducerHandle send_pipe_handle) {
+  if (result != net::OK) {
     RemoveSocket(params_->socket_id);
     error_ = net::ErrorToString(result);
+    results_ = api::sockets_tcp::Secure::Results::Create(result);
+    AsyncWorkCompleted();
+    return;
   }
-
+  auto socket =
+      std::make_unique<TLSSocket>(std::move(tls_socket), local_addr, peer_addr,
+                                  std::move(receive_pipe_handle),
+                                  std::move(send_pipe_handle), extension_id());
+  socket->set_persistent(persistent_);
+  socket->set_paused(paused_);
+  ReplaceSocket(params_->socket_id, socket.release());
   results_ = api::sockets_tcp::Secure::Results::Create(result);
   AsyncWorkCompleted();
 }
diff --git a/extensions/browser/api/sockets_tcp/sockets_tcp_api.h b/extensions/browser/api/sockets_tcp/sockets_tcp_api.h
index c4b3ee8..fd004e7 100644
--- a/extensions/browser/api/sockets_tcp/sockets_tcp_api.h
+++ b/extensions/browser/api/sockets_tcp/sockets_tcp_api.h
@@ -7,14 +7,16 @@
 
 #include <stddef.h>
 
+#include <memory>
+
 #include "base/gtest_prod_util.h"
 #include "base/macros.h"
 #include "extensions/browser/api/socket/socket_api.h"
 #include "extensions/common/api/sockets_tcp.h"
+#include "services/network/public/mojom/tcp_socket.mojom.h"
 
 namespace extensions {
 class ResumableTCPSocket;
-class TLSSocket;
 }
 
 namespace extensions {
@@ -108,9 +110,11 @@
 
   // AsyncApiFunction
   bool Prepare() override;
-  void Work() override;
+  void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(bool success);
+
   std::unique_ptr<sockets_tcp::SetKeepAlive::Params> params_;
 };
 
@@ -125,9 +129,11 @@
 
   // AsyncApiFunction
   bool Prepare() override;
-  void Work() override;
+  void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(bool success);
+
   std::unique_ptr<sockets_tcp::SetNoDelay::Params> params_;
 };
 
@@ -255,13 +261,16 @@
   void AsyncWorkStart() override;
 
  private:
-  virtual void TlsConnectDone(std::unique_ptr<extensions::TLSSocket> sock,
-                              int result);
+  void TlsConnectDone(int result,
+                      network::mojom::TLSClientSocketPtr tls_socket,
+                      const net::IPEndPoint& local_addr,
+                      const net::IPEndPoint& peer_addr,
+                      mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+                      mojo::ScopedDataPipeProducerHandle send_pipe_handle);
 
   bool paused_;
   bool persistent_;
   std::unique_ptr<sockets_tcp::Secure::Params> params_;
-  scoped_refptr<net::URLRequestContextGetter> url_request_getter_;
 
   DISALLOW_COPY_AND_ASSIGN(SocketsTcpSecureFunction);
 };
diff --git a/extensions/browser/api/sockets_tcp/sockets_tcp_apitest.cc b/extensions/browser/api/sockets_tcp/sockets_tcp_apitest.cc
index 4d62b093..918b2a0e 100644
--- a/extensions/browser/api/sockets_tcp/sockets_tcp_apitest.cc
+++ b/extensions/browser/api/sockets_tcp/sockets_tcp_apitest.cc
@@ -4,8 +4,14 @@
 
 #include <utility>
 
+#include "base/command_line.h"
 #include "base/memory/ref_counted.h"
 #include "base/strings/stringprintf.h"
+#include "content/public/common/content_features.h"
+#include "content/public/common/content_switches.h"
+#include "content/public/common/service_manager_connection.h"
+#include "content/public/common/service_names.mojom.h"
+#include "content/public/test/network_service_test_helper.h"
 #include "extensions/browser/api/dns/host_resolver_wrapper.h"
 #include "extensions/browser/api/dns/mock_host_resolver_creator.h"
 #include "extensions/browser/api/sockets_tcp/sockets_tcp_api.h"
@@ -16,8 +22,11 @@
 #include "extensions/shell/test/shell_test.h"
 #include "extensions/test/extension_test_message_listener.h"
 #include "extensions/test/result_catcher.h"
+#include "mojo/public/cpp/bindings/sync_call_restrictions.h"
 #include "net/dns/mock_host_resolver.h"
 #include "net/test/spawned_test_server/spawned_test_server.h"
+#include "services/network/public/cpp/features.h"
+#include "services/service_manager/public/cpp/connector.h"
 
 namespace extensions {
 
@@ -25,14 +34,18 @@
 
 class SocketsTcpApiTest : public ShellApiTest {
  public:
-  SocketsTcpApiTest()
-      : resolver_event_(base::WaitableEvent::ResetPolicy::MANUAL,
-                        base::WaitableEvent::InitialState::NOT_SIGNALED),
-        resolver_creator_(new MockHostResolverCreator()) {}
+  SocketsTcpApiTest() : resolver_creator_(new MockHostResolverCreator()) {
+    if (base::FeatureList::IsEnabled(network::features::kNetworkService)) {
+      base::CommandLine::ForCurrentProcess()->AppendSwitch(
+          switches::kUseMockCertVerifierForTesting);
+    }
+  }
 
   void SetUpOnMainThread() override {
     ShellApiTest::SetUpOnMainThread();
 
+    // TODO(xunjieli): Figure out what to do with this in-process host resolver
+    // which doesn't work with the out-of-process network service.
     HostResolverWrapper::GetInstance()->SetHostResolverForTesting(
         resolver_creator_->CreateMockHostResolver());
   }
@@ -45,8 +58,6 @@
   }
 
  private:
-  base::WaitableEvent resolver_event_;
-
   // The MockHostResolver asserts that it's used on the same thread on which
   // it's created, which is actually a stronger rule than its real counterpart.
   // But that's fine; it's good practice.
@@ -101,6 +112,19 @@
 }
 
 IN_PROC_BROWSER_TEST_F(SocketsTcpApiTest, SocketTcpExtensionTLS) {
+  network::mojom::NetworkServiceTestPtr network_service_test;
+  // If network service is running in a utility process, the cert of the
+  // SpawnedTestServer won't be recognized, so inject mock cert verifier through
+  // the test helper interface.
+  if (base::FeatureList::IsEnabled(network::features::kNetworkService)) {
+    content::ServiceManagerConnection::GetForProcess()
+        ->GetConnector()
+        ->BindInterface(content::mojom::kNetworkServiceName,
+                        &network_service_test);
+    mojo::ScopedAllowSyncCallForTesting allow_sync_call;
+    network_service_test->MockCertVerifierSetDefaultResult(net::OK);
+  }
+
   std::unique_ptr<net::SpawnedTestServer> test_https_server(
       new net::SpawnedTestServer(
           net::SpawnedTestServer::TYPE_HTTPS, net::BaseTestServer::SSLOptions(),
diff --git a/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.cc b/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.cc
index 021ea5f..45b8a780 100644
--- a/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.cc
+++ b/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.cc
@@ -4,6 +4,8 @@
 
 #include "extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h"
 
+#include <vector>
+
 #include "content/public/common/socket_permission_request.h"
 #include "extensions/browser/api/socket/tcp_socket.h"
 #include "extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.h"
@@ -84,8 +86,8 @@
 }
 
 void SocketsTcpServerCreateFunction::Work() {
-  ResumableTCPServerSocket* socket =
-      new ResumableTCPServerSocket(extension_->id());
+  auto* socket =
+      new ResumableTCPServerSocket(browser_context(), extension_->id());
 
   sockets_tcp_server::SocketProperties* properties = params_->properties.get();
   if (properties) {
@@ -190,10 +192,23 @@
     return;
   }
 
-  int net_result = socket->Listen(
+  socket->Listen(
       params_->address, params_->port,
       params_->backlog.get() ? *params_->backlog : kDefaultListenBacklog,
-      &error_);
+      base::BindOnce(&SocketsTcpServerListenFunction::OnCompleted, this));
+}
+
+void SocketsTcpServerListenFunction::OnCompleted(
+    int net_result,
+    const std::string& /* error_msg */) {
+  DCHECK_NE(net::ERR_IO_PENDING, net_result);
+
+  ResumableTCPServerSocket* socket = GetTcpSocket(params_->socket_id);
+  if (!socket) {
+    error_ = kSocketNotFoundError;
+    AsyncWorkCompleted();
+    return;
+  }
   results_ = sockets_tcp_server::Listen::Results::Create(net_result);
   if (net_result == net::OK) {
     socket_event_dispatcher_->OnServerSocketListen(extension_->id(),
diff --git a/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h b/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h
index b5f93c63..96ed487 100644
--- a/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h
+++ b/extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h
@@ -5,6 +5,9 @@
 #ifndef EXTENSIONS_BROWSER_API_SOCKETS_TCP_SERVER_SOCKETS_TCP_SERVER_API_H_
 #define EXTENSIONS_BROWSER_API_SOCKETS_TCP_SERVER_SOCKETS_TCP_SERVER_API_H_
 
+#include <memory>
+#include <string>
+
 #include "base/gtest_prod_util.h"
 #include "extensions/browser/api/socket/socket_api.h"
 #include "extensions/common/api/sockets_tcp_server.h"
@@ -98,6 +101,8 @@
   void AsyncWorkStart() override;
 
  private:
+  void OnCompleted(int result, const std::string& error_msg);
+
   std::unique_ptr<sockets_tcp_server::Listen::Params> params_;
   TCPServerSocketEventDispatcher* socket_event_dispatcher_;
 };
diff --git a/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.cc b/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.cc
index 4854e8b7..565a037 100644
--- a/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.cc
+++ b/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.cc
@@ -117,19 +117,24 @@
     return;
 
   socket->Accept(
-      base::Bind(&TCPServerSocketEventDispatcher::AcceptCallback, params));
+      base::BindOnce(&TCPServerSocketEventDispatcher::AcceptCallback, params));
 }
 
 // static
 void TCPServerSocketEventDispatcher::AcceptCallback(
     const AcceptParams& params,
     int result_code,
-    std::unique_ptr<net::TCPClientSocket> socket) {
+    network::mojom::TCPConnectedSocketPtr socket,
+    const base::Optional<net::IPEndPoint>& remote_addr,
+    mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+    mojo::ScopedDataPipeProducerHandle send_pipe_handle) {
   DCHECK_CURRENTLY_ON(params.thread_id);
+  DCHECK_GE(net::OK, result_code);
 
-  if (result_code >= 0) {
-    ResumableTCPSocket* client_socket =
-        new ResumableTCPSocket(std::move(socket), params.extension_id, true);
+  if (result_code == net::OK) {
+    ResumableTCPSocket* client_socket = new ResumableTCPSocket(
+        std::move(socket), std::move(receive_pipe_handle),
+        std::move(send_pipe_handle), remote_addr, params.extension_id);
     client_socket->set_paused(true);
     int client_socket_id = params.client_sockets->Add(client_socket);
 
diff --git a/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.h b/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.h
index 98fc928d..3da1e491 100644
--- a/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.h
+++ b/extensions/browser/api/sockets_tcp_server/tcp_server_socket_event_dispatcher.h
@@ -5,10 +5,14 @@
 #ifndef EXTENSIONS_BROWSER_API_SOCKETS_TCP_SERVER_TCP_SERVER_SOCKET_EVENT_DISPATCHER_H_
 #define EXTENSIONS_BROWSER_API_SOCKETS_TCP_SERVER_TCP_SERVER_SOCKET_EVENT_DISPATCHER_H_
 
+#include <memory>
+#include <string>
+
 #include "content/public/browser/browser_thread.h"
 #include "extensions/browser/api/api_resource_manager.h"
 #include "extensions/browser/api/sockets_tcp/sockets_tcp_api.h"
 #include "extensions/browser/api/sockets_tcp_server/sockets_tcp_server_api.h"
+#include "services/network/public/mojom/tcp_socket.mojom.h"
 
 namespace content {
 class BrowserContext;
@@ -77,9 +81,13 @@
   static void StartAccept(const AcceptParams& params);
 
   // Called when socket accepts a new connection.
-  static void AcceptCallback(const AcceptParams& params,
-                             int result_code,
-                             std::unique_ptr<net::TCPClientSocket> socket);
+  static void AcceptCallback(
+      const AcceptParams& params,
+      int result,
+      network::mojom::TCPConnectedSocketPtr socket,
+      const base::Optional<net::IPEndPoint>& remote_addr,
+      mojo::ScopedDataPipeConsumerHandle receive_pipe_handle,
+      mojo::ScopedDataPipeProducerHandle send_pipe_handle);
 
   // Post an extension event from |thread_id| to UI thread
   static void PostEvent(const AcceptParams& params,
diff --git a/extensions/browser/api/system_display/system_display_apitest.cc b/extensions/browser/api/system_display/system_display_apitest.cc
index 455341ea..54d9f99 100644
--- a/extensions/browser/api/system_display/system_display_apitest.cc
+++ b/extensions/browser/api/system_display/system_display_apitest.cc
@@ -244,7 +244,13 @@
 
 #if !defined(OS_CHROMEOS)
 
-IN_PROC_BROWSER_TEST_F(SystemDisplayApiTest, SetDisplay) {
+// TODO(crbug.com/866904): Fails sometimes on Win 10 dbg bots.
+#if defined(OS_WIN)
+#define MAYBE_SetDisplay DISABLED_SetDisplay
+#else
+#define MAYBE_SetDisplay SetDisplay
+#endif  // defined(OS_WIN)
+IN_PROC_BROWSER_TEST_F(SystemDisplayApiTest, MAYBE_SetDisplay) {
   scoped_refptr<SystemDisplaySetDisplayPropertiesFunction> set_info_function(
       new SystemDisplaySetDisplayPropertiesFunction());
 
diff --git a/gin/arguments.cc b/gin/arguments.cc
index b492fae6..82d5f43 100644
--- a/gin/arguments.cc
+++ b/gin/arguments.cc
@@ -48,7 +48,7 @@
   return info_->Holder()->CreationContext();
 }
 
-std::string V8TypeAsString(v8::Local<v8::Value> value) {
+std::string V8TypeAsString(v8::Isolate* isolate, v8::Local<v8::Value> value) {
   if (value.IsEmpty())
     return "<empty handle>";
   if (value->IsUndefined())
@@ -56,7 +56,7 @@
   if (value->IsNull())
     return "null";
   std::string result;
-  if (!ConvertFromV8(NULL, value, &result))
+  if (!ConvertFromV8(isolate, value, &result))
     return std::string();
   return result;
 }
@@ -67,7 +67,7 @@
 
   return ThrowTypeError(base::StringPrintf(
       "Error processing argument at index %d, conversion failure from %s",
-      next_ - 1, V8TypeAsString((*info_)[next_ - 1]).c_str()));
+      next_ - 1, V8TypeAsString(isolate_, (*info_)[next_ - 1]).c_str()));
 }
 
 void Arguments::ThrowTypeError(const std::string& message) const {
diff --git a/gin/converter.h b/gin/converter.h
index 72c1d70..a501714 100644
--- a/gin/converter.h
+++ b/gin/converter.h
@@ -255,6 +255,7 @@
 template<typename T>
 bool ConvertFromV8(v8::Isolate* isolate, v8::Local<v8::Value> input,
                    T* result) {
+  DCHECK(isolate);
   return Converter<T>::FromV8(isolate, input, result);
 }
 
diff --git a/google_apis/gcm/base/socket_stream.cc b/google_apis/gcm/base/socket_stream.cc
index 776813b..6674d9c 100644
--- a/google_apis/gcm/base/socket_stream.cc
+++ b/google_apis/gcm/base/socket_stream.cc
@@ -216,9 +216,7 @@
     net::StreamSocket* socket,
     const net::NetworkTrafficAnnotationTag& traffic_annotation)
     : socket_(socket),
-      io_buffer_(new net::IOBuffer(kDefaultBufferSize)),
-      write_buffer_(
-          new net::DrainableIOBuffer(io_buffer_.get(), kDefaultBufferSize)),
+      io_buffer_(new net::IOBufferWithSize(kDefaultBufferSize)),
       next_pos_(0),
       last_error_(net::OK),
       traffic_annotation_(traffic_annotation),
@@ -232,12 +230,12 @@
 bool SocketOutputStream::Next(void** data, int* size) {
   DCHECK_NE(GetState(), CLOSED);
   DCHECK_NE(GetState(), FLUSHING);
-  if (next_pos_ == write_buffer_->size())
+  if (next_pos_ == io_buffer_->size())
     return false;
 
-  *data = write_buffer_->data() + next_pos_;
-  *size = write_buffer_->size() - next_pos_;
-  next_pos_ = write_buffer_->size();
+  *data = io_buffer_->data() + next_pos_;
+  *size = io_buffer_->size() - next_pos_;
+  next_pos_ = io_buffer_->size();
   return true;
 }
 
@@ -259,15 +257,21 @@
 net::Error SocketOutputStream::Flush(const base::Closure& callback) {
   DCHECK_EQ(GetState(), READY);
 
+  if (!write_buffer_) {
+    write_buffer_ = base::MakeRefCounted<net::DrainableIOBuffer>(
+        io_buffer_.get(), next_pos_);
+  }
+
   if (!socket_->IsConnected()) {
     LOG(ERROR) << "Socket was disconnected, closing output stream";
     last_error_ = net::ERR_CONNECTION_CLOSED;
     return net::OK;
   }
 
-  DVLOG(1) << "Flushing " << next_pos_ << " bytes into socket.";
+  DVLOG(1) << "Flushing " << write_buffer_->BytesRemaining()
+           << " bytes into socket.";
   int result =
-      socket_->Write(write_buffer_.get(), next_pos_,
+      socket_->Write(write_buffer_.get(), write_buffer_->BytesRemaining(),
                      base::Bind(&SocketOutputStream::FlushCompletionCallback,
                                 weak_ptr_factory_.GetWeakPtr(), callback),
                      traffic_annotation_);
@@ -322,17 +326,17 @@
 
   DCHECK_GT(result, net::OK);
   last_error_ = net::OK;
+  write_buffer_->DidConsume(result);
 
-  if (write_buffer_->BytesConsumed() + result < next_pos_) {
+  if (write_buffer_->BytesRemaining() > 0) {
     DVLOG(1) << "Partial flush complete. Retrying.";
-     // Only a partial write was completed. Flush again to finish the write.
-    write_buffer_->DidConsume(result);
+    // Only a partial write was completed. Flush again to finish the write.
     Flush(callback);
     return;
   }
 
   DVLOG(1) << "Socket flush complete.";
-  write_buffer_->SetOffset(0);
+  write_buffer_ = nullptr;
   next_pos_ = 0;
   if (!callback.is_null())
     callback.Run();
diff --git a/google_apis/gcm/base/socket_stream.h b/google_apis/gcm/base/socket_stream.h
index 2b85088..c68acd00 100644
--- a/google_apis/gcm/base/socket_stream.h
+++ b/google_apis/gcm/base/socket_stream.h
@@ -23,6 +23,7 @@
 namespace net {
 class DrainableIOBuffer;
 class IOBuffer;
+class IOBufferWithSize;
 class StreamSocket;
 }  // namespace net
 
@@ -186,10 +187,10 @@
 
   // Internal net components.
   net::StreamSocket* const socket_;
-  const scoped_refptr<net::IOBuffer> io_buffer_;
+  const scoped_refptr<net::IOBufferWithSize> io_buffer_;
   // IOBuffer implementation that wraps the data within |io_buffer_| that hasn't
   // been written to the socket yet.
-  const scoped_refptr<net::DrainableIOBuffer> write_buffer_;
+  scoped_refptr<net::DrainableIOBuffer> write_buffer_;
 
   // Starting position of the data within |io_buffer_| to consume on subsequent
   // Next(..) call.  0 <= write_buffer_.BytesConsumed() <= next_pos_
diff --git a/google_apis/gcm/base/socket_stream_unittest.cc b/google_apis/gcm/base/socket_stream_unittest.cc
index 5f263e9f..7da26d1 100644
--- a/google_apis/gcm/base/socket_stream_unittest.cc
+++ b/google_apis/gcm/base/socket_stream_unittest.cc
@@ -7,6 +7,9 @@
 #include <stdint.h>
 
 #include <memory>
+#include <string>
+#include <utility>
+#include <vector>
 
 #include "base/bind.h"
 #include "base/macros.h"
@@ -32,6 +35,77 @@
 const char kWriteData[] = "write_data";
 const int kWriteDataSize = arraysize(kWriteData) - 1;
 
+// A net::StreamSocket that returns a partial write only for the first time.
+class FirstWritePartialSocket : public net::StreamSocket {
+ public:
+  FirstWritePartialSocket() {}
+  ~FirstWritePartialSocket() override {}
+
+  // Returns the data that is actually written to the socket.
+  const std::string& actual_data_written() { return actual_data_written_; }
+
+  // net::Socket implementation.
+  int Write(
+      net::IOBuffer* buf,
+      int buf_len,
+      net::CompletionOnceCallback callback,
+      const net::NetworkTrafficAnnotationTag& traffic_annotation) override {
+    // Make the first write as a partial write.
+    if (!write_invoked_) {
+      write_invoked_ = true;
+      actual_data_written_.append(buf->data(), buf_len / 2);
+      return buf_len / 2;
+    }
+    // For subsequent writes, write everything that caller has passed to us.
+    actual_data_written_.append(buf->data(), buf_len);
+    return buf_len;
+  }
+  int Read(net::IOBuffer* buf,
+           int buf_len,
+           net::CompletionOnceCallback callback) override {
+    return net::ERR_IO_PENDING;
+  }
+  int ReadIfReady(net::IOBuffer* buf,
+                  int buf_len,
+                  net::CompletionOnceCallback callback) override {
+    return net::ERR_IO_PENDING;
+  }
+  int CancelReadIfReady() override { return net::OK; }
+  int SetReceiveBufferSize(int32_t size) override { return net::OK; }
+  int SetSendBufferSize(int32_t size) override { return net::OK; }
+
+  // net::StreamSocket implementation.
+  int Connect(net::CompletionOnceCallback callback) override { return net::OK; }
+  void Disconnect() override {}
+  bool IsConnected() const override { return true; }
+  bool IsConnectedAndIdle() const override { return true; };
+  int GetPeerAddress(net::IPEndPoint* address) const override {
+    return net::OK;
+  }
+  int GetLocalAddress(net::IPEndPoint* address) const override {
+    return net::OK;
+  }
+  const net::NetLogWithSource& NetLog() const override { return net_log_; }
+  bool WasEverUsed() const override { return true; }
+  bool WasAlpnNegotiated() const override { return false; }
+  net::NextProto GetNegotiatedProtocol() const override {
+    return net::kProtoUnknown;
+  }
+  bool GetSSLInfo(net::SSLInfo* ssl_info) override { return false; }
+  void GetConnectionAttempts(net::ConnectionAttempts* out) const override {}
+  void ClearConnectionAttempts() override {}
+  void AddConnectionAttempts(const net::ConnectionAttempts& attempts) override {
+  }
+  int64_t GetTotalReceivedBytes() const override { return 0; }
+  void ApplySocketTag(const net::SocketTag& tag) override {}
+
+ private:
+  net::NetLogWithSource net_log_;
+  std::string actual_data_written_;
+  // Whether Write() has been invoked before.
+  bool write_invoked_ = false;
+};
+
 class GCMSocketStreamTest : public testing::Test {
  public:
   GCMSocketStreamTest();
@@ -59,6 +133,10 @@
   SocketOutputStream* output_stream() { return socket_output_stream_.get(); }
   net::StreamSocket* socket() { return socket_.get(); }
 
+  void set_socket_output_stream(std::unique_ptr<SocketOutputStream> stream) {
+    socket_output_stream_ = std::move(stream);
+  }
+
  private:
   void OpenConnection();
   void ResetInputStream();
@@ -329,6 +407,17 @@
                                                   kWriteDataSize)));
 }
 
+// Regression test for crbug.com/866635.
+TEST_F(GCMSocketStreamTest, WritePartialWithLengthChecking) {
+  auto socket = std::make_unique<FirstWritePartialSocket>();
+  auto socket_output_stream = std::make_unique<SocketOutputStream>(
+      socket.get(), TRAFFIC_ANNOTATION_FOR_TESTS);
+  set_socket_output_stream(std::move(socket_output_stream));
+  ASSERT_EQ(kWriteDataSize,
+            DoOutputStreamWrite(base::StringPiece(kWriteData, kWriteDataSize)));
+  EXPECT_EQ(kWriteData, socket->actual_data_written());
+}
+
 // Write a message completely asynchronously (returns IO_PENDING before
 // finishing the write in two go's).
 TEST_F(GCMSocketStreamTest, WriteNone) {
diff --git a/gpu/command_buffer/client/gles2_implementation.cc b/gpu/command_buffer/client/gles2_implementation.cc
index 33cd9dc..cd23a01 100644
--- a/gpu/command_buffer/client/gles2_implementation.cc
+++ b/gpu/command_buffer/client/gles2_implementation.cc
@@ -41,6 +41,7 @@
 #include "gpu/command_buffer/client/transfer_buffer.h"
 #include "gpu/command_buffer/client/vertex_array_object_manager.h"
 #include "gpu/command_buffer/common/context_creation_attribs.h"
+#include "gpu/command_buffer/common/gles2_cmd_utils.h"
 #include "gpu/command_buffer/common/id_allocator.h"
 #include "gpu/command_buffer/common/swap_buffers_complete_params.h"
 #include "gpu/command_buffer/common/sync_token.h"
@@ -124,11 +125,6 @@
   }
 }
 
-// A 32-bit and 64-bit compatible way of converting a pointer to a GLuint.
-GLuint ToGLuint(const void* ptr) {
-  return static_cast<GLuint>(reinterpret_cast<size_t>(ptr));
-}
-
 static base::AtomicSequenceNumber g_flush_id;
 
 uint32_t GenerateNextFlushId() {
diff --git a/gpu/command_buffer/client/vertex_array_object_manager.cc b/gpu/command_buffer/client/vertex_array_object_manager.cc
index 453de8c..0690c5e 100644
--- a/gpu/command_buffer/client/vertex_array_object_manager.cc
+++ b/gpu/command_buffer/client/vertex_array_object_manager.cc
@@ -11,6 +11,7 @@
 #include "base/macros.h"
 #include "gpu/command_buffer/client/gles2_cmd_helper.h"
 #include "gpu/command_buffer/client/gles2_implementation.h"
+#include "gpu/command_buffer/common/gles2_cmd_utils.h"
 
 namespace gpu {
 namespace gles2 {
@@ -19,11 +20,6 @@
   return (size + 3) & ~3;
 }
 
-// A 32-bit and 64-bit compatible way of converting a pointer to a GLuint.
-static GLuint ToGLuint(const void* ptr) {
-  return static_cast<GLuint>(reinterpret_cast<size_t>(ptr));
-}
-
 // This class tracks VertexAttribPointers and helps emulate client side buffers.
 //
 // The way client side buffers work is we shadow all the Vertex Attribs so we
diff --git a/gpu/command_buffer/common/gles2_cmd_utils.h b/gpu/command_buffer/common/gles2_cmd_utils.h
index e959a0e5..61ffa634 100644
--- a/gpu/command_buffer/common/gles2_cmd_utils.h
+++ b/gpu/command_buffer/common/gles2_cmd_utils.h
@@ -53,6 +53,12 @@
   return checked.IsValid();
 }
 
+// A 32-bit and 64-bit compatible way of converting a pointer to a
+// 32-bit usigned integer, suitable to be stored in a GLuint.
+inline uint32_t ToGLuint(const void* ptr) {
+  return static_cast<uint32_t>(reinterpret_cast<size_t>(ptr));
+}
+
 // Returns the address of the first byte after a struct.
 template <typename T>
 const volatile void* AddressAfterStruct(const volatile T& pod) {
diff --git a/gpu/command_buffer/service/texture_manager.cc b/gpu/command_buffer/service/texture_manager.cc
index 4c4b7c60..bf9a384 100644
--- a/gpu/command_buffer/service/texture_manager.cc
+++ b/gpu/command_buffer/service/texture_manager.cc
@@ -387,11 +387,6 @@
   return feature_info->IsWebGL2OrES3Context();
 }
 
-// A 32-bit and 64-bit compatible way of converting a pointer to a GLuint.
-GLuint ToGLuint(const void* ptr) {
-  return static_cast<GLuint>(reinterpret_cast<size_t>(ptr));
-}
-
 base::LazyInstance<const FormatTypeValidator>::Leaky g_format_type_validator =
     LAZY_INSTANCE_INITIALIZER;
 
diff --git a/gpu/config/gpu_driver_bug_list.json b/gpu/config/gpu_driver_bug_list.json
index d3b824d7..d9d588a 100644
--- a/gpu/config/gpu_driver_bug_list.json
+++ b/gpu/config/gpu_driver_bug_list.json
@@ -2866,8 +2866,8 @@
       "os": {
         "type": "linux"
       },
-      "vendor_id": "0x15ad",
-      "device_id": ["0x0405"],
+      "gl_vendor": "VMware.*",
+      "gl_renderer": "Gallium.*",
       "features": [
         "disable_timestamp_queries"
       ],
diff --git a/gpu/vulkan/vulkan_command_pool.h b/gpu/vulkan/vulkan_command_pool.h
index b54d5917..bd1b84c 100644
--- a/gpu/vulkan/vulkan_command_pool.h
+++ b/gpu/vulkan/vulkan_command_pool.h
@@ -10,13 +10,14 @@
 #include <memory>
 
 #include "base/macros.h"
+#include "gpu/vulkan/vulkan_export.h"
 
 namespace gpu {
 
 class VulkanCommandBuffer;
 class VulkanDeviceQueue;
 
-class VulkanCommandPool {
+class VULKAN_EXPORT VulkanCommandPool {
  public:
   explicit VulkanCommandPool(VulkanDeviceQueue* device_queue);
   ~VulkanCommandPool();
diff --git a/ios/chrome/app/main_controller.mm b/ios/chrome/app/main_controller.mm
index 191b4e1..da96b98 100644
--- a/ios/chrome/app/main_controller.mm
+++ b/ios/chrome/app/main_controller.mm
@@ -417,7 +417,7 @@
 // |currentBVC| can be made active.
 - (void)activateBVCAndMakeCurrentBVCPrimary;
 // Sets |currentBVC| as the root view controller for the window.
-- (void)displayCurrentBVC;
+- (void)displayCurrentBVCAndFocusOmnibox:(BOOL)focusOmnibox;
 // Shows the tab switcher UI.
 - (void)showTabSwitcher;
 // Starts a voice search on the current BVC.
@@ -458,8 +458,10 @@
 - (bool)mustShowRestoreInfobar;
 // Begins the process of dismissing the tab switcher with the given current
 // model, switching which BVC is suspended if necessary, but not updating the
-// UI.
-- (void)beginDismissingTabSwitcherWithCurrentModel:(TabModel*)tabModel;
+// UI.  The omnibox will be focused after the tab switcher dismissal is
+// completed if |focusOmnibox| is YES.
+- (void)beginDismissingTabSwitcherWithCurrentModel:(TabModel*)tabModel
+                                      focusOmnibox:(BOOL)focusOmnibox;
 // Completes the process of dismissing the tab switcher, removing it from the
 // screen and showing the appropriate BVC.
 - (void)finishDismissingTabSwitcher;
@@ -1800,7 +1802,7 @@
   [_browserViewWrangler setCurrentBVC:bvc storageSwitcher:self];
 
   if (!_dismissingTabSwitcher)
-    [self displayCurrentBVC];
+    [self displayCurrentBVCAndFocusOmnibox:NO];
 
   // Tell the BVC that was made current that it can use the web.
   [self activateBVCAndMakeCurrentBVCPrimary];
@@ -1894,9 +1896,16 @@
   [self switchGlobalStateToMode:mode];
 }
 
-- (void)displayCurrentBVC {
+- (void)displayCurrentBVCAndFocusOmnibox:(BOOL)focusOmnibox {
+  ProceduralBlock completion = nil;
+  if (focusOmnibox) {
+    __weak BrowserViewController* weakCurrentBVC = self.currentBVC;
+    completion = ^{
+      [weakCurrentBVC.dispatcher focusOmnibox];
+    };
+  }
   [self.viewControllerSwapper showTabViewController:self.currentBVC
-                                         completion:nil];
+                                         completion:completion];
   [self.currentBVC.dispatcher
       setIncognitoContentVisible:(self.currentBVC == self.otrBVC)];
 }
@@ -2016,7 +2025,7 @@
     [self dismissModalDialogsWithCompletion:nil dismissOmnibox:YES];
     [_tabSwitcher tabSwitcherDismissWithModel:tabModel animated:NO];
   } else {
-    [self beginDismissingTabSwitcherWithCurrentModel:tabModel];
+    [self beginDismissingTabSwitcherWithCurrentModel:tabModel focusOmnibox:NO];
     [self finishDismissingTabSwitcher];
   }
 }
@@ -2024,8 +2033,10 @@
 #pragma mark - TabSwitcherDelegate
 
 - (void)tabSwitcher:(id<TabSwitcher>)tabSwitcher
-    shouldFinishWithActiveModel:(TabModel*)tabModel {
-  [self beginDismissingTabSwitcherWithCurrentModel:tabModel];
+    shouldFinishWithActiveModel:(TabModel*)tabModel
+                   focusOmnibox:(BOOL)focusOmnibox {
+  [self beginDismissingTabSwitcherWithCurrentModel:tabModel
+                                      focusOmnibox:focusOmnibox];
 }
 
 - (void)tabSwitcherDismissTransitionDidEnd:(id<TabSwitcher>)tabSwitcher {
@@ -2041,7 +2052,8 @@
 
 #pragma mark - TabSwitcherDelegate helper methods
 
-- (void)beginDismissingTabSwitcherWithCurrentModel:(TabModel*)tabModel {
+- (void)beginDismissingTabSwitcherWithCurrentModel:(TabModel*)tabModel
+                                      focusOmnibox:(BOOL)focusOmnibox {
   DCHECK(tabModel == self.mainTabModel || tabModel == self.otrTabModel);
 
   _dismissingTabSwitcher = YES;
@@ -2059,7 +2071,7 @@
   // _dismissingTabSwitcher is YES.  When the presentation experiment is
   // enabled, force the BVC transition to start.
   if (TabSwitcherPresentsBVCEnabled()) {
-    [self displayCurrentBVC];
+    [self displayCurrentBVCAndFocusOmnibox:focusOmnibox];
   }
 }
 
@@ -2092,7 +2104,7 @@
   _modeToDisplayOnTabSwitcherDismissal = TabSwitcherDismissalMode::NONE;
 
   // Displaying the current BVC dismisses the tab switcher.
-  [self displayCurrentBVC];
+  [self displayCurrentBVCAndFocusOmnibox:NO];
 
   ProceduralBlock action = [self completionBlockForTriggeringAction:
                                      self.NTPActionAfterTabSwitcherDismissal];
diff --git a/ios/chrome/app/resources/ios_resources.grd b/ios/chrome/app/resources/ios_resources.grd
index 9c7dbfa..2cb2ac88 100644
--- a/ios/chrome/app/resources/ios_resources.grd
+++ b/ios/chrome/app/resources/ios_resources.grd
@@ -11,6 +11,8 @@
       <include name="IDR_IOS_OMAHA_HTML" file="omaha/omaha.html" flattenhtml="true" allowexternalscript="true" type="BINDATA" compress="gzip" />
       <include name="IDR_IOS_OMAHA_JS" file="omaha/omaha.js" type="BINDATA" compress="gzip" />
       <include name="IDR_CHROME_BROWSER_MANIFEST_OVERLAY" file="${root_gen_dir}/ios/chrome/app/chrome_browser_manifest_overlay.json" use_base_dir="false" type="BINDATA" />
+      <include name="IDR_IOS_UKM_INTERNALS_HTML" file="../../../../components/ukm/debug/ukm_internals.html" flattenhtml="true" allowexternalscript="true" compress="gzip" type="BINDATA" />
+      <include name="IDR_IOS_UKM_INTERNALS_JS" file="../../../../components/ukm/debug/ukm_internals.js" flattenhtml="true" compress="gzip" type="BINDATA" />
     </includes>
   </release>
 </grit>
diff --git a/ios/chrome/browser/autofill/BUILD.gn b/ios/chrome/browser/autofill/BUILD.gn
index 89f891f9..75658a2 100644
--- a/ios/chrome/browser/autofill/BUILD.gn
+++ b/ios/chrome/browser/autofill/BUILD.gn
@@ -14,7 +14,6 @@
     "form_input_accessory_view_controller.h",
     "form_input_accessory_view_controller.mm",
     "form_input_accessory_view_delegate.h",
-    "form_input_accessory_view_provider.h",
     "form_suggestion_controller.h",
     "form_suggestion_controller.mm",
     "form_suggestion_label.h",
@@ -30,6 +29,7 @@
     "validation_rules_storage_factory.h",
   ]
   deps = [
+    ":autofill_shared",
     "resources:autofill_close",
     "resources:autofill_close_pressed",
     "resources:autofill_keyboard_background",
@@ -72,6 +72,14 @@
   libs = [ "QuartzCore.framework" ]
 }
 
+source_set("autofill_shared") {
+  configs += [ "//build/config/compiler:enable_arc" ]
+  sources = [
+    "form_input_accessory_view_provider.h",
+  ]
+  libs = [ "UIKit.framework" ]
+}
+
 source_set("autofill_internal") {
   configs += [ "//build/config/compiler:enable_arc" ]
   sources = [
diff --git a/ios/chrome/browser/metrics/ukm_egtest.mm b/ios/chrome/browser/metrics/ukm_egtest.mm
index 50674449..09cd952 100644
--- a/ios/chrome/browser/metrics/ukm_egtest.mm
+++ b/ios/chrome/browser/metrics/ukm_egtest.mm
@@ -38,6 +38,7 @@
 #endif
 
 using chrome_test_util::AccountsSyncButton;
+using chrome_test_util::ButtonWithAccessibilityLabel;
 using chrome_test_util::ButtonWithAccessibilityLabelId;
 using chrome_test_util::ClearBrowsingDataCollectionView;
 using chrome_test_util::GetIncognitoTabCount;
@@ -194,13 +195,17 @@
 // Signs out of sync.
 void SignOut() {
   [ChromeEarlGreyUI openSettingsMenu];
-  [[EarlGrey selectElementWithMatcher:SettingsAccountButton()]
+  [ChromeEarlGreyUI tapSettingsMenuButton:SettingsAccountButton()];
+
+  // Remove |identity| from the device.
+  ChromeIdentity* identity = [SigninEarlGreyUtils fakeIdentity1];
+  [[EarlGrey
+      selectElementWithMatcher:ButtonWithAccessibilityLabel(identity.userEmail)]
       performAction:grey_tap()];
-  [ChromeEarlGreyUI tapAccountsMenuButton:SignOutAccountsButton()];
-  [[EarlGrey selectElementWithMatcher:
-                 ButtonWithAccessibilityLabelId(
-                     IDS_IOS_DISCONNECT_DIALOG_CONTINUE_BUTTON_MOBILE)]
+  [[EarlGrey
+      selectElementWithMatcher:ButtonWithAccessibilityLabel(@"Remove account")]
       performAction:grey_tap()];
+
   [[EarlGrey selectElementWithMatcher:SettingsDoneButton()]
       performAction:grey_tap()];
 
@@ -216,11 +221,6 @@
 
 @implementation UKMTestCase
 
-// Per crbug.com/853992, Entire test suite is failing regularly.
-+ (NSArray*)testInvocations {
-  return @[];
-}
-
 + (void)setUp {
   [super setUp];
   if (!base::FeatureList::IsEnabled(ukm::kUkmFeature)) {
@@ -466,7 +466,8 @@
 
 // testMetricsReporting not needed, since iOS doesn't use sampling.
 
-- (void)testHistoryDelete {
+// TODO(crbug.com/866598): Re-enable this test.
+- (void)DISABLED_testHistoryDelete {
   uint64_t original_client_id = metrics::UkmEGTestHelper::client_id();
 
   const ukm::SourceId kDummySourceId = 0x54321;
diff --git a/ios/chrome/browser/ui/authentication/BUILD.gn b/ios/chrome/browser/ui/authentication/BUILD.gn
index 451a166..0bb9fa0 100644
--- a/ios/chrome/browser/ui/authentication/BUILD.gn
+++ b/ios/chrome/browser/ui/authentication/BUILD.gn
@@ -52,6 +52,7 @@
     "//ios/chrome/browser/tabs",
     "//ios/chrome/browser/ui",
     "//ios/chrome/browser/ui/alert_coordinator",
+    "//ios/chrome/browser/ui/authentication/consent_bump",
     "//ios/chrome/browser/ui/collection_view/cells",
     "//ios/chrome/browser/ui/colors",
     "//ios/chrome/browser/ui/commands",
@@ -179,6 +180,7 @@
     "//components/signin/core/browser",
     "//components/signin/core/browser:account_info",
     "//components/unified_consent",
+    "//ios/chrome/app/strings:ios_strings_grit",
     "//ios/chrome/browser/browser_state",
     "//ios/chrome/browser/signin",
     "//ios/chrome/browser/ui/authentication/unified_consent:unified_consent_ui",
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/BUILD.gn b/ios/chrome/browser/ui/authentication/consent_bump/BUILD.gn
new file mode 100644
index 0000000..a0ed320
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/BUILD.gn
@@ -0,0 +1,26 @@
+# Copyright 2018 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+import("//build/config/chrome_build.gni")
+
+source_set("consent_bump") {
+  configs += [ "//build/config/compiler:enable_arc" ]
+  sources = [
+    "consent_bump_coordinator.h",
+    "consent_bump_coordinator.mm",
+    "consent_bump_personalization_coordinator.h",
+    "consent_bump_personalization_coordinator.mm",
+    "consent_bump_personalization_view_controller.h",
+    "consent_bump_personalization_view_controller.mm",
+    "consent_bump_view_controller.h",
+    "consent_bump_view_controller.mm",
+    "consent_bump_view_controller_delegate.h",
+  ]
+  deps = [
+    "//base",
+    "//ios/chrome/browser/ui/authentication/unified_consent",
+    "//ios/chrome/browser/ui/coordinators:chrome_coordinators",
+    "//ios/chrome/common/ui_util",
+  ]
+}
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.h b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.h
new file mode 100644
index 0000000..181289c0
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.h
@@ -0,0 +1,18 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_COORDINATOR_H_
+#define IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_COORDINATOR_H_
+
+#import "ios/chrome/browser/ui/coordinators/chrome_coordinator.h"
+
+// Coordinator handling the consent bump.
+@interface ConsentBumpCoordinator : ChromeCoordinator
+
+// ViewController associated with this coordinator.
+@property(nonatomic, strong, readonly) UIViewController* viewController;
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_COORDINATOR_H_
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.mm b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.mm
new file mode 100644
index 0000000..d704b09
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.mm
@@ -0,0 +1,115 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_coordinator.h"
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.h"
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.h"
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller_delegate.h"
+#import "ios/chrome/browser/ui/authentication/unified_consent/unified_consent_coordinator.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+namespace {
+// Type of child coordinator presented by this coordinator.
+typedef NS_ENUM(NSInteger, PresentedCoordinator) {
+  PresentedCoordinatorUnifiedConsent,
+  PresentedCoordinatorPersonalization,
+};
+}  // namespace
+
+@interface ConsentBumpCoordinator ()<ConsentBumpViewControllerDelegate>
+
+// Which child coordinator is currently presented.
+@property(nonatomic, assign) PresentedCoordinator presentedCoordinator;
+
+// The ViewController of this coordinator, redefined as a
+// ConsentBumpViewController.
+@property(nonatomic, strong)
+    ConsentBumpViewController* consentBumpViewController;
+// The child coordinator presenting the unified consent.
+@property(nonatomic, strong)
+    UnifiedConsentCoordinator* unifiedConsentCoordinator;
+// The child coordinator presenting the presonalization content.
+@property(nonatomic, strong)
+    ConsentBumpPersonalizationCoordinator* personalizationCoordinator;
+
+@end
+
+@implementation ConsentBumpCoordinator
+
+@synthesize presentedCoordinator = _presentedCoordinator;
+@synthesize consentBumpViewController = _consentBumpViewController;
+@synthesize unifiedConsentCoordinator = _unifiedConsentCoordinator;
+@synthesize personalizationCoordinator = _personalizationCoordinator;
+
+#pragma mark - Properties
+
+- (UIViewController*)viewController {
+  return self.consentBumpViewController;
+}
+
+#pragma mark - ChromeCoordinator
+
+- (void)start {
+  self.consentBumpViewController = [[ConsentBumpViewController alloc] init];
+  self.consentBumpViewController.delegate = self;
+
+  self.unifiedConsentCoordinator = [[UnifiedConsentCoordinator alloc] init];
+  [self.unifiedConsentCoordinator start];
+  self.presentedCoordinator = PresentedCoordinatorUnifiedConsent;
+
+  self.consentBumpViewController.contentViewController =
+      self.unifiedConsentCoordinator.viewController;
+}
+
+- (void)stop {
+  self.consentBumpViewController = nil;
+  self.unifiedConsentCoordinator = nil;
+  self.personalizationCoordinator = nil;
+}
+
+#pragma mark - ConsentBumpViewControllerDelegate
+
+- (void)consentBumpViewControllerDidTapPrimaryButton:
+    (ConsentBumpViewController*)consentBumpViewController {
+  switch (self.presentedCoordinator) {
+    case PresentedCoordinatorUnifiedConsent:
+      // TODO(crbug.com/866506): Consent bump accepted.
+      break;
+    case PresentedCoordinatorPersonalization:
+      // TODO(crbug.com/866506): Clarify what should be the behavior at this
+      // point.
+      break;
+  }
+}
+
+- (void)consentBumpViewControllerDidTapSecondaryButton:
+    (ConsentBumpViewController*)consentBumpViewController {
+  switch (self.presentedCoordinator) {
+    case PresentedCoordinatorUnifiedConsent:
+      // Present the personlization.
+      if (!self.personalizationCoordinator) {
+        self.personalizationCoordinator =
+            [[ConsentBumpPersonalizationCoordinator alloc]
+                initWithBaseViewController:nil];
+        [self.personalizationCoordinator start];
+      }
+      self.consentBumpViewController.contentViewController =
+          self.personalizationCoordinator.viewController;
+      self.presentedCoordinator = PresentedCoordinatorPersonalization;
+      break;
+
+    case PresentedCoordinatorPersonalization:
+      // Go back to the unified consent.
+      self.consentBumpViewController.contentViewController =
+          self.unifiedConsentCoordinator.viewController;
+      self.presentedCoordinator = PresentedCoordinatorUnifiedConsent;
+      break;
+  }
+}
+
+@end
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.h b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.h
new file mode 100644
index 0000000..4611b12d
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.h
@@ -0,0 +1,18 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_COORDINATOR_H_
+#define IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_COORDINATOR_H_
+
+#import "ios/chrome/browser/ui/coordinators/chrome_coordinator.h"
+
+// Coordinator for handling the ConsentBump Personalization screen.
+@interface ConsentBumpPersonalizationCoordinator : ChromeCoordinator
+
+// The view Controller for this coordinator.
+@property(nonatomic, strong, readonly) UIViewController* viewController;
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_COORDINATOR_H_
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.mm b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.mm
new file mode 100644
index 0000000..9dff249
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.mm
@@ -0,0 +1,42 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_coordinator.h"
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+@interface ConsentBumpPersonalizationCoordinator ()
+
+// The view controller, redefined as ConsentBumpPersonalizationViewController.
+@property(nonatomic, strong)
+    ConsentBumpPersonalizationViewController* personalizationViewController;
+
+@end
+
+@implementation ConsentBumpPersonalizationCoordinator
+
+@synthesize personalizationViewController = _personalizationViewController;
+
+#pragma mark - Properties
+
+- (UIViewController*)viewController {
+  return self.personalizationViewController;
+}
+
+#pragma mark - ChromeCoordinator
+
+- (void)start {
+  self.personalizationViewController =
+      [[ConsentBumpPersonalizationViewController alloc] init];
+}
+
+- (void)stop {
+  self.personalizationViewController = nil;
+}
+
+@end
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.h b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.h
new file mode 100644
index 0000000..064ba0f
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.h
@@ -0,0 +1,15 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_VIEW_CONTROLLER_H_
+#define IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_VIEW_CONTROLLER_H_
+
+#import <UIKit/UIKit.h>
+
+// View controller displaying the Personalization screen.
+@interface ConsentBumpPersonalizationViewController : UIViewController
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_PERSONALIZATION_VIEW_CONTROLLER_H_
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.mm b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.mm
new file mode 100644
index 0000000..b1d3f554
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.mm
@@ -0,0 +1,17 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_personalization_view_controller.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+@implementation ConsentBumpPersonalizationViewController
+
+- (void)viewDidLoad {
+  self.view.backgroundColor = [UIColor blueColor];
+}
+
+@end
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.h b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.h
new file mode 100644
index 0000000..3640b4a
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.h
@@ -0,0 +1,23 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_H_
+#define IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_H_
+
+#import <UIKit/UIKit.h>
+
+@protocol ConsentBumpViewControllerDelegate;
+
+// View Controller handling the ConsentBump screen.
+@interface ConsentBumpViewController : UIViewController
+
+// Delegate for the view controller.
+@property(nonatomic, weak) id<ConsentBumpViewControllerDelegate> delegate;
+
+// View controller displayed as child view controller for this ViewController.
+@property(nonatomic, strong) UIViewController* contentViewController;
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_H_
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.mm b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.mm
new file mode 100644
index 0000000..ef7f439
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.mm
@@ -0,0 +1,182 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller.h"
+
+#import "ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller_delegate.h"
+#import "ios/chrome/common/ui_util/constraints_ui_util.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+namespace {
+const CGFloat kMargin = 16;
+const CGFloat kGradientHeight = 40;
+}  // namespace
+
+@interface ConsentBumpViewController ()
+
+@property(nonatomic, strong) UIView* buttonContainer;
+@property(nonatomic, strong) UIButton* primaryButton;
+@property(nonatomic, strong) UIButton* secondaryButton;
+@property(nonatomic, strong) UIView* gradientView;
+@property(nonatomic, strong) CAGradientLayer* gradientLayer;
+
+@end
+
+@implementation ConsentBumpViewController
+
+@synthesize delegate = _delegate;
+@synthesize contentViewController = _contentViewController;
+@synthesize buttonContainer = _buttonContainer;
+@synthesize primaryButton = _primaryButton;
+@synthesize secondaryButton = _secondaryButton;
+@synthesize gradientView = _gradientView;
+@synthesize gradientLayer = _gradientLayer;
+
+#pragma mark - Public
+
+- (void)setContentViewController:(UIViewController*)contentViewController {
+  if (_contentViewController == contentViewController)
+    return;
+
+  // Remove previous VC.
+  [_contentViewController willMoveToParentViewController:nil];
+  [_contentViewController.view removeFromSuperview];
+  [_contentViewController removeFromParentViewController];
+
+  _contentViewController = contentViewController;
+
+  if (!contentViewController)
+    return;
+
+  [self addChildViewController:contentViewController];
+
+  contentViewController.view.translatesAutoresizingMaskIntoConstraints = NO;
+  [self.view insertSubview:contentViewController.view
+              belowSubview:self.gradientView];
+  AddSameConstraintsToSides(
+      self.view, contentViewController.view,
+      LayoutSides::kTop | LayoutSides::kLeading | LayoutSides::kTrailing);
+  [contentViewController.view.bottomAnchor
+      constraintEqualToAnchor:self.buttonContainer.topAnchor]
+      .active = YES;
+
+  [contentViewController didMoveToParentViewController:self];
+}
+
+#pragma mark - Property
+
+- (UIButton*)primaryButton {
+  if (!_primaryButton) {
+    _primaryButton = [UIButton buttonWithType:UIButtonTypeSystem];
+    _primaryButton.translatesAutoresizingMaskIntoConstraints = NO;
+    [_primaryButton setTitle:@"Primary Button" forState:UIControlStateNormal];
+    _primaryButton.backgroundColor = [UIColor blueColor];
+    [_primaryButton setContentHuggingPriority:UILayoutPriorityDefaultHigh
+                                      forAxis:UILayoutConstraintAxisVertical];
+    [_primaryButton addTarget:self
+                       action:@selector(primaryButtonCallback)
+             forControlEvents:UIControlEventTouchUpInside];
+  }
+  return _primaryButton;
+}
+
+- (UIButton*)secondaryButton {
+  if (!_secondaryButton) {
+    _secondaryButton = [UIButton buttonWithType:UIButtonTypeSystem];
+    _secondaryButton.translatesAutoresizingMaskIntoConstraints = NO;
+    [_secondaryButton setTitle:@"Secondary Button"
+                      forState:UIControlStateNormal];
+    [_secondaryButton setContentHuggingPriority:UILayoutPriorityDefaultHigh
+                                        forAxis:UILayoutConstraintAxisVertical];
+    [_secondaryButton addTarget:self
+                         action:@selector(secondaryButtonCallback)
+               forControlEvents:UIControlEventTouchUpInside];
+  }
+  return _secondaryButton;
+}
+
+- (UIView*)buttonContainer {
+  if (!_buttonContainer) {
+    _buttonContainer = [[UIView alloc] init];
+    _buttonContainer.translatesAutoresizingMaskIntoConstraints = NO;
+  }
+  return _buttonContainer;
+}
+
+- (UIView*)gradientView {
+  if (!_gradientView) {
+    _gradientView = [[UIView alloc] initWithFrame:CGRectZero];
+    _gradientView.userInteractionEnabled = NO;
+    _gradientView.translatesAutoresizingMaskIntoConstraints = NO;
+    [_gradientView.layer insertSublayer:self.gradientLayer atIndex:0];
+  }
+  return _gradientView;
+}
+
+- (CAGradientLayer*)gradientLayer {
+  if (!_gradientLayer) {
+    _gradientLayer = [CAGradientLayer layer];
+    _gradientLayer.colors = @[
+      (id)[[UIColor colorWithWhite:1 alpha:0] CGColor],
+      (id)[self.view.backgroundColor CGColor]
+    ];
+  }
+  return _gradientLayer;
+}
+
+#pragma mark - UIViewController
+
+- (void)viewDidLayoutSubviews {
+  [super viewDidLayoutSubviews];
+  self.gradientLayer.frame = self.gradientView.bounds;
+}
+
+- (void)viewDidLoad {
+  self.view.backgroundColor = [UIColor whiteColor];
+
+  // Add subviews.
+  [self.buttonContainer addSubview:self.primaryButton];
+  [self.buttonContainer addSubview:self.secondaryButton];
+  [self.view addSubview:self.buttonContainer];
+  [self.view addSubview:self.gradientView];
+
+  // Constraints.
+  id<LayoutGuideProvider> safeArea = SafeAreaLayoutGuideForView(self.view);
+  AddSameConstraintsToSides(self.view, self.gradientView,
+                            LayoutSides::kLeading | LayoutSides::kTrailing);
+  AddSameConstraintsToSides(
+      safeArea, self.buttonContainer,
+      LayoutSides::kBottom | LayoutSides::kLeading | LayoutSides::kTrailing);
+  AddSameConstraintsToSidesWithInsets(
+      self.secondaryButton, self.buttonContainer,
+      LayoutSides::kLeading | LayoutSides::kTop | LayoutSides::kBottom,
+      ChromeDirectionalEdgeInsetsMake(kMargin, kMargin, kMargin, 0));
+  AddSameConstraintsToSidesWithInsets(
+      self.primaryButton, self.buttonContainer,
+      LayoutSides::kTrailing | LayoutSides::kTop | LayoutSides::kBottom,
+      ChromeDirectionalEdgeInsetsMake(kMargin, 0, kMargin, kMargin));
+  [NSLayoutConstraint activateConstraints:@[
+    [self.gradientView.heightAnchor constraintEqualToConstant:kGradientHeight],
+    [self.gradientView.bottomAnchor
+        constraintEqualToAnchor:self.buttonContainer.topAnchor],
+    [self.primaryButton.leadingAnchor
+        constraintGreaterThanOrEqualToAnchor:self.secondaryButton
+                                                 .trailingAnchor],
+  ]];
+}
+
+#pragma mark - Private
+
+- (void)primaryButtonCallback {
+  [self.delegate consentBumpViewControllerDidTapPrimaryButton:self];
+}
+
+- (void)secondaryButtonCallback {
+  [self.delegate consentBumpViewControllerDidTapSecondaryButton:self];
+}
+
+@end
diff --git a/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller_delegate.h b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller_delegate.h
new file mode 100644
index 0000000..8a35f48
--- /dev/null
+++ b/ios/chrome/browser/ui/authentication/consent_bump/consent_bump_view_controller_delegate.h
@@ -0,0 +1,26 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_DELEGATE_H_
+#define IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_DELEGATE_H_
+
+#import <UIKit/UIKit.h>
+
+@class ConsentBumpViewController;
+
+// Protocol for the delegate of the ConsentBumpViewController.
+@protocol ConsentBumpViewControllerDelegate<NSObject>
+
+// Notifies the delegate that the primary button of the view controller has been
+// pressed.
+- (void)consentBumpViewControllerDidTapPrimaryButton:
+    (ConsentBumpViewController*)consentBumpViewController;
+// Notifies the delegate that the secondary button of the view controller has
+// been pressed.
+- (void)consentBumpViewControllerDidTapSecondaryButton:
+    (ConsentBumpViewController*)consentBumpViewController;
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_UI_AUTHENTICATION_CONSENT_BUMP_CONSENT_BUMP_VIEW_CONTROLLER_DELEGATE_H_
diff --git a/ios/chrome/browser/ui/authentication/signin_earl_grey_ui.mm b/ios/chrome/browser/ui/authentication/signin_earl_grey_ui.mm
index 85ada002..d1c976fe 100644
--- a/ios/chrome/browser/ui/authentication/signin_earl_grey_ui.mm
+++ b/ios/chrome/browser/ui/authentication/signin_earl_grey_ui.mm
@@ -6,12 +6,14 @@
 
 #import <EarlGrey/EarlGrey.h>
 
+#include "base/mac/foundation_util.h"
 #include "components/unified_consent/feature.h"
 #include "ios/chrome/browser/ui/authentication/signin_confirmation_view_controller.h"
 #import "ios/chrome/browser/ui/authentication/signin_earlgrey_utils.h"
 #import "ios/chrome/browser/ui/authentication/unified_consent/identity_chooser/identity_chooser_cell.h"
 #import "ios/chrome/browser/ui/authentication/unified_consent/identity_picker_view.h"
 #import "ios/chrome/browser/ui/authentication/unified_consent/unified_consent_view_controller.h"
+#include "ios/chrome/grit/ios_strings.h"
 #import "ios/chrome/test/earl_grey/chrome_earl_grey_ui.h"
 #import "ios/chrome/test/earl_grey/chrome_matchers.h"
 #import "ios/public/provider/chrome/browser/signin/chrome_identity.h"
@@ -21,6 +23,27 @@
 #error "This file requires ARC support."
 #endif
 
+namespace {
+
+// Returns a matcher to test whether the element is a scroll view with a content
+// smaller than the scroll view bounds.
+id<GREYMatcher> ContentViewSmallerThanScrollView() {
+  MatchesBlock matches = ^BOOL(UIView* view) {
+    UIScrollView* scrollView = base::mac::ObjCCast<UIScrollView>(view);
+    return scrollView &&
+           scrollView.contentSize.height < scrollView.bounds.size.height;
+  };
+  DescribeToBlock describe = ^void(id<GREYDescription> description) {
+    [description appendText:
+                     @"Not a scroll view or the scroll view content is bigger "
+                     @"than the scroll view bounds"];
+  };
+  return [[GREYElementMatcherBlock alloc] initWithMatchesBlock:matches
+                                              descriptionBlock:describe];
+}
+
+}  // namespace
+
 using chrome_test_util::AccountConsistencyConfirmationOkButton;
 using chrome_test_util::AccountConsistencySetupSigninButton;
 using chrome_test_util::ButtonWithAccessibilityLabel;
@@ -66,27 +89,42 @@
 }
 
 + (void)confirmSigninConfirmationDialog {
-  // Confirm sign in. "More" button is shown on short devices (e.g. iPhone 5s,
-  // iPhone SE), so needs to scroll first to dismiss the "More" button before
-  // taping on "OK".
-  // Cannot directly scroll on |kSignInConfirmationCollectionViewId| because it
-  // is a MDC collection view, not a UICollectionView, so itself is not
-  // scrollable.
-  // Wait until the sync confirmation is displayed.
-  id<GREYMatcher> signinUICollectionViewMatcher = nil;
+  // To confirm the dialog, the scroll view content has to be scrolled to the
+  // bottom to transform "MORE" button into the validation button.
+  // EarlGrey fails to scroll to the bottom, using grey_scrollToContentEdge(),
+  // if the scroll view doesn't bounce and by default a scroll view doesn't
+  // bounce when the content fits into the scroll view (the scroll never ends).
+  // To test if the content fits into the scroll view,
+  // ContentViewSmallerThanScrollView() matcher is used on the signin scroll
+  // view.
+  // If the matcher fails, then the scroll view should be scrolled to the
+  // bottom.
+  // Once to the bottom, the consent can be confirmed.
+  id<GREYMatcher> confirmationScrollViewMatcher = nil;
   [[GREYUIThreadExecutor sharedInstance] drainUntilIdle];
   if (base::FeatureList::IsEnabled(unified_consent::kUnifiedConsent)) {
-    signinUICollectionViewMatcher =
+    confirmationScrollViewMatcher =
         grey_accessibilityID(kUnifiedConsentScrollViewIdentifier);
   } else {
-    signinUICollectionViewMatcher = grey_allOf(
+    confirmationScrollViewMatcher = grey_allOf(
         grey_ancestor(
             grey_accessibilityID(kSigninConfirmationCollectionViewId)),
         grey_kindOfClass([UICollectionView class]), nil);
   }
-  [[EarlGrey selectElementWithMatcher:signinUICollectionViewMatcher]
-      performAction:grey_scrollToContentEdge(kGREYContentEdgeBottom)];
-
+  NSError* error = nil;
+  [[EarlGrey selectElementWithMatcher:confirmationScrollViewMatcher]
+      assertWithMatcher:ContentViewSmallerThanScrollView()
+                  error:&error];
+  if (error) {
+    // If the consent is bigger than the scroll view, the primary button should
+    // be "MORE".
+    [[EarlGrey selectElementWithMatcher:
+                   chrome_test_util::ButtonWithAccessibilityLabelId(
+                       IDS_IOS_ACCOUNT_CONSISTENCY_CONFIRMATION_SCROLL_BUTTON)]
+        assertWithMatcher:grey_notNil()];
+    [[EarlGrey selectElementWithMatcher:confirmationScrollViewMatcher]
+        performAction:grey_scrollToContentEdge(kGREYContentEdgeBottom)];
+  }
   [[EarlGrey selectElementWithMatcher:AccountConsistencyConfirmationOkButton()]
       performAction:grey_tap()];
 }
diff --git a/ios/chrome/browser/ui/authentication/unified_consent/unified_consent_view_controller.mm b/ios/chrome/browser/ui/authentication/unified_consent/unified_consent_view_controller.mm
index f261b31a..cf0c232 100644
--- a/ios/chrome/browser/ui/authentication/unified_consent/unified_consent_view_controller.mm
+++ b/ios/chrome/browser/ui/authentication/unified_consent/unified_consent_view_controller.mm
@@ -171,7 +171,6 @@
   self.scrollView = [[UIScrollView alloc] initWithFrame:self.view.bounds];
   self.scrollView.translatesAutoresizingMaskIntoConstraints = NO;
   self.scrollView.accessibilityIdentifier = kUnifiedConsentScrollViewIdentifier;
-  self.scrollView.alwaysBounceVertical = YES;
   if (@available(iOS 11, *)) {
     // The observed behavior was buggy. When the view appears on the screen,
     // the scrollvie was not scrolled all the way to the top. Adjusting the
diff --git a/ios/chrome/browser/ui/browser_view_controller.mm b/ios/chrome/browser/ui/browser_view_controller.mm
index 5faac3fd..faeb2ae 100644
--- a/ios/chrome/browser/ui/browser_view_controller.mm
+++ b/ios/chrome/browser/ui/browser_view_controller.mm
@@ -1753,6 +1753,15 @@
     [self.primaryToolbarCoordinator.viewController.view setNeedsLayout];
     [self.primaryToolbarCoordinator.viewController.view layoutIfNeeded];
   }
+
+  // Native content pages depend on |self.view|'s safeArea.  If the BVC is
+  // presented underneath another view (such as the first time welcome view),
+  // the BVC has no safe area set during webController's layout initial, and
+  // won't automatically get another layout without forcing it here.
+  Tab* currentTab = [_model currentTab];
+  if ([self isTabNativePage:currentTab]) {
+    [currentTab.webController.view setNeedsLayout];
+  }
 }
 
 - (void)viewDidLayoutSubviews {
diff --git a/ios/chrome/browser/ui/settings/autofill_settings_egtest.mm b/ios/chrome/browser/ui/settings/autofill_settings_egtest.mm
index f129259..99c2bb1 100644
--- a/ios/chrome/browser/ui/settings/autofill_settings_egtest.mm
+++ b/ios/chrome/browser/ui/settings/autofill_settings_egtest.mm
@@ -127,9 +127,6 @@
 
 // Test that submitting a form ensures saving the data as an autofill profile.
 - (void)testAutofillProfileSaving {
-  // TODO(crbug.com/866730): Re-enable when fixed
-  EARL_GREY_TEST_DISABLED(@"Test broken.");
-
   [self loadAndSubmitTheForm];
   [self openEditAddress:@"George Washington, 1600 Pennsylvania Ave NW"];
 
@@ -150,9 +147,6 @@
 // Test that editing country names is followed by validating the value and
 // replacing it with a canonical one.
 - (void)testAutofillProfileEditing {
-  // TODO(crbug.com/866730): Re-enable when fixed
-  EARL_GREY_TEST_DISABLED(@"Test broken.");
-
   [self loadAndSubmitTheForm];
   [self openEditAddress:@"George Washington, 1600 Pennsylvania Ave NW"];
 
@@ -187,9 +181,6 @@
 
 // Test that the page for viewing autofill profile details is accessible.
 - (void)testAccessibilityOnAutofillProfileViewPage {
-  // TODO(crbug.com/866730): Re-enable when fixed
-  EARL_GREY_TEST_DISABLED(@"Test broken.");
-
   [self loadAndSubmitTheForm];
   [self openEditAddress:@"George Washington, 1600 Pennsylvania Ave NW"];
   chrome_test_util::VerifyAccessibilityForCurrentScreen();
@@ -199,9 +190,6 @@
 
 // Test that the page for editing autofill profile details is accessible.
 - (void)testAccessibilityOnAutofillProfileEditPage {
-  // TODO(crbug.com/866730): Re-enable when fixed
-  EARL_GREY_TEST_DISABLED(@"Test broken.");
-
   [self loadAndSubmitTheForm];
   [self openEditAddress:@"George Washington, 1600 Pennsylvania Ave NW"];
   // Switch on edit mode.
@@ -216,9 +204,6 @@
 // Checks that if the autofill profiles and credit cards list view is in edit
 // mode, the "autofill" and "wallet" switch items are disabled.
 - (void)testListViewEditMode {
-  // TODO(crbug.com/866730): Re-enable when fixed
-  EARL_GREY_TEST_DISABLED(@"Test broken.");
-
   [self loadAndSubmitTheForm];
 
   [ChromeEarlGreyUI openSettingsMenu];
diff --git a/ios/chrome/browser/ui/stack_view/stack_view_controller.mm b/ios/chrome/browser/ui/stack_view/stack_view_controller.mm
index 5e2c131..e8146e4 100644
--- a/ios/chrome/browser/ui/stack_view/stack_view_controller.mm
+++ b/ios/chrome/browser/ui/stack_view/stack_view_controller.mm
@@ -2085,7 +2085,8 @@
   [NSObject cancelPreviousPerformRequestsWithTarget:self];
 
   [_delegate tabSwitcher:self
-      shouldFinishWithActiveModel:_activeCardSet.tabModel];
+      shouldFinishWithActiveModel:_activeCardSet.tabModel
+                     focusOmnibox:NO];
 
   [self animateTransitionWithStyle:STACK_TRANSITION_STYLE_DISMISSING];
 }
@@ -2200,7 +2201,8 @@
   [_activeCardSet.tabModel setCurrentTab:tab];
 
   [_delegate tabSwitcher:self
-      shouldFinishWithActiveModel:_activeCardSet.tabModel];
+      shouldFinishWithActiveModel:_activeCardSet.tabModel
+                     focusOmnibox:NO];
 
   CGFloat statusBarHeight = StatusBarHeight();
   CGRect viewBounds, remainder;
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_adaptor.mm b/ios/chrome/browser/ui/tab_grid/tab_grid_adaptor.mm
index 2409a3e..74382d8 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_adaptor.mm
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_adaptor.mm
@@ -85,7 +85,9 @@
 
   // Tell the delegate to display the tab.
   DCHECK(self.delegate);
-  [self.delegate tabSwitcher:self shouldFinishWithActiveModel:targetModel];
+  [self.delegate tabSwitcher:self
+      shouldFinishWithActiveModel:targetModel
+                     focusOmnibox:NO];
 
   return tab;
 }
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator.mm b/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator.mm
index 224674f..a8cf93c 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator.mm
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator.mm
@@ -338,7 +338,7 @@
 
 #pragma mark - TabPresentationDelegate
 
-- (void)showActiveTabInPage:(TabGridPage)page {
+- (void)showActiveTabInPage:(TabGridPage)page focusOmnibox:(BOOL)focusOmnibox {
   DCHECK(self.regularTabModel && self.incognitoTabModel);
   TabModel* activeTabModel;
   switch (page) {
@@ -357,7 +357,8 @@
   // Trigger the transition through the TabSwitcher delegate. This will in turn
   // call back into this coordinator via the ViewControllerSwapping protocol.
   [self.tabSwitcher.delegate tabSwitcher:self.tabSwitcher
-             shouldFinishWithActiveModel:activeTabModel];
+             shouldFinishWithActiveModel:activeTabModel
+                            focusOmnibox:focusOmnibox];
 }
 
 #pragma mark - BrowserCommands
@@ -399,19 +400,22 @@
 
 - (void)showActiveRegularTabFromRecentTabs {
   [self.tabSwitcher.delegate tabSwitcher:self.tabSwitcher
-             shouldFinishWithActiveModel:self.regularTabModel];
+             shouldFinishWithActiveModel:self.regularTabModel
+                            focusOmnibox:NO];
 }
 
 #pragma mark - HistoryPresentationDelegate
 
 - (void)showActiveRegularTabFromHistory {
   [self.tabSwitcher.delegate tabSwitcher:self.tabSwitcher
-             shouldFinishWithActiveModel:self.regularTabModel];
+             shouldFinishWithActiveModel:self.regularTabModel
+                            focusOmnibox:NO];
 }
 
 - (void)showActiveIncognitoTabFromHistory {
   [self.tabSwitcher.delegate tabSwitcher:self.tabSwitcher
-             shouldFinishWithActiveModel:self.incognitoTabModel];
+             shouldFinishWithActiveModel:self.incognitoTabModel
+                            focusOmnibox:NO];
 }
 
 @end
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator_unittest.mm b/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator_unittest.mm
index ace48fdb..845b86ef 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator_unittest.mm
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_coordinator_unittest.mm
@@ -23,7 +23,8 @@
 @implementation TestTabSwitcherDelegate
 @synthesize didEndCalled = _didEndCalled;
 - (void)tabSwitcher:(id<TabSwitcher>)tabSwitcher
-    shouldFinishWithActiveModel:(TabModel*)tabModel {
+    shouldFinishWithActiveModel:(TabModel*)tabModel
+                   focusOmnibox:(BOOL)focusOmnibox {
   // No-op.
 }
 
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_page_control.mm b/ios/chrome/browser/ui/tab_grid/tab_grid_page_control.mm
index 120b075d..1b2df4c 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_page_control.mm
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_page_control.mm
@@ -20,7 +20,7 @@
 
 // Structure of this control:
 //
-// The page control is similar to a UISegmentedCoffee in appearance, but not in
+// The page control is similar to a UISegmentedControl in appearance, but not in
 // function. This control doesn't have segments that highlight; instead there
 // is a white "slider" that moves across the view onto whichever segment is
 // active. Each segment has an image and (optionally) a label. When the slider
@@ -302,13 +302,11 @@
   CGPoint position = [touch locationInView:self];
   CGFloat deltaX = position.x - self.dragStart.x;
   // Convert to position change.
-  CGFloat postionChange = deltaX / self.sliderRange;
+  CGFloat positionChange = deltaX / self.sliderRange;
 
-  self.sliderPosition = self.dragStartPosition + postionChange;
+  self.sliderPosition = self.dragStartPosition + positionChange;
   [self sendActionsForControlEvents:UIControlEventValueChanged];
-
-  // If the touch is now outside of the control, stop tracking it.
-  return self.touchInside;
+  return YES;
 }
 
 - (void)endTrackingWithTouch:(UITouch*)touch withEvent:(UIEvent*)event {
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.h b/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.h
index d051c80..447ffe2 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.h
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.h
@@ -20,8 +20,9 @@
 // Delegate protocol for an object that can handle presenting ("opening") tabs
 // from the tab grid.
 @protocol TabPresentationDelegate<NSObject>
-// Show the active tab in |page|, presented on top of the tab grid.
-- (void)showActiveTabInPage:(TabGridPage)page;
+// Show the active tab in |page|, presented on top of the tab grid.  The
+// omnibox will be focused after the animation if |focusOmnibox| is YES.
+- (void)showActiveTabInPage:(TabGridPage)page focusOmnibox:(BOOL)focusOmnibox;
 @end
 
 // View controller representing a tab switcher. The tab switcher has an
diff --git a/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.mm b/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.mm
index f9645b0a..78b612d 100644
--- a/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.mm
+++ b/ios/chrome/browser/ui/tab_grid/tab_grid_view_controller.mm
@@ -886,7 +886,7 @@
 
 // Tells the appropriate delegate to create a new item, and then tells the
 // presentation delegate to show the new item.
-- (void)openNewTabInPage:(TabGridPage)page {
+- (void)openNewTabInPage:(TabGridPage)page focusOmnibox:(BOOL)focusOmnibox {
   switch (page) {
     case TabGridPageIncognitoTabs:
       [self.incognitoTabsDelegate addNewItem];
@@ -907,22 +907,23 @@
       break;
   }
   self.activePage = page;
-  [self.tabPresentationDelegate showActiveTabInPage:page];
+  [self.tabPresentationDelegate showActiveTabInPage:page
+                                       focusOmnibox:focusOmnibox];
 }
 
 // Creates and shows a new regular tab.
-- (void)openNewRegularTab {
-  [self openNewTabInPage:TabGridPageRegularTabs];
+- (void)openNewRegularTabForKeyboardCommand {
+  [self openNewTabInPage:TabGridPageRegularTabs focusOmnibox:YES];
 }
 
 // Creates and shows a new incognito tab.
-- (void)openNewIncognitoTab {
-  [self openNewTabInPage:TabGridPageIncognitoTabs];
+- (void)openNewIncognitoTabForKeyboardCommand {
+  [self openNewTabInPage:TabGridPageIncognitoTabs focusOmnibox:YES];
 }
 
 // Creates and shows a new tab in the current page.
-- (void)openNewTabInCurrentPage {
-  [self openNewTabInPage:self.currentPage];
+- (void)openNewTabInCurrentPageForKeyboardCommand {
+  [self openNewTabInPage:self.currentPage focusOmnibox:YES];
 }
 
 // Broadcasts whether incognito tabs are showing.
@@ -957,7 +958,8 @@
         base::UserMetricsAction("MobileTabSwitcherOpenIncognitoTab"));
   }
   self.activePage = self.currentPage;
-  [self.tabPresentationDelegate showActiveTabInPage:self.currentPage];
+  [self.tabPresentationDelegate showActiveTabInPage:self.currentPage
+                                       focusOmnibox:NO];
   gridViewController.showsSelectionUpdates = YES;
 }
 
@@ -1009,7 +1011,8 @@
   // being triggered on release after tabs have been closed and the button
   // disabled. Ensure that action is only taken on a valid state.
   if (![[self gridViewControllerForPage:newActivePage] isGridEmpty]) {
-    [self.tabPresentationDelegate showActiveTabInPage:newActivePage];
+    [self.tabPresentationDelegate showActiveTabInPage:newActivePage
+                                         focusOmnibox:NO];
     // Record when users exit the tab grid to return to the current foreground
     // tab.
     // TODO(crbug.com/856965) : Rename metrics.
@@ -1041,7 +1044,7 @@
 }
 
 - (void)newTabButtonTapped:(id)sender {
-  [self openNewTabInCurrentPage];
+  [self openNewTabInPage:self.currentPage focusOmnibox:NO];
   // Record only when a new tab is created through the + button.
   // TODO(crbug.com/856965) : Rename metrics.
   base::RecordAction(base::UserMetricsAction("MobileToolbarStackViewNewTab"));
@@ -1075,24 +1078,24 @@
 #pragma mark - UIResponder
 
 - (NSArray*)keyCommands {
-  UIKeyCommand* newWindowShortcut =
-      [UIKeyCommand keyCommandWithInput:@"n"
-                          modifierFlags:UIKeyModifierCommand
-                                 action:@selector(openNewRegularTab)
-                   discoverabilityTitle:l10n_util::GetNSStringWithFixup(
-                                            IDS_IOS_TOOLS_MENU_NEW_TAB)];
+  UIKeyCommand* newWindowShortcut = [UIKeyCommand
+       keyCommandWithInput:@"n"
+             modifierFlags:UIKeyModifierCommand
+                    action:@selector(openNewRegularTabForKeyboardCommand)
+      discoverabilityTitle:l10n_util::GetNSStringWithFixup(
+                               IDS_IOS_TOOLS_MENU_NEW_TAB)];
   UIKeyCommand* newIncognitoWindowShortcut = [UIKeyCommand
        keyCommandWithInput:@"n"
              modifierFlags:UIKeyModifierCommand | UIKeyModifierShift
-                    action:@selector(openNewIncognitoTab)
+                    action:@selector(openNewIncognitoTabForKeyboardCommand)
       discoverabilityTitle:l10n_util::GetNSStringWithFixup(
                                IDS_IOS_TOOLS_MENU_NEW_INCOGNITO_TAB)];
-  UIKeyCommand* newTabShortcut =
-      [UIKeyCommand keyCommandWithInput:@"t"
-                          modifierFlags:UIKeyModifierCommand
-                                 action:@selector(openNewTabInCurrentPage)
-                   discoverabilityTitle:l10n_util::GetNSStringWithFixup(
-                                            IDS_IOS_TOOLS_MENU_NEW_TAB)];
+  UIKeyCommand* newTabShortcut = [UIKeyCommand
+       keyCommandWithInput:@"t"
+             modifierFlags:UIKeyModifierCommand
+                    action:@selector(openNewTabInCurrentPageForKeyboardCommand)
+      discoverabilityTitle:l10n_util::GetNSStringWithFixup(
+                               IDS_IOS_TOOLS_MENU_NEW_TAB)];
   return @[ newWindowShortcut, newIncognitoWindowShortcut, newTabShortcut ];
 }
 
diff --git a/ios/chrome/browser/ui/tab_switcher/tab_switcher.h b/ios/chrome/browser/ui/tab_switcher/tab_switcher.h
index 282bab0..8b1da3d 100644
--- a/ios/chrome/browser/ui/tab_switcher/tab_switcher.h
+++ b/ios/chrome/browser/ui/tab_switcher/tab_switcher.h
@@ -28,7 +28,8 @@
 // Informs the delegate the stack controller should be dismissed with the given
 // active model.
 - (void)tabSwitcher:(id<TabSwitcher>)tabSwitcher
-    shouldFinishWithActiveModel:(TabModel*)tabModel;
+    shouldFinishWithActiveModel:(TabModel*)tabModel
+                   focusOmnibox:(BOOL)focusOmnibox;
 
 // Informs the delegate that the stack controller is done and should be
 // dismissed.
diff --git a/ios/chrome/browser/ui/tab_switcher/tab_switcher_controller.mm b/ios/chrome/browser/ui/tab_switcher/tab_switcher_controller.mm
index 1269ec2..bef8d3c 100644
--- a/ios/chrome/browser/ui/tab_switcher/tab_switcher_controller.mm
+++ b/ios/chrome/browser/ui/tab_switcher/tab_switcher_controller.mm
@@ -872,7 +872,9 @@
   DCHECK(model);
   [[self presentedViewController] dismissViewControllerAnimated:NO
                                                      completion:nil];
-  [self.delegate tabSwitcher:self shouldFinishWithActiveModel:model];
+  [self.delegate tabSwitcher:self
+      shouldFinishWithActiveModel:model
+                     focusOmnibox:NO];
   [self performTabSwitcherTransition:TransitionType::TRANSITION_DISMISS
                            withModel:model
                             animated:animated
diff --git a/ios/chrome/browser/ui/webui/BUILD.gn b/ios/chrome/browser/ui/webui/BUILD.gn
index ed080bd5..468737f 100644
--- a/ios/chrome/browser/ui/webui/BUILD.gn
+++ b/ios/chrome/browser/ui/webui/BUILD.gn
@@ -20,8 +20,8 @@
     "suggestions_ui.h",
     "terms_ui.h",
     "terms_ui.mm",
-    "url_keyed_metrics_ui.cc",
-    "url_keyed_metrics_ui.h",
+    "ukm_internals_ui.cc",
+    "ukm_internals_ui.h",
     "version_handler.cc",
     "version_handler.h",
     "version_ui.h",
@@ -46,6 +46,7 @@
     "//components/version_info",
     "//components/version_ui",
     "//google_apis",
+    "//ios/chrome/app/resources:ios_resources",
     "//ios/chrome/app/strings",
     "//ios/chrome/browser",
     "//ios/chrome/browser/browser_state",
diff --git a/ios/chrome/browser/ui/webui/chrome_web_ui_ios_controller_factory.mm b/ios/chrome/browser/ui/webui/chrome_web_ui_ios_controller_factory.mm
index 2b8b130..77fd1b9 100644
--- a/ios/chrome/browser/ui/webui/chrome_web_ui_ios_controller_factory.mm
+++ b/ios/chrome/browser/ui/webui/chrome_web_ui_ios_controller_factory.mm
@@ -20,7 +20,7 @@
 #include "ios/chrome/browser/ui/webui/suggestions_ui.h"
 #include "ios/chrome/browser/ui/webui/sync_internals/sync_internals_ui.h"
 #include "ios/chrome/browser/ui/webui/terms_ui.h"
-#include "ios/chrome/browser/ui/webui/url_keyed_metrics_ui.h"
+#include "ios/chrome/browser/ui/webui/ukm_internals_ui.h"
 #include "ios/chrome/browser/ui/webui/version_ui.h"
 #include "url/gurl.h"
 
@@ -91,7 +91,7 @@
   if (url_host == kChromeUIFlagsHost)
     return &NewWebUIIOS<FlagsUI>;
   if (url_host == kChromeUIURLKeyedMetricsHost)
-    return &NewWebUIIOSWithHost<URLKeyedMetricsUI>;
+    return &NewWebUIIOS<UkmInternalsUI>;
 
   return nullptr;
 }
diff --git a/ios/chrome/browser/ui/webui/ukm_internals_ui.cc b/ios/chrome/browser/ui/webui/ukm_internals_ui.cc
new file mode 100644
index 0000000..477780b2
--- /dev/null
+++ b/ios/chrome/browser/ui/webui/ukm_internals_ui.cc
@@ -0,0 +1,88 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "ios/chrome/browser/ui/webui/ukm_internals_ui.h"
+
+#include <string>
+
+#include "base/memory/ref_counted_memory.h"
+#include "components/metrics_services_manager/metrics_services_manager.h"
+#include "components/ukm/debug/ukm_debug_data_extractor.h"
+#include "components/ukm/ukm_service.h"
+#include "ios/chrome/browser/application_context.h"
+#include "ios/chrome/browser/browser_state/chrome_browser_state.h"
+#include "ios/chrome/browser/chrome_url_constants.h"
+#include "ios/chrome/grit/ios_resources.h"
+#include "ios/web/public/url_data_source_ios.h"
+#include "ios/web/public/web_ui_ios_data_source.h"
+#include "ios/web/public/webui/web_ui_ios.h"
+#include "ios/web/public/webui/web_ui_ios_message_handler.h"
+
+namespace {
+
+web::WebUIIOSDataSource* CreateUkmInternalsUIHTMLSource() {
+  web::WebUIIOSDataSource* source =
+      web::WebUIIOSDataSource::Create(kChromeUIURLKeyedMetricsHost);
+
+  source->AddResourcePath("ukm_internals.js", IDR_IOS_UKM_INTERNALS_JS);
+  source->SetDefaultResource(IDR_IOS_UKM_INTERNALS_HTML);
+  source->UseGzip();
+  return source;
+}
+
+// The handler for Javascript messages for the chrome://ukm/ page.
+class UkmMessageHandler : public web::WebUIIOSMessageHandler {
+ public:
+  explicit UkmMessageHandler(const ukm::UkmService* ukm_service);
+  ~UkmMessageHandler() override;
+
+  // web::WebUIIOSMessageHandler implementation.
+  void RegisterMessages() override;
+
+ private:
+  void HandleRequestUkmData(const base::ListValue* args);
+
+  const ukm::UkmService* ukm_service_;
+
+  DISALLOW_COPY_AND_ASSIGN(UkmMessageHandler);
+};
+
+UkmMessageHandler::UkmMessageHandler(const ukm::UkmService* ukm_service)
+    : ukm_service_(ukm_service) {}
+
+UkmMessageHandler::~UkmMessageHandler() {}
+
+void UkmMessageHandler::RegisterMessages() {
+  web_ui()->RegisterMessageCallback(
+      "requestUkmData",
+      base::BindRepeating(&UkmMessageHandler::HandleRequestUkmData,
+                          base::Unretained(this)));
+}
+
+void UkmMessageHandler::HandleRequestUkmData(const base::ListValue* args) {
+  base::Value ukm_debug_data =
+      ukm::debug::UkmDebugDataExtractor::GetStructuredData(ukm_service_);
+
+  std::string callback_id;
+  args->GetString(0, &callback_id);
+  web_ui()->ResolveJavascriptCallback(base::Value(callback_id),
+                                      std::move(ukm_debug_data));
+}
+
+}  // namespace
+
+// Changes to this class should be in sync with its non-iOS equivalent
+// chrome/browser/ui/webui/ukm/ukm_internals_ui.cc
+UkmInternalsUI::UkmInternalsUI(web::WebUIIOS* web_ui)
+    : web::WebUIIOSController(web_ui) {
+  ukm::UkmService* ukm_service =
+      GetApplicationContext()->GetMetricsServicesManager()->GetUkmService();
+  web_ui->AddMessageHandler(std::make_unique<UkmMessageHandler>(ukm_service));
+
+  // Set up the chrome://ukm/ source.
+  web::WebUIIOSDataSource::Add(ios::ChromeBrowserState::FromWebUIIOS(web_ui),
+                               CreateUkmInternalsUIHTMLSource());
+}
+
+UkmInternalsUI::~UkmInternalsUI() {}
diff --git a/ios/chrome/browser/ui/webui/ukm_internals_ui.h b/ios/chrome/browser/ui/webui/ukm_internals_ui.h
new file mode 100644
index 0000000..73e583d
--- /dev/null
+++ b/ios/chrome/browser/ui/webui/ukm_internals_ui.h
@@ -0,0 +1,25 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_UI_WEBUI_UKM_INTERNALS_UI_H_
+#define IOS_CHROME_BROWSER_UI_WEBUI_UKM_INTERNALS_UI_H_
+
+#include "base/macros.h"
+#include "ios/web/public/webui/web_ui_ios_controller.h"
+
+namespace web {
+class WebUIIOS;
+}
+
+// The WebUI controller for chrome://ukm.
+class UkmInternalsUI : public web::WebUIIOSController {
+ public:
+  explicit UkmInternalsUI(web::WebUIIOS* web_ui);
+  ~UkmInternalsUI() override;
+
+ private:
+  DISALLOW_COPY_AND_ASSIGN(UkmInternalsUI);
+};
+
+#endif  // IOS_CHROME_BROWSER_UI_WEBUI_UKM_INTERNALS_UI_H_
diff --git a/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.cc b/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.cc
deleted file mode 100644
index d9851e6..0000000
--- a/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.cc
+++ /dev/null
@@ -1,86 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "ios/chrome/browser/ui/webui/url_keyed_metrics_ui.h"
-
-#include <string>
-
-#include "base/memory/ref_counted_memory.h"
-#include "components/metrics_services_manager/metrics_services_manager.h"
-#include "components/ukm/debug/ukm_debug_data_extractor.h"
-#include "components/ukm/ukm_service.h"
-#include "ios/chrome/browser/application_context.h"
-#include "ios/chrome/browser/browser_state/chrome_browser_state.h"
-#include "ios/web/public/url_data_source_ios.h"
-
-namespace {
-
-class URLKeyedMetricsUIHTMLSource : public web::URLDataSourceIOS {
- public:
-  // Construct a data source for the specified |source_name|.
-  explicit URLKeyedMetricsUIHTMLSource(const std::string& source_name);
-
-  // web::URLDataSourceIOS implementation.
-  std::string GetSource() const override;
-  void StartDataRequest(
-      const std::string& path,
-      const web::URLDataSourceIOS::GotDataCallback& callback) override;
-  std::string GetMimeType(const std::string& path) const override;
-  bool ShouldDenyXFrameOptions() const override;
-
- private:
-  ~URLKeyedMetricsUIHTMLSource() override;
-
-  ukm::UkmService* GetUkmService();
-  std::string source_name_;
-
-  DISALLOW_IMPLICIT_CONSTRUCTORS(URLKeyedMetricsUIHTMLSource);
-};
-
-}  // namespace
-
-// URLKeyedMetricsUIHTMLSource -------------------------------------------------
-
-URLKeyedMetricsUIHTMLSource::URLKeyedMetricsUIHTMLSource(
-    const std::string& source_name)
-    : source_name_(source_name) {}
-
-URLKeyedMetricsUIHTMLSource::~URLKeyedMetricsUIHTMLSource() {}
-
-std::string URLKeyedMetricsUIHTMLSource::GetSource() const {
-  return source_name_;
-}
-
-void URLKeyedMetricsUIHTMLSource::StartDataRequest(
-    const std::string& path,
-    const web::URLDataSourceIOS::GotDataCallback& callback) {
-  // TODO(crbug.com/843181): Use GetStructuredData() instead.
-  std::string data =
-      ukm::debug::UkmDebugDataExtractor::GetHTMLData(GetUkmService());
-  callback.Run(base::RefCountedString::TakeString(&data));
-}
-
-std::string URLKeyedMetricsUIHTMLSource::GetMimeType(
-    const std::string& path) const {
-  return "text/html";
-}
-
-bool URLKeyedMetricsUIHTMLSource::ShouldDenyXFrameOptions() const {
-  return web::URLDataSourceIOS::ShouldDenyXFrameOptions();
-}
-
-ukm::UkmService* URLKeyedMetricsUIHTMLSource::GetUkmService() {
-  return GetApplicationContext()->GetMetricsServicesManager()->GetUkmService();
-}
-
-// URLKeyedMetricsUI -----------------------------------------------------------
-
-URLKeyedMetricsUI::URLKeyedMetricsUI(web::WebUIIOS* web_ui,
-                                     const std::string& name)
-    : web::WebUIIOSController(web_ui) {
-  web::URLDataSourceIOS::Add(ios::ChromeBrowserState::FromWebUIIOS(web_ui),
-                             new URLKeyedMetricsUIHTMLSource(name));
-}
-
-URLKeyedMetricsUI::~URLKeyedMetricsUI() {}
diff --git a/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.h b/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.h
deleted file mode 100644
index 90a65d9..0000000
--- a/ios/chrome/browser/ui/webui/url_keyed_metrics_ui.h
+++ /dev/null
@@ -1,25 +0,0 @@
-// Copyright 2018 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef IOS_CHROME_BROWSER_UI_WEBUI_URL_KEYED_METRICS_UI_H_
-#define IOS_CHROME_BROWSER_UI_WEBUI_URL_KEYED_METRICS_UI_H_
-
-#include "base/macros.h"
-#include "ios/web/public/webui/web_ui_ios_controller.h"
-
-namespace web {
-class WebUIIOS;
-}
-
-// The WebUI controller for chrome://ukm.
-class URLKeyedMetricsUI : public web::WebUIIOSController {
- public:
-  URLKeyedMetricsUI(web::WebUIIOS* web_ui, const std::string& name);
-  ~URLKeyedMetricsUI() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(URLKeyedMetricsUI);
-};
-
-#endif  // IOS_CHROME_BROWSER_UI_WEBUI_URL_KEYED_METRICS_UI_H_
diff --git a/ios/web/public/webui/web_ui_ios.h b/ios/web/public/webui/web_ui_ios.h
index cc97807..feaf67b 100644
--- a/ios/web/public/webui/web_ui_ios.h
+++ b/ios/web/public/webui/web_ui_ios.h
@@ -81,6 +81,18 @@
   virtual void CallJavascriptFunction(
       const std::string& function_name,
       const std::vector<const base::Value*>& args) = 0;
+
+  // Helper method for responding to Javascript requests initiated with
+  // cr.sendWithPromise() (defined in cr.js) for the case where the returned
+  // promise should be resolved (request succeeded).
+  virtual void ResolveJavascriptCallback(const base::Value& callback_id,
+                                         const base::Value& response) = 0;
+
+  // Helper method for responding to Javascript requests initiated with
+  // cr.sendWithPromise() (defined in cr.js), for the case where the returned
+  // promise should be rejected (request failed).
+  virtual void RejectJavascriptCallback(const base::Value& callback_id,
+                                        const base::Value& response) = 0;
 };
 
 }  // namespace web
diff --git a/ios/web/web_state/navigation_context_impl.h b/ios/web/web_state/navigation_context_impl.h
index 0af55de..76e1034 100644
--- a/ios/web/web_state/navigation_context_impl.h
+++ b/ios/web/web_state/navigation_context_impl.h
@@ -73,6 +73,11 @@
   bool IsLoadingErrorPage() const;
   void SetLoadingErrorPage(bool is_loading_error_page);
 
+  // true if this navigation context is a placeholder navigation associated with
+  // a native view URL and the native content is already presented.
+  bool IsNativeContentPresented() const;
+  void SetIsNativeContentPresented(bool is_native_content_presented);
+
  private:
   NavigationContextImpl(WebState* web_state,
                         const GURL& url,
@@ -95,6 +100,7 @@
   int navigation_item_unique_id_ = -1;
   WKNavigationType wk_navigation_type_ = WKNavigationTypeOther;
   bool is_loading_error_page_ = false;
+  bool is_native_content_presented_ = false;
 
   DISALLOW_COPY_AND_ASSIGN(NavigationContextImpl);
 };
diff --git a/ios/web/web_state/navigation_context_impl.mm b/ios/web/web_state/navigation_context_impl.mm
index f7c20a17..322929a 100644
--- a/ios/web/web_state/navigation_context_impl.mm
+++ b/ios/web/web_state/navigation_context_impl.mm
@@ -158,6 +158,15 @@
   is_loading_error_page_ = is_loading_error_page;
 }
 
+bool NavigationContextImpl::IsNativeContentPresented() const {
+  return is_native_content_presented_;
+}
+
+void NavigationContextImpl::SetIsNativeContentPresented(
+    bool is_native_content_presented) {
+  is_native_content_presented_ = is_native_content_presented;
+}
+
 NavigationContextImpl::NavigationContextImpl(WebState* web_state,
                                              const GURL& url,
                                              bool has_user_gesture,
diff --git a/ios/web/web_state/ui/crw_web_controller.mm b/ios/web/web_state/ui/crw_web_controller.mm
index a4c59f1..4e48ec8 100644
--- a/ios/web/web_state/ui/crw_web_controller.mm
+++ b/ios/web/web_state/ui/crw_web_controller.mm
@@ -568,14 +568,13 @@
 - (void)updateCurrentBackForwardListItemHolder;
 
 // Presents native content using the native controller for |item| without
-// notifying WebStateObservers. This is used when loading native view for a new
-// navigation to avoid the delay introduced by placeholder navigation.
+// notifying WebStateObservers. This method does not modify the underlying web
+// view. It simply covers the web view with the native content.
+// |-didLoadNativeContentForNavigationItem| must be called some time later
+// to notify WebStateObservers.
 - (void)presentNativeContentForNavigationItem:(web::NavigationItem*)item;
-// Presents native content using the native controller for |item| and notifies
-// WebStateObservers the completion of this navigation. This method does not
-// modify the underlying web view. It simply covers the web view with the native
-// content.
-- (void)loadNativeContentForNavigationItem:(web::NavigationItem*)item;
+// Notifies WebStateObservers the completion of this navigation.
+- (void)didLoadNativeContentForNavigationItem:(web::NavigationItem*)item;
 // Loads a blank page directly into WKWebView as a placeholder for a Native View
 // or WebUI URL. This page has the URL about:blank?for=<encoded original URL>.
 // The completion handler is called in the |webView:didFinishNavigation|
@@ -1799,12 +1798,15 @@
   if (!web::GetWebClient()->IsSlimNavigationManagerEnabled()) {
     // Free the web view.
     [self removeWebView];
-    [self loadNativeContentForNavigationItem:self.currentNavItem];
+    [self presentNativeContentForNavigationItem:self.currentNavItem];
+    [self didLoadNativeContentForNavigationItem:self.currentNavItem];
   } else {
     // Just present the native view now. Leave the rest of native content load
     // until the placeholder navigation finishes.
     [self presentNativeContentForNavigationItem:self.currentNavItem];
-    [self loadPlaceholderInWebViewForURL:self.currentNavItem->GetVirtualURL()];
+    web::NavigationContextImpl* context = [self
+        loadPlaceholderInWebViewForURL:self.currentNavItem->GetVirtualURL()];
+    context->SetIsNativeContentPresented(true);
   }
 }
 
@@ -1826,9 +1828,7 @@
   }
 }
 
-- (void)loadNativeContentForNavigationItem:(web::NavigationItem*)item {
-  [self presentNativeContentForNavigationItem:item];
-
+- (void)didLoadNativeContentForNavigationItem:(web::NavigationItem*)item {
   const GURL targetURL = item ? item->GetURL() : GURL::EmptyGURL();
   const web::Referrer referrer;
   std::unique_ptr<web::NavigationContextImpl> navigationContext =
@@ -1915,8 +1915,9 @@
   if (!_containerView)
     return;
 
-  // WKBasedNavigationManagerImpl requires web usage to be enabled to load any
-  // URL. So bail if web usage is disabled, and let the URL be loaded when web
+  // WKBasedNavigationManagerImpl needs WKWebView to load native views, but
+  // WKWebView cannot be created while web usage is disabled to avoid breaking
+  // clearing browser data. Bail now and let the URL be loaded when web
   // usage is enabled again. This can happen when purging web pages when an
   // interstitial is presented over a native view. See https://crbug.com/865985
   // for details.
@@ -4900,7 +4901,12 @@
       }
 
       if ([self shouldLoadURLInNativeView:item->GetURL()]) {
-        [self loadNativeContentForNavigationItem:item];
+        // Native content may have already been presented if this navigation is
+        // started in |-loadCurrentURLInNativeView|. If not, present it now.
+        if (!context || !context->IsNativeContentPresented()) {
+          [self presentNativeContentForNavigationItem:item];
+        }
+        [self didLoadNativeContentForNavigationItem:item];
       } else if (isWebUIURL) {
         DCHECK(_webUIManager);
         [_webUIManager loadWebUIForURL:item->GetURL()];
diff --git a/ios/web/webui/web_ui_ios_impl.h b/ios/web/webui/web_ui_ios_impl.h
index 5a8e27fc..a431a4f 100644
--- a/ios/web/webui/web_ui_ios_impl.h
+++ b/ios/web/webui/web_ui_ios_impl.h
@@ -56,6 +56,10 @@
   void CallJavascriptFunction(
       const std::string& function_name,
       const std::vector<const base::Value*>& args) override;
+  void ResolveJavascriptCallback(const base::Value& callback_id,
+                                 const base::Value& response) override;
+  void RejectJavascriptCallback(const base::Value& callback_id,
+                                const base::Value& response) override;
 
  private:
   // Executes JavaScript asynchronously on the page.
diff --git a/ios/web/webui/web_ui_ios_impl.mm b/ios/web/webui/web_ui_ios_impl.mm
index 3345d08..3056181 100644
--- a/ios/web/webui/web_ui_ios_impl.mm
+++ b/ios/web/webui/web_ui_ios_impl.mm
@@ -121,6 +121,20 @@
   ExecuteJavascript(GetJavascriptCall(function_name, args));
 }
 
+void WebUIIOSImpl::ResolveJavascriptCallback(const base::Value& callback_id,
+                                             const base::Value& response) {
+  // cr.webUIResponse is a global JS function exposed from cr.js.
+  CallJavascriptFunction("cr.webUIResponse", callback_id, base::Value(true),
+                         response);
+}
+
+void WebUIIOSImpl::RejectJavascriptCallback(const base::Value& callback_id,
+                                            const base::Value& response) {
+  // cr.webUIResponse is a global JS function exposed from cr.js.
+  CallJavascriptFunction("cr.webUIResponse", callback_id, base::Value(false),
+                         response);
+}
+
 void WebUIIOSImpl::RegisterMessageCallback(const std::string& message,
                                            const MessageCallback& callback) {
   message_callbacks_.insert(std::make_pair(message, callback));
diff --git a/mojo/public/java/bindings/src/org/chromium/mojo/bindings/Connector.java b/mojo/public/java/bindings/src/org/chromium/mojo/bindings/Connector.java
index 45f1fc7..54612093 100644
--- a/mojo/public/java/bindings/src/org/chromium/mojo/bindings/Connector.java
+++ b/mojo/public/java/bindings/src/org/chromium/mojo/bindings/Connector.java
@@ -158,7 +158,11 @@
     private void onError(MojoException exception) {
         close();
         if (mErrorHandler != null) {
-            mErrorHandler.onConnectionError(exception);
+            try {
+                mErrorHandler.onConnectionError(exception);
+            } catch (RuntimeException e) {
+                ExceptionHandler.DefaultExceptionHandler.getInstance().handleException(e);
+            }
         }
     }
 
diff --git a/net/BUILD.gn b/net/BUILD.gn
index 380809e6..9c8dee6 100644
--- a/net/BUILD.gn
+++ b/net/BUILD.gn
@@ -1351,6 +1351,24 @@
       "third_party/quic/core/frames/quic_stream_id_blocked_frame.h",
       "third_party/quic/core/frames/quic_window_update_frame.cc",
       "third_party/quic/core/frames/quic_window_update_frame.h",
+      "third_party/quic/core/http/quic_client_promised_info.cc",
+      "third_party/quic/core/http/quic_client_promised_info.h",
+      "third_party/quic/core/http/quic_client_push_promise_index.cc",
+      "third_party/quic/core/http/quic_client_push_promise_index.h",
+      "third_party/quic/core/http/quic_header_list.cc",
+      "third_party/quic/core/http/quic_header_list.h",
+      "third_party/quic/core/http/quic_headers_stream.cc",
+      "third_party/quic/core/http/quic_headers_stream.h",
+      "third_party/quic/core/http/quic_server_session_base.cc",
+      "third_party/quic/core/http/quic_server_session_base.h",
+      "third_party/quic/core/http/quic_spdy_client_session_base.cc",
+      "third_party/quic/core/http/quic_spdy_client_session_base.h",
+      "third_party/quic/core/http/quic_spdy_session.cc",
+      "third_party/quic/core/http/quic_spdy_session.h",
+      "third_party/quic/core/http/quic_spdy_stream.cc",
+      "third_party/quic/core/http/quic_spdy_stream.h",
+      "third_party/quic/core/http/spdy_utils.cc",
+      "third_party/quic/core/http/spdy_utils.h",
       "third_party/quic/core/packet_number_indexed_queue.h",
       "third_party/quic/core/quic_ack_listener_interface.cc",
       "third_party/quic/core/quic_ack_listener_interface.h",
@@ -1365,10 +1383,6 @@
       "third_party/quic/core/quic_buffer_allocator.h",
       "third_party/quic/core/quic_buffered_packet_store.cc",
       "third_party/quic/core/quic_buffered_packet_store.h",
-      "third_party/quic/core/quic_client_promised_info.cc",
-      "third_party/quic/core/quic_client_promised_info.h",
-      "third_party/quic/core/quic_client_push_promise_index.cc",
-      "third_party/quic/core/quic_client_push_promise_index.h",
       "third_party/quic/core/quic_config.cc",
       "third_party/quic/core/quic_config.h",
       "third_party/quic/core/quic_connection.cc",
@@ -1401,10 +1415,6 @@
       "third_party/quic/core/quic_flow_controller.h",
       "third_party/quic/core/quic_framer.cc",
       "third_party/quic/core/quic_framer.h",
-      "third_party/quic/core/quic_header_list.cc",
-      "third_party/quic/core/quic_header_list.h",
-      "third_party/quic/core/quic_headers_stream.cc",
-      "third_party/quic/core/quic_headers_stream.h",
       "third_party/quic/core/quic_one_block_arena.h",
       "third_party/quic/core/quic_packet_creator.cc",
       "third_party/quic/core/quic_packet_creator.h",
@@ -1420,20 +1430,12 @@
       "third_party/quic/core/quic_sent_packet_manager.h",
       "third_party/quic/core/quic_server_id.cc",
       "third_party/quic/core/quic_server_id.h",
-      "third_party/quic/core/quic_server_session_base.cc",
-      "third_party/quic/core/quic_server_session_base.h",
       "third_party/quic/core/quic_session.cc",
       "third_party/quic/core/quic_session.h",
       "third_party/quic/core/quic_simple_buffer_allocator.cc",
       "third_party/quic/core/quic_simple_buffer_allocator.h",
       "third_party/quic/core/quic_socket_address_coder.cc",
       "third_party/quic/core/quic_socket_address_coder.h",
-      "third_party/quic/core/quic_spdy_client_session_base.cc",
-      "third_party/quic/core/quic_spdy_client_session_base.h",
-      "third_party/quic/core/quic_spdy_session.cc",
-      "third_party/quic/core/quic_spdy_session.h",
-      "third_party/quic/core/quic_spdy_stream.cc",
-      "third_party/quic/core/quic_spdy_stream.h",
       "third_party/quic/core/quic_stream.cc",
       "third_party/quic/core/quic_stream.h",
       "third_party/quic/core/quic_stream_frame_data_producer.h",
@@ -1463,8 +1465,6 @@
       "third_party/quic/core/quic_versions.h",
       "third_party/quic/core/quic_write_blocked_list.cc",
       "third_party/quic/core/quic_write_blocked_list.h",
-      "third_party/quic/core/spdy_utils.cc",
-      "third_party/quic/core/spdy_utils.h",
       "third_party/quic/core/tls_client_handshaker.cc",
       "third_party/quic/core/tls_client_handshaker.h",
       "third_party/quic/core/tls_handshaker.cc",
@@ -3105,8 +3105,6 @@
     "third_party/quic/test_tools/quic_session_peer.h",
     "third_party/quic/test_tools/quic_spdy_session_peer.cc",
     "third_party/quic/test_tools/quic_spdy_session_peer.h",
-    "third_party/quic/test_tools/quic_spdy_stream_peer.cc",
-    "third_party/quic/test_tools/quic_spdy_stream_peer.h",
     "third_party/quic/test_tools/quic_stream_peer.cc",
     "third_party/quic/test_tools/quic_stream_peer.h",
     "third_party/quic/test_tools/quic_stream_send_buffer_peer.cc",
@@ -3197,17 +3195,17 @@
   sources = [
     "third_party/quic/core/chlo_extractor.cc",
     "third_party/quic/core/chlo_extractor.h",
+    "third_party/quic/core/http/quic_spdy_client_session.cc",
+    "third_party/quic/core/http/quic_spdy_client_session.h",
+    "third_party/quic/core/http/quic_spdy_client_stream.cc",
+    "third_party/quic/core/http/quic_spdy_client_stream.h",
+    "third_party/quic/core/http/quic_spdy_server_stream_base.cc",
+    "third_party/quic/core/http/quic_spdy_server_stream_base.h",
     "third_party/quic/core/quic_dispatcher.cc",
     "third_party/quic/core/quic_dispatcher.h",
     "third_party/quic/core/quic_packet_writer_wrapper.cc",
     "third_party/quic/core/quic_packet_writer_wrapper.h",
     "third_party/quic/core/quic_process_packet_interface.h",
-    "third_party/quic/core/quic_spdy_client_session.cc",
-    "third_party/quic/core/quic_spdy_client_session.h",
-    "third_party/quic/core/quic_spdy_client_stream.cc",
-    "third_party/quic/core/quic_spdy_client_stream.h",
-    "third_party/quic/core/quic_spdy_server_stream_base.cc",
-    "third_party/quic/core/quic_spdy_server_stream_base.h",
     "third_party/quic/core/quic_time_wait_list_manager.cc",
     "third_party/quic/core/quic_time_wait_list_manager.h",
     "third_party/quic/core/stateless_rejector.cc",
@@ -4975,13 +4973,15 @@
     "third_party/quic/core/crypto/quic_tls_adapter_test.cc",
     "third_party/quic/core/crypto/transport_parameters_test.cc",
     "third_party/quic/core/frames/quic_frames_test.cc",
+    "third_party/quic/core/http/quic_client_promised_info_test.cc",
+    "third_party/quic/core/http/quic_client_push_promise_index_test.cc",
+    "third_party/quic/core/http/quic_header_list_test.cc",
+    "third_party/quic/core/http/quic_headers_stream_test.cc",
     "third_party/quic/core/packet_number_indexed_queue_test.cc",
     "third_party/quic/core/quic_alarm_test.cc",
     "third_party/quic/core/quic_arena_scoped_ptr_test.cc",
     "third_party/quic/core/quic_bandwidth_test.cc",
     "third_party/quic/core/quic_buffered_packet_store_test.cc",
-    "third_party/quic/core/quic_client_promised_info_test.cc",
-    "third_party/quic/core/quic_client_push_promise_index_test.cc",
     "third_party/quic/core/quic_config_test.cc",
     "third_party/quic/core/quic_connection_test.cc",
     "third_party/quic/core/quic_control_frame_manager_test.cc",
@@ -4992,8 +4992,6 @@
     "third_party/quic/core/quic_error_codes_test.cc",
     "third_party/quic/core/quic_flow_controller_test.cc",
     "third_party/quic/core/quic_framer_test.cc",
-    "third_party/quic/core/quic_header_list_test.cc",
-    "third_party/quic/core/quic_headers_stream_test.cc",
     "third_party/quic/core/quic_ietf_framer_test.cc",
     "third_party/quic/core/tls_handshaker_test.cc",
 
@@ -5065,6 +5063,10 @@
     "test/run_all_unittests.cc",
     "test/tcp_socket_proxy_unittest.cc",
     "third_party/nist-pkits/pkits_testcases-inl.h",
+    "third_party/quic/core/http/quic_server_session_base_test.cc",
+    "third_party/quic/core/http/quic_spdy_session_test.cc",
+    "third_party/quic/core/http/quic_spdy_stream_test.cc",
+    "third_party/quic/core/http/spdy_utils_test.cc",
     "third_party/quic/core/quic_dispatcher_test.cc",
     "third_party/quic/core/quic_one_block_arena_test.cc",
     "third_party/quic/core/quic_packet_creator_test.cc",
@@ -5072,12 +5074,9 @@
     "third_party/quic/core/quic_received_packet_manager_test.cc",
     "third_party/quic/core/quic_sent_packet_manager_test.cc",
     "third_party/quic/core/quic_server_id_test.cc",
-    "third_party/quic/core/quic_server_session_base_test.cc",
     "third_party/quic/core/quic_session_test.cc",
     "third_party/quic/core/quic_simple_buffer_allocator_test.cc",
     "third_party/quic/core/quic_socket_address_coder_test.cc",
-    "third_party/quic/core/quic_spdy_session_test.cc",
-    "third_party/quic/core/quic_spdy_stream_test.cc",
     "third_party/quic/core/quic_stream_send_buffer_test.cc",
     "third_party/quic/core/quic_stream_sequencer_buffer_test.cc",
     "third_party/quic/core/quic_stream_sequencer_test.cc",
@@ -5092,7 +5091,6 @@
     "third_party/quic/core/quic_version_manager_test.cc",
     "third_party/quic/core/quic_versions_test.cc",
     "third_party/quic/core/quic_write_blocked_list_test.cc",
-    "third_party/quic/core/spdy_utils_test.cc",
     "third_party/quic/platform/api/quic_endian_test.cc",
     "third_party/quic/platform/api/quic_hostname_utils_test.cc",
     "third_party/quic/platform/api/quic_lru_cache_test.cc",
@@ -5271,12 +5269,12 @@
   if (is_linux) {
     sources += [
       "third_party/quic/core/chlo_extractor_test.cc",
-      "third_party/quic/core/end_to_end_test.cc",
+      "third_party/quic/core/http/end_to_end_test.cc",
+      "third_party/quic/core/http/quic_spdy_client_session_test.cc",
+      "third_party/quic/core/http/quic_spdy_client_stream_test.cc",
+      "third_party/quic/core/http/quic_spdy_server_stream_base_test.cc",
       "third_party/quic/core/quic_epoll_alarm_factory_test.cc",
       "third_party/quic/core/quic_epoll_connection_helper_test.cc",
-      "third_party/quic/core/quic_spdy_client_session_test.cc",
-      "third_party/quic/core/quic_spdy_client_stream_test.cc",
-      "third_party/quic/core/quic_spdy_server_stream_base_test.cc",
       "third_party/quic/core/stateless_rejector_test.cc",
       "third_party/quic/platform/impl/quic_epoll_clock_test.cc",
       "third_party/quic/platform/impl/quic_socket_utils_test.cc",
diff --git a/net/quic/chromium/bidirectional_stream_quic_impl_unittest.cc b/net/quic/chromium/bidirectional_stream_quic_impl_unittest.cc
index d943b472..4a3475a 100644
--- a/net/quic/chromium/bidirectional_stream_quic_impl_unittest.cc
+++ b/net/quic/chromium/bidirectional_stream_quic_impl_unittest.cc
@@ -38,8 +38,8 @@
 #include "net/third_party/quic/core/crypto/crypto_protocol.h"
 #include "net/third_party/quic/core/crypto/quic_decrypter.h"
 #include "net/third_party/quic/core/crypto/quic_encrypter.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/quic/platform/api/quic_text_utils.h"
diff --git a/net/quic/chromium/quic_chromium_client_session.cc b/net/quic/chromium/quic_chromium_client_session.cc
index 1cc80d6..7de78f8 100644
--- a/net/quic/chromium/quic_chromium_client_session.cc
+++ b/net/quic/chromium/quic_chromium_client_session.cc
@@ -39,8 +39,8 @@
 #include "net/ssl/ssl_connection_status_flags.h"
 #include "net/ssl/ssl_info.h"
 #include "net/ssl/token_binding.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/traffic_annotation/network_traffic_annotation.h"
 #include "third_party/boringssl/src/include/openssl/ssl.h"
diff --git a/net/quic/chromium/quic_chromium_client_session.h b/net/quic/chromium/quic_chromium_client_session.h
index ecae249..43af292 100644
--- a/net/quic/chromium/quic_chromium_client_session.h
+++ b/net/quic/chromium/quic_chromium_client_session.h
@@ -38,11 +38,11 @@
 #include "net/spdy/http2_priority_dependencies.h"
 #include "net/spdy/multiplexed_session.h"
 #include "net/spdy/server_push_delegate.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
 #include "net/third_party/quic/core/quic_crypto_client_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
 #include "net/third_party/quic/core/quic_time.h"
 #include "net/traffic_annotation/network_traffic_annotation.h"
 
diff --git a/net/quic/chromium/quic_chromium_client_session_test.cc b/net/quic/chromium/quic_chromium_client_session_test.cc
index f6d2b78..68d0b3c 100644
--- a/net/quic/chromium/quic_chromium_client_session_test.cc
+++ b/net/quic/chromium/quic_chromium_client_session_test.cc
@@ -39,7 +39,7 @@
 #include "net/third_party/quic/core/crypto/crypto_protocol.h"
 #include "net/third_party/quic/core/crypto/quic_decrypter.h"
 #include "net/third_party/quic/core/crypto/quic_encrypter.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 #include "net/third_party/quic/core/quic_packet_writer.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
diff --git a/net/quic/chromium/quic_chromium_client_stream.cc b/net/quic/chromium/quic_chromium_client_stream.cc
index 8a7e629..fae140f 100644
--- a/net/quic/chromium/quic_chromium_client_stream.cc
+++ b/net/quic/chromium/quic_chromium_client_stream.cc
@@ -16,9 +16,9 @@
 #include "net/quic/chromium/quic_chromium_client_session.h"
 #include "net/quic/chromium/quic_http_utils.h"
 #include "net/spdy/spdy_log_util.h"
-#include "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_write_blocked_list.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 
 namespace net {
 namespace {
diff --git a/net/quic/chromium/quic_chromium_client_stream.h b/net/quic/chromium/quic_chromium_client_stream.h
index 75ce4f9..dea459f1 100644
--- a/net/quic/chromium/quic_chromium_client_stream.h
+++ b/net/quic/chromium/quic_chromium_client_stream.h
@@ -22,7 +22,7 @@
 #include "net/http/http_response_info.h"
 #include "net/http/http_stream.h"
 #include "net/log/net_log_with_source.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/traffic_annotation/network_traffic_annotation.h"
 
diff --git a/net/quic/chromium/quic_chromium_client_stream_test.cc b/net/quic/chromium/quic_chromium_client_stream_test.cc
index 39a97df..39de871 100644
--- a/net/quic/chromium/quic_chromium_client_stream_test.cc
+++ b/net/quic/chromium/quic_chromium_client_stream_test.cc
@@ -15,10 +15,10 @@
 #include "net/quic/chromium/quic_chromium_client_session.h"
 #include "net/test/gtest_util.h"
 #include "net/test/test_with_scoped_task_environment.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/quic/test_tools/crypto_test_utils.h"
diff --git a/net/quic/chromium/quic_connection_logger.h b/net/quic/chromium/quic_connection_logger.h
index 9716677..2763935 100644
--- a/net/quic/chromium/quic_connection_logger.h
+++ b/net/quic/chromium/quic_connection_logger.h
@@ -18,9 +18,9 @@
 #include "net/log/net_log_with_source.h"
 #include "net/socket/socket_performance_watcher.h"
 #include "net/third_party/quic/core/crypto/crypto_handshake_message.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_session.h"
 
 namespace base {
 class HistogramBase;
diff --git a/net/quic/chromium/quic_http_stream.cc b/net/quic/chromium/quic_http_stream.cc
index d097326..69aef5b 100644
--- a/net/quic/chromium/quic_http_stream.cc
+++ b/net/quic/chromium/quic_http_stream.cc
@@ -21,10 +21,10 @@
 #include "net/quic/chromium/quic_http_utils.h"
 #include "net/spdy/spdy_http_utils.h"
 #include "net/ssl/ssl_info.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_stream_sequencer.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/spdy/core/spdy_frame_builder.h"
 #include "net/third_party/spdy/core/spdy_framer.h"
diff --git a/net/quic/chromium/quic_http_stream.h b/net/quic/chromium/quic_http_stream.h
index 38c805c6..44be6562 100644
--- a/net/quic/chromium/quic_http_stream.h
+++ b/net/quic/chromium/quic_http_stream.h
@@ -24,7 +24,7 @@
 #include "net/quic/chromium/quic_chromium_client_session.h"
 #include "net/quic/chromium/quic_chromium_client_stream.h"
 #include "net/spdy/multiplexed_http_stream.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
 #include "net/third_party/quic/core/quic_packets.h"
 
 namespace net {
diff --git a/net/quic/chromium/quic_http_stream_test.cc b/net/quic/chromium/quic_http_stream_test.cc
index 2eea213..94686e0 100644
--- a/net/quic/chromium/quic_http_stream_test.cc
+++ b/net/quic/chromium/quic_http_stream_test.cc
@@ -48,9 +48,9 @@
 #include "net/third_party/quic/core/crypto/crypto_protocol.h"
 #include "net/third_party/quic/core/crypto/quic_decrypter.h"
 #include "net/third_party/quic/core/crypto/quic_encrypter.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_write_blocked_list.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/quic/test_tools/crypto_test_utils.h"
diff --git a/net/quic/chromium/quic_stream_factory.cc b/net/quic/chromium/quic_stream_factory.cc
index 74af500..56929b42 100644
--- a/net/quic/chromium/quic_stream_factory.cc
+++ b/net/quic/chromium/quic_stream_factory.cc
@@ -53,7 +53,7 @@
 #include "net/ssl/token_binding.h"
 #include "net/third_party/quic/core/crypto/proof_verifier.h"
 #include "net/third_party/quic/core/crypto/quic_random.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_clock.h"
diff --git a/net/quic/chromium/quic_stream_factory.h b/net/quic/chromium/quic_stream_factory.h
index 7fe42c6..29db22a 100644
--- a/net/quic/chromium/quic_stream_factory.h
+++ b/net/quic/chromium/quic_stream_factory.h
@@ -35,7 +35,7 @@
 #include "net/quic/chromium/quic_session_key.h"
 #include "net/socket/client_socket_pool.h"
 #include "net/ssl/ssl_config_service.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
 #include "net/third_party/quic/core/quic_config.h"
 #include "net/third_party/quic/core/quic_crypto_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
diff --git a/net/quic/chromium/quic_stream_factory_test.cc b/net/quic/chromium/quic_stream_factory_test.cc
index f550b40b..65eab8e 100644
--- a/net/quic/chromium/quic_stream_factory_test.cc
+++ b/net/quic/chromium/quic_stream_factory_test.cc
@@ -49,7 +49,7 @@
 #include "net/third_party/quic/core/crypto/quic_crypto_client_config.h"
 #include "net/third_party/quic/core/crypto/quic_decrypter.h"
 #include "net/third_party/quic/core/crypto/quic_encrypter.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 #include "net/third_party/quic/platform/api/quic_test.h"
 #include "net/third_party/quic/test_tools/mock_clock.h"
 #include "net/third_party/quic/test_tools/mock_random.h"
diff --git a/net/spdy/spdy_session.cc b/net/spdy/spdy_session.cc
index c48e02f..2049ec2 100644
--- a/net/spdy/spdy_session.cc
+++ b/net/spdy/spdy_session.cc
@@ -54,7 +54,7 @@
 #include "net/ssl/channel_id_service.h"
 #include "net/ssl/ssl_cipher_suite_names.h"
 #include "net/ssl/ssl_connection_status_flags.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/spdy/core/spdy_frame_builder.h"
 #include "net/third_party/spdy/core/spdy_protocol.h"
 #include "url/url_constants.h"
diff --git a/net/third_party/quic/core/congestion_control/bbr_sender.h b/net/third_party/quic/core/congestion_control/bbr_sender.h
index ba1595aa..6e1f275 100644
--- a/net/third_party/quic/core/congestion_control/bbr_sender.h
+++ b/net/third_party/quic/core/congestion_control/bbr_sender.h
@@ -96,6 +96,8 @@
             QuicPacketCount initial_tcp_congestion_window,
             QuicPacketCount max_tcp_congestion_window,
             QuicRandom* random);
+  BbrSender(const BbrSender&) = delete;
+  BbrSender& operator=(const BbrSender&) = delete;
   ~BbrSender() override;
 
   // Start implementation of SendAlgorithmInterface.
@@ -384,8 +386,6 @@
   bool probe_rtt_disabled_if_app_limited_;
   bool app_limited_since_last_probe_rtt_;
   QuicTime::Delta min_rtt_since_last_probe_rtt_;
-
-  DISALLOW_COPY_AND_ASSIGN(BbrSender);
 };
 
 QUIC_EXPORT_PRIVATE std::ostream& operator<<(std::ostream& os,
diff --git a/net/third_party/quic/core/congestion_control/cubic_bytes.h b/net/third_party/quic/core/congestion_control/cubic_bytes.h
index ef1e382..990dae8f 100644
--- a/net/third_party/quic/core/congestion_control/cubic_bytes.h
+++ b/net/third_party/quic/core/congestion_control/cubic_bytes.h
@@ -26,6 +26,8 @@
 class QUIC_EXPORT_PRIVATE CubicBytes {
  public:
   explicit CubicBytes(const QuicClock* clock);
+  CubicBytes(const CubicBytes&) = delete;
+  CubicBytes& operator=(const CubicBytes&) = delete;
 
   void SetNumConnections(int num_connections);
 
@@ -94,8 +96,6 @@
 
   // Last congestion window in packets computed by cubic function.
   QuicByteCount last_target_congestion_window_;
-
-  DISALLOW_COPY_AND_ASSIGN(CubicBytes);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/congestion_control/general_loss_algorithm.h b/net/third_party/quic/core/congestion_control/general_loss_algorithm.h
index 3409279f..e69e0c7 100644
--- a/net/third_party/quic/core/congestion_control/general_loss_algorithm.h
+++ b/net/third_party/quic/core/congestion_control/general_loss_algorithm.h
@@ -27,6 +27,8 @@
 
   GeneralLossAlgorithm();
   explicit GeneralLossAlgorithm(LossDetectionType loss_type);
+  GeneralLossAlgorithm(const GeneralLossAlgorithm&) = delete;
+  GeneralLossAlgorithm& operator=(const GeneralLossAlgorithm&) = delete;
   ~GeneralLossAlgorithm() override {}
 
   LossDetectionType GetLossDetectionType() const override;
@@ -70,8 +72,6 @@
   QuicPacketNumber largest_previously_acked_;
   // The largest lost packet.
   QuicPacketNumber largest_lost_;
-
-  DISALLOW_COPY_AND_ASSIGN(GeneralLossAlgorithm);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/congestion_control/hybrid_slow_start.h b/net/third_party/quic/core/congestion_control/hybrid_slow_start.h
index cbc2b890..a171bcc 100644
--- a/net/third_party/quic/core/congestion_control/hybrid_slow_start.h
+++ b/net/third_party/quic/core/congestion_control/hybrid_slow_start.h
@@ -28,6 +28,8 @@
 class QUIC_EXPORT_PRIVATE HybridSlowStart {
  public:
   HybridSlowStart();
+  HybridSlowStart(const HybridSlowStart&) = delete;
+  HybridSlowStart& operator=(const HybridSlowStart&) = delete;
 
   void OnPacketAcked(QuicPacketNumber acked_packet_number);
 
@@ -75,8 +77,6 @@
   QuicPacketNumber end_packet_number_;  // End of the receive round.
   uint32_t rtt_sample_count_;  // Number of rtt samples in the current round.
   QuicTime::Delta current_min_rtt_;  // The minimum rtt of current round.
-
-  DISALLOW_COPY_AND_ASSIGN(HybridSlowStart);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/congestion_control/pacing_sender.h b/net/third_party/quic/core/congestion_control/pacing_sender.h
index f8eddcd..bb018b5 100644
--- a/net/third_party/quic/core/congestion_control/pacing_sender.h
+++ b/net/third_party/quic/core/congestion_control/pacing_sender.h
@@ -32,6 +32,8 @@
 class QUIC_EXPORT_PRIVATE PacingSender {
  public:
   PacingSender();
+  PacingSender(const PacingSender&) = delete;
+  PacingSender& operator=(const PacingSender&) = delete;
   ~PacingSender();
 
   // Sets the underlying sender. Does not take ownership of |sender|. |sender|
@@ -99,8 +101,6 @@
   // Indicates whether pacing throttles the sending. If true, make up for lost
   // time.
   bool pacing_limited_;
-
-  DISALLOW_COPY_AND_ASSIGN(PacingSender);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/congestion_control/rtt_stats.h b/net/third_party/quic/core/congestion_control/rtt_stats.h
index 168de9d..93d10166 100644
--- a/net/third_party/quic/core/congestion_control/rtt_stats.h
+++ b/net/third_party/quic/core/congestion_control/rtt_stats.h
@@ -25,6 +25,8 @@
 class QUIC_EXPORT_PRIVATE RttStats {
  public:
   RttStats();
+  RttStats(const RttStats&) = delete;
+  RttStats& operator=(const RttStats&) = delete;
 
   // Updates the RTT from an incoming ack which is received |send_delta| after
   // the packet is sent and the peer reports the ack being delayed |ack_delay|.
@@ -101,8 +103,6 @@
   QuicTime::Delta max_ack_delay_;
   // Whether to ignore the peer's max ack delay.
   bool ignore_max_ack_delay_;
-
-  DISALLOW_COPY_AND_ASSIGN(RttStats);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/congestion_control/tcp_cubic_sender_bytes.h b/net/third_party/quic/core/congestion_control/tcp_cubic_sender_bytes.h
index 9651db5..4d550ec 100644
--- a/net/third_party/quic/core/congestion_control/tcp_cubic_sender_bytes.h
+++ b/net/third_party/quic/core/congestion_control/tcp_cubic_sender_bytes.h
@@ -40,6 +40,8 @@
                       QuicPacketCount initial_tcp_congestion_window,
                       QuicPacketCount max_congestion_window,
                       QuicConnectionStats* stats);
+  TcpCubicSenderBytes(const TcpCubicSenderBytes&) = delete;
+  TcpCubicSenderBytes& operator=(const TcpCubicSenderBytes&) = delete;
   ~TcpCubicSenderBytes() override;
 
   // Start implementation of SendAlgorithmInterface.
@@ -164,8 +166,6 @@
 
   // The minimum window when exiting slow start with large reduction.
   QuicByteCount min_slow_start_exit_window_;
-
-  DISALLOW_COPY_AND_ASSIGN(TcpCubicSenderBytes);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aead_base_decrypter.h b/net/third_party/quic/core/crypto/aead_base_decrypter.h
index e8467e1..5650cf3 100644
--- a/net/third_party/quic/core/crypto/aead_base_decrypter.h
+++ b/net/third_party/quic/core/crypto/aead_base_decrypter.h
@@ -26,6 +26,8 @@
                     size_t auth_tag_size,
                     size_t nonce_size,
                     bool use_ietf_nonce_construction);
+  AeadBaseDecrypter(const AeadBaseDecrypter&) = delete;
+  AeadBaseDecrypter& operator=(const AeadBaseDecrypter&) = delete;
   ~AeadBaseDecrypter() override;
 
   // QuicDecrypter implementation
@@ -67,8 +69,6 @@
   unsigned char iv_[kMaxNonceSize];
 
   ScopedEVPAEADCtx ctx_;
-
-  DISALLOW_COPY_AND_ASSIGN(AeadBaseDecrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aead_base_encrypter.h b/net/third_party/quic/core/crypto/aead_base_encrypter.h
index 3bf975e..41ea7f4c 100644
--- a/net/third_party/quic/core/crypto/aead_base_encrypter.h
+++ b/net/third_party/quic/core/crypto/aead_base_encrypter.h
@@ -26,6 +26,8 @@
                     size_t auth_tag_size,
                     size_t nonce_size,
                     bool use_ietf_nonce_construction);
+  AeadBaseEncrypter(const AeadBaseEncrypter&) = delete;
+  AeadBaseEncrypter& operator=(const AeadBaseEncrypter&) = delete;
   ~AeadBaseEncrypter() override;
 
   // QuicEncrypter implementation
@@ -74,8 +76,6 @@
   unsigned char iv_[kMaxNonceSize];
 
   ScopedEVPAEADCtx ctx_;
-
-  DISALLOW_COPY_AND_ASSIGN(AeadBaseEncrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_128_gcm_12_decrypter.h b/net/third_party/quic/core/crypto/aes_128_gcm_12_decrypter.h
index 78f3b22..a9027695 100644
--- a/net/third_party/quic/core/crypto/aes_128_gcm_12_decrypter.h
+++ b/net/third_party/quic/core/crypto/aes_128_gcm_12_decrypter.h
@@ -27,12 +27,11 @@
   };
 
   Aes128Gcm12Decrypter();
+  Aes128Gcm12Decrypter(const Aes128Gcm12Decrypter&) = delete;
+  Aes128Gcm12Decrypter& operator=(const Aes128Gcm12Decrypter&) = delete;
   ~Aes128Gcm12Decrypter() override;
 
   uint32_t cipher_id() const override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes128Gcm12Decrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h b/net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h
index d24c21e..06939bb 100644
--- a/net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h
+++ b/net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h
@@ -25,10 +25,9 @@
   };
 
   Aes128Gcm12Encrypter();
+  Aes128Gcm12Encrypter(const Aes128Gcm12Encrypter&) = delete;
+  Aes128Gcm12Encrypter& operator=(const Aes128Gcm12Encrypter&) = delete;
   ~Aes128Gcm12Encrypter() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes128Gcm12Encrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_128_gcm_decrypter.h b/net/third_party/quic/core/crypto/aes_128_gcm_decrypter.h
index bca4fad..347af07 100644
--- a/net/third_party/quic/core/crypto/aes_128_gcm_decrypter.h
+++ b/net/third_party/quic/core/crypto/aes_128_gcm_decrypter.h
@@ -25,12 +25,11 @@
   };
 
   Aes128GcmDecrypter();
+  Aes128GcmDecrypter(const Aes128GcmDecrypter&) = delete;
+  Aes128GcmDecrypter& operator=(const Aes128GcmDecrypter&) = delete;
   ~Aes128GcmDecrypter() override;
 
   uint32_t cipher_id() const override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes128GcmDecrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_128_gcm_encrypter.h b/net/third_party/quic/core/crypto/aes_128_gcm_encrypter.h
index e78d6fd..8682db5 100644
--- a/net/third_party/quic/core/crypto/aes_128_gcm_encrypter.h
+++ b/net/third_party/quic/core/crypto/aes_128_gcm_encrypter.h
@@ -23,10 +23,9 @@
   };
 
   Aes128GcmEncrypter();
+  Aes128GcmEncrypter(const Aes128GcmEncrypter&) = delete;
+  Aes128GcmEncrypter& operator=(const Aes128GcmEncrypter&) = delete;
   ~Aes128GcmEncrypter() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes128GcmEncrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_256_gcm_decrypter.h b/net/third_party/quic/core/crypto/aes_256_gcm_decrypter.h
index 16618eb..1e5d094 100644
--- a/net/third_party/quic/core/crypto/aes_256_gcm_decrypter.h
+++ b/net/third_party/quic/core/crypto/aes_256_gcm_decrypter.h
@@ -25,12 +25,11 @@
   };
 
   Aes256GcmDecrypter();
+  Aes256GcmDecrypter(const Aes256GcmDecrypter&) = delete;
+  Aes256GcmDecrypter& operator=(const Aes256GcmDecrypter&) = delete;
   ~Aes256GcmDecrypter() override;
 
   uint32_t cipher_id() const override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes256GcmDecrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/aes_256_gcm_encrypter.h b/net/third_party/quic/core/crypto/aes_256_gcm_encrypter.h
index 5169b7b..6feb94d7 100644
--- a/net/third_party/quic/core/crypto/aes_256_gcm_encrypter.h
+++ b/net/third_party/quic/core/crypto/aes_256_gcm_encrypter.h
@@ -23,10 +23,9 @@
   };
 
   Aes256GcmEncrypter();
+  Aes256GcmEncrypter(const Aes256GcmEncrypter&) = delete;
+  Aes256GcmEncrypter& operator=(const Aes256GcmEncrypter&) = delete;
   ~Aes256GcmEncrypter() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(Aes256GcmEncrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/cert_compressor.h b/net/third_party/quic/core/crypto/cert_compressor.h
index daa6cc9..0669519 100644
--- a/net/third_party/quic/core/crypto/cert_compressor.h
+++ b/net/third_party/quic/core/crypto/cert_compressor.h
@@ -30,6 +30,8 @@
 //      methods and a small chunk of common substrings.
 class QUIC_EXPORT_PRIVATE CertCompressor {
  public:
+  CertCompressor() = delete;
+
   // CompressChain compresses the certificates in |certs| and returns a
   // compressed representation. |common_sets| contains the common certificate
   // sets known locally and |client_common_set_hashes| contains the hashes of
@@ -48,9 +50,6 @@
                               const std::vector<QuicString>& cached_certs,
                               const CommonCertSets* common_sets,
                               std::vector<QuicString>* out_certs);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(CertCompressor);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/chacha20_poly1305_decrypter.h b/net/third_party/quic/core/crypto/chacha20_poly1305_decrypter.h
index 9707b3ad..fc9b36a 100644
--- a/net/third_party/quic/core/crypto/chacha20_poly1305_decrypter.h
+++ b/net/third_party/quic/core/crypto/chacha20_poly1305_decrypter.h
@@ -27,12 +27,12 @@
   };
 
   ChaCha20Poly1305Decrypter();
+  ChaCha20Poly1305Decrypter(const ChaCha20Poly1305Decrypter&) = delete;
+  ChaCha20Poly1305Decrypter& operator=(const ChaCha20Poly1305Decrypter&) =
+      delete;
   ~ChaCha20Poly1305Decrypter() override;
 
   uint32_t cipher_id() const override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ChaCha20Poly1305Decrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/chacha20_poly1305_encrypter.h b/net/third_party/quic/core/crypto/chacha20_poly1305_encrypter.h
index a995631..d5d762ec 100644
--- a/net/third_party/quic/core/crypto/chacha20_poly1305_encrypter.h
+++ b/net/third_party/quic/core/crypto/chacha20_poly1305_encrypter.h
@@ -25,10 +25,10 @@
   };
 
   ChaCha20Poly1305Encrypter();
+  ChaCha20Poly1305Encrypter(const ChaCha20Poly1305Encrypter&) = delete;
+  ChaCha20Poly1305Encrypter& operator=(const ChaCha20Poly1305Encrypter&) =
+      delete;
   ~ChaCha20Poly1305Encrypter() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ChaCha20Poly1305Encrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/chacha20_poly1305_tls_decrypter.h b/net/third_party/quic/core/crypto/chacha20_poly1305_tls_decrypter.h
index b6b477d8..07feb50 100644
--- a/net/third_party/quic/core/crypto/chacha20_poly1305_tls_decrypter.h
+++ b/net/third_party/quic/core/crypto/chacha20_poly1305_tls_decrypter.h
@@ -26,12 +26,12 @@
   };
 
   ChaCha20Poly1305TlsDecrypter();
+  ChaCha20Poly1305TlsDecrypter(const ChaCha20Poly1305TlsDecrypter&) = delete;
+  ChaCha20Poly1305TlsDecrypter& operator=(const ChaCha20Poly1305TlsDecrypter&) =
+      delete;
   ~ChaCha20Poly1305TlsDecrypter() override;
 
   uint32_t cipher_id() const override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ChaCha20Poly1305TlsDecrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/chacha20_poly1305_tls_encrypter.h b/net/third_party/quic/core/crypto/chacha20_poly1305_tls_encrypter.h
index 930956c..2db7c9b 100644
--- a/net/third_party/quic/core/crypto/chacha20_poly1305_tls_encrypter.h
+++ b/net/third_party/quic/core/crypto/chacha20_poly1305_tls_encrypter.h
@@ -24,10 +24,10 @@
   };
 
   ChaCha20Poly1305TlsEncrypter();
+  ChaCha20Poly1305TlsEncrypter(const ChaCha20Poly1305TlsEncrypter&) = delete;
+  ChaCha20Poly1305TlsEncrypter& operator=(const ChaCha20Poly1305TlsEncrypter&) =
+      delete;
   ~ChaCha20Poly1305TlsEncrypter() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ChaCha20Poly1305TlsEncrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/channel_id.h b/net/third_party/quic/core/crypto/channel_id.h
index 99a5ee3..2a91474 100644
--- a/net/third_party/quic/core/crypto/channel_id.h
+++ b/net/third_party/quic/core/crypto/channel_id.h
@@ -66,6 +66,8 @@
 // ChannelIDVerifier verifies ChannelID signatures.
 class QUIC_EXPORT_PRIVATE ChannelIDVerifier {
  public:
+  ChannelIDVerifier() = delete;
+
   // kContextStr is prepended to the data to be signed in order to ensure that
   // a ChannelID signature cannot be used in a different context. (The
   // terminating NUL byte is inclued.)
@@ -89,9 +91,6 @@
                         QuicStringPiece signed_data,
                         QuicStringPiece signature,
                         bool is_channel_id_signature);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ChannelIDVerifier);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/common_cert_set.cc b/net/third_party/quic/core/crypto/common_cert_set.cc
index 4151d39..e81d111b 100644
--- a/net/third_party/quic/core/crypto/common_cert_set.cc
+++ b/net/third_party/quic/core/crypto/common_cert_set.cc
@@ -148,10 +148,11 @@
 
  private:
   CommonCertSetsQUIC() {}
+  CommonCertSetsQUIC(const CommonCertSetsQUIC&) = delete;
+  CommonCertSetsQUIC& operator=(const CommonCertSetsQUIC&) = delete;
   ~CommonCertSetsQUIC() override {}
 
   friend QuicSingletonFriend<CommonCertSetsQUIC>;
-  DISALLOW_COPY_AND_ASSIGN(CommonCertSetsQUIC);
 };
 
 }  // anonymous namespace
diff --git a/net/third_party/quic/core/crypto/crypto_handshake.h b/net/third_party/quic/core/crypto/crypto_handshake.h
index 129b733..77d8599 100644
--- a/net/third_party/quic/core/crypto/crypto_handshake.h
+++ b/net/third_party/quic/core/crypto/crypto_handshake.h
@@ -168,6 +168,8 @@
   static const char kForwardSecureLabel[];
 
   QuicCryptoConfig();
+  QuicCryptoConfig(const QuicCryptoConfig&) = delete;
+  QuicCryptoConfig& operator=(const QuicCryptoConfig&) = delete;
   ~QuicCryptoConfig();
 
   // Key exchange methods. The following two members' values correspond by
@@ -181,9 +183,6 @@
   QuicTagVector tb_key_params;
 
   const CommonCertSets* common_cert_sets;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoConfig);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/crypto_secret_boxer.h b/net/third_party/quic/core/crypto/crypto_secret_boxer.h
index 33beb208c..8723045f 100644
--- a/net/third_party/quic/core/crypto/crypto_secret_boxer.h
+++ b/net/third_party/quic/core/crypto/crypto_secret_boxer.h
@@ -25,6 +25,8 @@
 class QUIC_EXPORT_PRIVATE CryptoSecretBoxer {
  public:
   CryptoSecretBoxer();
+  CryptoSecretBoxer(const CryptoSecretBoxer&) = delete;
+  CryptoSecretBoxer& operator=(const CryptoSecretBoxer&) = delete;
   ~CryptoSecretBoxer();
 
   // GetKeySize returns the number of bytes in a key.
@@ -59,8 +61,6 @@
   // state_ is an opaque pointer to whatever additional state the concrete
   // implementation of CryptoSecretBoxer requires.
   std::unique_ptr<State> state_ GUARDED_BY(lock_);
-
-  DISALLOW_COPY_AND_ASSIGN(CryptoSecretBoxer);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/crypto_utils.h b/net/third_party/quic/core/crypto/crypto_utils.h
index f4930ec7..59294fd 100644
--- a/net/third_party/quic/core/crypto/crypto_utils.h
+++ b/net/third_party/quic/core/crypto/crypto_utils.h
@@ -27,6 +27,8 @@
 
 class QUIC_EXPORT_PRIVATE CryptoUtils {
  public:
+  CryptoUtils() = delete;
+
   // Diversification is a utility class that's used to act like a union type.
   // Values can be created by calling the functions like |NoDiversification|,
   // below.
@@ -218,9 +220,6 @@
   static void HashHandshakeMessage(const CryptoHandshakeMessage& message,
                                    QuicString* output,
                                    Perspective perspective);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(CryptoUtils);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/null_decrypter.h b/net/third_party/quic/core/crypto/null_decrypter.h
index 97bc337e..49d22fe 100644
--- a/net/third_party/quic/core/crypto/null_decrypter.h
+++ b/net/third_party/quic/core/crypto/null_decrypter.h
@@ -25,6 +25,8 @@
 class QUIC_EXPORT_PRIVATE NullDecrypter : public QuicDecrypter {
  public:
   explicit NullDecrypter(Perspective perspective);
+  NullDecrypter(const NullDecrypter&) = delete;
+  NullDecrypter& operator=(const NullDecrypter&) = delete;
   ~NullDecrypter() override {}
 
   // QuicDecrypter implementation
@@ -54,8 +56,6 @@
                           QuicStringPiece data2) const;
 
   Perspective perspective_;
-
-  DISALLOW_COPY_AND_ASSIGN(NullDecrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/null_encrypter.h b/net/third_party/quic/core/crypto/null_encrypter.h
index 1ad9e25..a549d466 100644
--- a/net/third_party/quic/core/crypto/null_encrypter.h
+++ b/net/third_party/quic/core/crypto/null_encrypter.h
@@ -22,6 +22,8 @@
 class QUIC_EXPORT_PRIVATE NullEncrypter : public QuicEncrypter {
  public:
   explicit NullEncrypter(Perspective perspective);
+  NullEncrypter(const NullEncrypter&) = delete;
+  NullEncrypter& operator=(const NullEncrypter&) = delete;
   ~NullEncrypter() override {}
 
   // QuicEncrypter implementation
@@ -47,8 +49,6 @@
   size_t GetHashLength() const;
 
   Perspective perspective_;
-
-  DISALLOW_COPY_AND_ASSIGN(NullEncrypter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/p256_key_exchange.h b/net/third_party/quic/core/crypto/p256_key_exchange.h
index ae01620..c7496829 100644
--- a/net/third_party/quic/core/crypto/p256_key_exchange.h
+++ b/net/third_party/quic/core/crypto/p256_key_exchange.h
@@ -54,12 +54,12 @@
   // |kUncompressedP256PointBytes| bytes.
   P256KeyExchange(bssl::UniquePtr<EC_KEY> private_key,
                   const uint8_t* public_key);
+  P256KeyExchange(const P256KeyExchange&) = delete;
+  P256KeyExchange& operator=(const P256KeyExchange&) = delete;
 
   bssl::UniquePtr<EC_KEY> private_key_;
   // The public key stored as an uncompressed P-256 point.
   uint8_t public_key_[kUncompressedP256PointBytes];
-
-  DISALLOW_COPY_AND_ASSIGN(P256KeyExchange);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/proof_source.h b/net/third_party/quic/core/crypto/proof_source.h
index 3a683dc1..938a4c6 100644
--- a/net/third_party/quic/core/crypto/proof_source.h
+++ b/net/third_party/quic/core/crypto/proof_source.h
@@ -26,14 +26,13 @@
   // certificates.
   struct QUIC_EXPORT_PRIVATE Chain : public QuicReferenceCounted {
     explicit Chain(const std::vector<QuicString>& certs);
+    Chain(const Chain&) = delete;
+    Chain& operator=(const Chain&) = delete;
 
     const std::vector<QuicString> certs;
 
    protected:
     ~Chain() override;
-
-   private:
-    DISALLOW_COPY_AND_ASSIGN(Chain);
   };
 
   // Details is an abstract class which acts as a container for any
diff --git a/net/third_party/quic/core/crypto/quic_crypto_client_config.h b/net/third_party/quic/core/crypto/quic_crypto_client_config.h
index 0f4cb3d..6882d2a 100644
--- a/net/third_party/quic/core/crypto/quic_crypto_client_config.h
+++ b/net/third_party/quic/core/crypto/quic_crypto_client_config.h
@@ -58,6 +58,8 @@
     };
 
     CachedState();
+    CachedState(const CachedState&) = delete;
+    CachedState& operator=(const CachedState&) = delete;
     ~CachedState();
 
     // IsComplete returns true if this object contains enough information to
@@ -192,8 +194,6 @@
     // nonces and connection_ids together in one queue.
     QuicQueue<QuicConnectionId> server_designated_connection_ids_;
     QuicQueue<QuicString> server_nonces_;
-
-    DISALLOW_COPY_AND_ASSIGN(CachedState);
   };
 
   // Used to filter server ids for partial config deletion.
@@ -207,6 +207,8 @@
 
   QuicCryptoClientConfig(std::unique_ptr<ProofVerifier> proof_verifier,
                          bssl::UniquePtr<SSL_CTX> ssl_ctx);
+  QuicCryptoClientConfig(const QuicCryptoClientConfig&) = delete;
+  QuicCryptoClientConfig& operator=(const QuicCryptoClientConfig&) = delete;
   ~QuicCryptoClientConfig();
 
   // LookupOrCreate returns a CachedState for the given |server_id|. If no such
@@ -405,8 +407,6 @@
   // If non-empty, the client will operate in the pre-shared key mode by
   // incorporating |pre_shared_key_| into the key schedule.
   QuicString pre_shared_key_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoClientConfig);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/crypto/quic_crypto_server_config.cc b/net/third_party/quic/core/crypto/quic_crypto_server_config.cc
index 540dd02..b3a0473 100644
--- a/net/third_party/quic/core/crypto/quic_crypto_server_config.cc
+++ b/net/third_party/quic/core/crypto/quic_crypto_server_config.cc
@@ -77,6 +77,9 @@
           result,
       std::unique_ptr<ValidateClientHelloResultCallback>* done_cb)
       : result_(std::move(result)), done_cb_(done_cb) {}
+  ValidateClientHelloHelper(const ValidateClientHelloHelper&) = delete;
+  ValidateClientHelloHelper& operator=(const ValidateClientHelloHelper&) =
+      delete;
 
   ~ValidateClientHelloHelper() {
     QUIC_BUG_IF(done_cb_ != nullptr)
@@ -102,8 +105,6 @@
   QuicReferenceCountedPointer<ValidateClientHelloResultCallback::Result>
       result_;
   std::unique_ptr<ValidateClientHelloResultCallback>* done_cb_;
-
-  DISALLOW_COPY_AND_ASSIGN(ValidateClientHelloHelper);
 };
 
 // static
diff --git a/net/third_party/quic/core/crypto/quic_crypto_server_config.h b/net/third_party/quic/core/crypto/quic_crypto_server_config.h
index d8f81a63..56bee07 100644
--- a/net/third_party/quic/core/crypto/quic_crypto_server_config.h
+++ b/net/third_party/quic/core/crypto/quic_crypto_server_config.h
@@ -73,11 +73,11 @@
 class PrimaryConfigChangedCallback {
  public:
   PrimaryConfigChangedCallback();
+  PrimaryConfigChangedCallback(const PrimaryConfigChangedCallback&) = delete;
+  PrimaryConfigChangedCallback& operator=(const PrimaryConfigChangedCallback&) =
+      delete;
   virtual ~PrimaryConfigChangedCallback();
   virtual void Run(const QuicString& scid) = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(PrimaryConfigChangedCallback);
 };
 
 // Callback used to accept the result of the |client_hello| validation step.
@@ -103,27 +103,29 @@
   };
 
   ValidateClientHelloResultCallback();
+  ValidateClientHelloResultCallback(const ValidateClientHelloResultCallback&) =
+      delete;
+  ValidateClientHelloResultCallback& operator=(
+      const ValidateClientHelloResultCallback&) = delete;
   virtual ~ValidateClientHelloResultCallback();
   virtual void Run(QuicReferenceCountedPointer<Result> result,
                    std::unique_ptr<ProofSource::Details> details) = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ValidateClientHelloResultCallback);
 };
 
 // Callback used to accept the result of the ProcessClientHello method.
 class QUIC_EXPORT_PRIVATE ProcessClientHelloResultCallback {
  public:
   ProcessClientHelloResultCallback();
+  ProcessClientHelloResultCallback(const ProcessClientHelloResultCallback&) =
+      delete;
+  ProcessClientHelloResultCallback& operator=(
+      const ProcessClientHelloResultCallback&) = delete;
   virtual ~ProcessClientHelloResultCallback();
   virtual void Run(QuicErrorCode error,
                    const QuicString& error_details,
                    std::unique_ptr<CryptoHandshakeMessage> message,
                    std::unique_ptr<DiversificationNonce> diversification_nonce,
                    std::unique_ptr<ProofSource::Details> details) = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(ProcessClientHelloResultCallback);
 };
 
 // Callback used to receive the results of a call to
@@ -132,10 +134,11 @@
  public:
   BuildServerConfigUpdateMessageResultCallback() = default;
   virtual ~BuildServerConfigUpdateMessageResultCallback() {}
+  BuildServerConfigUpdateMessageResultCallback(
+      const BuildServerConfigUpdateMessageResultCallback&) = delete;
+  BuildServerConfigUpdateMessageResultCallback& operator=(
+      const BuildServerConfigUpdateMessageResultCallback&) = delete;
   virtual void Run(bool ok, const CryptoHandshakeMessage& message) = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(BuildServerConfigUpdateMessageResultCallback);
 };
 
 // Object that is interested in built rejections (which include REJ, SREJ and
@@ -144,12 +147,11 @@
  public:
   RejectionObserver() = default;
   virtual ~RejectionObserver() {}
+  RejectionObserver(const RejectionObserver&) = delete;
+  RejectionObserver& operator=(const RejectionObserver&) = delete;
   // Called after a rejection is built.
   virtual void OnRejectionBuilt(const std::vector<uint32_t>& reasons,
                                 CryptoHandshakeMessage* out) const = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(RejectionObserver);
 };
 
 // QuicCryptoServerConfig contains the crypto configuration of a QUIC server.
@@ -200,6 +202,8 @@
                          QuicRandom* server_nonce_entropy,
                          std::unique_ptr<ProofSource> proof_source,
                          bssl::UniquePtr<SSL_CTX> ssl_ctx);
+  QuicCryptoServerConfig(const QuicCryptoServerConfig&) = delete;
+  QuicCryptoServerConfig& operator=(const QuicCryptoServerConfig&) = delete;
   ~QuicCryptoServerConfig();
 
   // TESTING is a magic parameter for passing to the constructor in tests.
@@ -413,6 +417,8 @@
                                      public QuicReferenceCounted {
    public:
     Config();
+    Config(const Config&) = delete;
+    Config& operator=(const Config&) = delete;
 
     // TODO(rtenneti): since this is a class, we should probably do
     // getters/setters here.
@@ -467,8 +473,6 @@
 
    private:
     ~Config() override;
-
-    DISALLOW_COPY_AND_ASSIGN(Config);
   };
 
   typedef std::map<ServerConfigID, QuicReferenceCountedPointer<Config>>
@@ -773,8 +777,6 @@
   // If non-empty, the server will operate in the pre-shared key mode by
   // incorporating |pre_shared_key_| into the key schedule.
   QuicString pre_shared_key_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoServerConfig);
 };
 
 struct QUIC_EXPORT_PRIVATE QuicSignedServerConfig
diff --git a/net/third_party/quic/core/crypto/quic_random.cc b/net/third_party/quic/core/crypto/quic_random.cc
index a7a95691..3aa7bc9 100644
--- a/net/third_party/quic/core/crypto/quic_random.cc
+++ b/net/third_party/quic/core/crypto/quic_random.cc
@@ -23,10 +23,11 @@
 
  private:
   DefaultRandom() {}
+  DefaultRandom(const DefaultRandom&) = delete;
+  DefaultRandom& operator=(const DefaultRandom&) = delete;
   ~DefaultRandom() override {}
 
   friend QuicSingletonFriend<DefaultRandom>;
-  DISALLOW_COPY_AND_ASSIGN(DefaultRandom);
 };
 
 DefaultRandom* DefaultRandom::GetInstance() {
diff --git a/net/third_party/quic/core/crypto/scoped_evp_aead_ctx.h b/net/third_party/quic/core/crypto/scoped_evp_aead_ctx.h
index 8de9edd..d0855e8 100644
--- a/net/third_party/quic/core/crypto/scoped_evp_aead_ctx.h
+++ b/net/third_party/quic/core/crypto/scoped_evp_aead_ctx.h
@@ -15,14 +15,14 @@
 class ScopedEVPAEADCtx {
  public:
   ScopedEVPAEADCtx();
+  ScopedEVPAEADCtx(const ScopedEVPAEADCtx&) = delete;
+  ScopedEVPAEADCtx& operator=(const ScopedEVPAEADCtx&) = delete;
   ~ScopedEVPAEADCtx();
 
   EVP_AEAD_CTX* get();
 
  private:
   EVP_AEAD_CTX ctx_;
-
-  DISALLOW_COPY_AND_ASSIGN(ScopedEVPAEADCtx);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/frames/quic_stream_frame.h b/net/third_party/quic/core/frames/quic_stream_frame.h
index 0e585cd..cf6d981 100644
--- a/net/third_party/quic/core/frames/quic_stream_frame.h
+++ b/net/third_party/quic/core/frames/quic_stream_frame.h
@@ -41,9 +41,8 @@
                   QuicStreamOffset offset,
                   const char* data_buffer,
                   QuicPacketLength data_length);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamFrame);
+  QuicStreamFrame(const QuicStreamFrame&) = delete;
+  QuicStreamFrame& operator=(const QuicStreamFrame&) = delete;
 };
 static_assert(sizeof(QuicStreamFrame) <= 64,
               "Keep the QuicStreamFrame size to a cacheline.");
diff --git a/net/third_party/quic/core/end_to_end_test.cc b/net/third_party/quic/core/http/end_to_end_test.cc
similarity index 99%
rename from net/third_party/quic/core/end_to_end_test.cc
rename to net/third_party/quic/core/http/end_to_end_test.cc
index bb39a7b..d38e060 100644
--- a/net/third_party/quic/core/end_to_end_test.cc
+++ b/net/third_party/quic/core/http/end_to_end_test.cc
@@ -22,6 +22,8 @@
 #include "net/test/test_with_scoped_task_environment.h"
 #include "net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h"
 #include "net/third_party/quic/core/crypto/null_encrypter.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_epoll_connection_helper.h"
 #include "net/third_party/quic/core/quic_framer.h"
 #include "net/third_party/quic/core/quic_packet_creator.h"
@@ -29,8 +31,6 @@
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
 #include "net/third_party/quic/core/quic_session.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_utils.h"
 #include "net/third_party/quic/platform/api/quic_expect_bug.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
diff --git a/net/third_party/quic/core/quic_client_promised_info.cc b/net/third_party/quic/core/http/quic_client_promised_info.cc
similarity index 97%
rename from net/third_party/quic/core/quic_client_promised_info.cc
rename to net/third_party/quic/core/http/quic_client_promised_info.cc
index ff835e2..b9ecc72 100644
--- a/net/third_party/quic/core/quic_client_promised_info.cc
+++ b/net/third_party/quic/core/http/quic_client_promised_info.cc
@@ -2,11 +2,11 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 
 #include <utility>
 
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
diff --git a/net/third_party/quic/core/quic_client_promised_info.h b/net/third_party/quic/core/http/quic_client_promised_info.h
similarity index 86%
rename from net/third_party/quic/core/quic_client_promised_info.h
rename to net/third_party/quic/core/http/quic_client_promised_info.h
index 41e2259..c56eb6e 100644
--- a/net/third_party/quic/core/quic_client_promised_info.h
+++ b/net/third_party/quic/core/http/quic_client_promised_info.h
@@ -2,16 +2,16 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PROMISED_INFO_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PROMISED_INFO_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PROMISED_INFO_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PROMISED_INFO_H_
 
 #include <cstddef>
 
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_alarm.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 #include "net/third_party/spdy/core/spdy_framer.h"
@@ -33,6 +33,8 @@
   QuicClientPromisedInfo(QuicSpdyClientSessionBase* session,
                          QuicStreamId id,
                          QuicString url);
+  QuicClientPromisedInfo(const QuicClientPromisedInfo&) = delete;
+  QuicClientPromisedInfo& operator=(const QuicClientPromisedInfo&) = delete;
   virtual ~QuicClientPromisedInfo();
 
   void Init();
@@ -105,10 +107,8 @@
   // The promise will commit suicide eventually if it is not claimed by a GET
   // first.
   std::unique_ptr<QuicAlarm> cleanup_alarm_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicClientPromisedInfo);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PROMISED_INFO_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PROMISED_INFO_H_
diff --git a/net/third_party/quic/core/quic_client_promised_info_test.cc b/net/third_party/quic/core/http/quic_client_promised_info_test.cc
similarity index 96%
rename from net/third_party/quic/core/quic_client_promised_info_test.cc
rename to net/third_party/quic/core/http/quic_client_promised_info_test.cc
index 44cd5d6..619b08af 100644
--- a/net/third_party/quic/core/quic_client_promised_info_test.cc
+++ b/net/third_party/quic/core/http/quic_client_promised_info_test.cc
@@ -2,13 +2,13 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 
 #include <memory>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
@@ -41,6 +41,9 @@
         crypto_config_(crypto_test_utils::ProofVerifierForTesting(),
                        TlsClientHandshaker::CreateSslCtx()),
         authorized_(true) {}
+  MockQuicSpdyClientSession(const MockQuicSpdyClientSession&) = delete;
+  MockQuicSpdyClientSession& operator=(const MockQuicSpdyClientSession&) =
+      delete;
   ~MockQuicSpdyClientSession() override {}
 
   bool IsAuthorized(const QuicString& authority) override {
@@ -55,8 +58,6 @@
   QuicCryptoClientConfig crypto_config_;
 
   bool authorized_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSpdyClientSession);
 };
 
 class QuicClientPromisedInfoTest : public QuicTest {
diff --git a/net/third_party/quic/core/quic_client_push_promise_index.cc b/net/third_party/quic/core/http/quic_client_push_promise_index.cc
similarity index 86%
rename from net/third_party/quic/core/quic_client_push_promise_index.cc
rename to net/third_party/quic/core/http/quic_client_push_promise_index.cc
index 5b29eaa0..4549123 100644
--- a/net/third_party/quic/core/quic_client_push_promise_index.cc
+++ b/net/third_party/quic/core/http/quic_client_push_promise_index.cc
@@ -2,10 +2,10 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
 
-#include "net/third_party/quic/core/quic_client_promised_info.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 
 using spdy::SpdyHeaderBlock;
diff --git a/net/third_party/quic/core/quic_client_push_promise_index.h b/net/third_party/quic/core/http/quic_client_push_promise_index.h
similarity index 86%
rename from net/third_party/quic/core/quic_client_push_promise_index.h
rename to net/third_party/quic/core/http/quic_client_push_promise_index.h
index fea5178d..bc62419 100644
--- a/net/third_party/quic/core/quic_client_push_promise_index.h
+++ b/net/third_party/quic/core/http/quic_client_push_promise_index.h
@@ -2,10 +2,10 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
 
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
 #include "net/third_party/quic/core/quic_types.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
@@ -53,13 +53,15 @@
 
    protected:
     TryHandle() {}
+    TryHandle(const TryHandle&) = delete;
+    TryHandle& operator=(const TryHandle&) = delete;
     ~TryHandle();
-
-   private:
-    DISALLOW_COPY_AND_ASSIGN(TryHandle);
   };
 
   QuicClientPushPromiseIndex();
+  QuicClientPushPromiseIndex(const QuicClientPushPromiseIndex&) = delete;
+  QuicClientPushPromiseIndex& operator=(const QuicClientPushPromiseIndex&) =
+      delete;
   virtual ~QuicClientPushPromiseIndex();
 
   // Called by client code, used to enforce affinity between requests
@@ -89,10 +91,8 @@
 
  private:
   QuicPromisedByUrlMap promised_by_url_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicClientPushPromiseIndex);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_CLIENT_PUSH_PROMISE_INDEX_H_
diff --git a/net/third_party/quic/core/quic_client_push_promise_index_test.cc b/net/third_party/quic/core/http/quic_client_push_promise_index_test.cc
similarity index 91%
rename from net/third_party/quic/core/quic_client_push_promise_index_test.cc
rename to net/third_party/quic/core/http/quic_client_push_promise_index_test.cc
index 9c7c4d9e..27c3e33 100644
--- a/net/third_party/quic/core/quic_client_push_promise_index_test.cc
+++ b/net/third_party/quic/core/http/quic_client_push_promise_index_test.cc
@@ -2,10 +2,10 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
 
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 #include "net/third_party/quic/platform/api/quic_test.h"
@@ -34,14 +34,15 @@
                               push_promise_index),
         crypto_config_(crypto_test_utils::ProofVerifierForTesting(),
                        TlsClientHandshaker::CreateSslCtx()) {}
+  MockQuicSpdyClientSession(const MockQuicSpdyClientSession&) = delete;
+  MockQuicSpdyClientSession& operator=(const MockQuicSpdyClientSession&) =
+      delete;
   ~MockQuicSpdyClientSession() override {}
 
   MOCK_METHOD1(CloseStream, void(QuicStreamId stream_id));
 
  private:
   QuicCryptoClientConfig crypto_config_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSpdyClientSession);
 };
 
 class QuicClientPushPromiseIndexTest : public QuicTest {
diff --git a/net/third_party/quic/core/quic_header_list.cc b/net/third_party/quic/core/http/quic_header_list.cc
similarity index 97%
rename from net/third_party/quic/core/quic_header_list.cc
rename to net/third_party/quic/core/http/quic_header_list.cc
index 87e2a92..e0589680 100644
--- a/net/third_party/quic/core/quic_header_list.cc
+++ b/net/third_party/quic/core/http/quic_header_list.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 "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
 
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
diff --git a/net/third_party/quic/core/quic_header_list.h b/net/third_party/quic/core/http/quic_header_list.h
similarity index 94%
rename from net/third_party/quic/core/quic_header_list.h
rename to net/third_party/quic/core/http/quic_header_list.h
index 88d5e2a..016b484 100644
--- a/net/third_party/quic/core/quic_header_list.h
+++ b/net/third_party/quic/core/http/quic_header_list.h
@@ -2,8 +2,8 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADER_LIST_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADER_LIST_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADER_LIST_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADER_LIST_H_
 
 #include <algorithm>
 #include <functional>
@@ -85,4 +85,4 @@
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADER_LIST_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADER_LIST_H_
diff --git a/net/third_party/quic/core/quic_header_list_test.cc b/net/third_party/quic/core/http/quic_header_list_test.cc
similarity index 97%
rename from net/third_party/quic/core/quic_header_list_test.cc
rename to net/third_party/quic/core/http/quic_header_list_test.cc
index 9928c7a..af745ac 100644
--- a/net/third_party/quic/core/quic_header_list_test.cc
+++ b/net/third_party/quic/core/http/quic_header_list_test.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 "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
 
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
diff --git a/net/third_party/quic/core/quic_headers_stream.cc b/net/third_party/quic/core/http/quic_headers_stream.cc
similarity index 97%
rename from net/third_party/quic/core/quic_headers_stream.cc
rename to net/third_party/quic/core/http/quic_headers_stream.cc
index 115d9a7..b0c914e 100644
--- a/net/third_party/quic/core/quic_headers_stream.cc
+++ b/net/third_party/quic/core/http/quic_headers_stream.cc
@@ -2,9 +2,9 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_headers_stream.h"
+#include "net/third_party/quic/core/http/quic_headers_stream.h"
 
-#include "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 #include "net/third_party/quic/platform/api/quic_arraysize.h"
 #include "net/third_party/quic/platform/api/quic_flag_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
diff --git a/net/third_party/quic/core/quic_headers_stream.h b/net/third_party/quic/core/http/quic_headers_stream.h
similarity index 89%
rename from net/third_party/quic/core/quic_headers_stream.h
rename to net/third_party/quic/core/http/quic_headers_stream.h
index 0c0187d1..3221522 100644
--- a/net/third_party/quic/core/quic_headers_stream.h
+++ b/net/third_party/quic/core/http/quic_headers_stream.h
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADERS_STREAM_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADERS_STREAM_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADERS_STREAM_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADERS_STREAM_H_
 
 #include <cstddef>
 #include <memory>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_stream.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
@@ -30,6 +30,8 @@
 class QUIC_EXPORT_PRIVATE QuicHeadersStream : public QuicStream {
  public:
   explicit QuicHeadersStream(QuicSpdySession* session);
+  QuicHeadersStream(const QuicHeadersStream&) = delete;
+  QuicHeadersStream& operator=(const QuicHeadersStream&) = delete;
   ~QuicHeadersStream() override;
 
   // QuicStream implementation
@@ -87,10 +89,8 @@
 
   // Headers that have not been fully acked.
   QuicDeque<CompressedHeaderInfo> unacked_headers_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicHeadersStream);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_HEADERS_STREAM_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_HEADERS_STREAM_H_
diff --git a/net/third_party/quic/core/quic_headers_stream_test.cc b/net/third_party/quic/core/http/quic_headers_stream_test.cc
similarity index 98%
rename from net/third_party/quic/core/quic_headers_stream_test.cc
rename to net/third_party/quic/core/http/quic_headers_stream_test.cc
index 8016e2a..441403e 100644
--- a/net/third_party/quic/core/quic_headers_stream_test.cc
+++ b/net/third_party/quic/core/http/quic_headers_stream_test.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 "net/third_party/quic/core/quic_headers_stream.h"
+#include "net/third_party/quic/core/http/quic_headers_stream.h"
 
 #include <cstdint>
 #include <ostream>
@@ -10,9 +10,9 @@
 #include <utility>
 #include <vector>
 
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_data_writer.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_expect_bug.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
@@ -75,11 +75,11 @@
 class MockQuicHpackDebugVisitor : public QuicHpackDebugVisitor {
  public:
   MockQuicHpackDebugVisitor() : QuicHpackDebugVisitor() {}
+  MockQuicHpackDebugVisitor(const MockQuicHpackDebugVisitor&) = delete;
+  MockQuicHpackDebugVisitor& operator=(const MockQuicHpackDebugVisitor&) =
+      delete;
 
   MOCK_METHOD1(OnUseEntry, void(QuicTime::Delta elapsed));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicHpackDebugVisitor);
 };
 
 namespace {
diff --git a/net/third_party/quic/core/quic_server_session_base.cc b/net/third_party/quic/core/http/quic_server_session_base.cc
similarity index 98%
rename from net/third_party/quic/core/quic_server_session_base.cc
rename to net/third_party/quic/core/http/quic_server_session_base.cc
index e9838213..432fafe 100644
--- a/net/third_party/quic/core/quic_server_session_base.cc
+++ b/net/third_party/quic/core/http/quic_server_session_base.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 "net/third_party/quic/core/quic_server_session_base.h"
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
 
 #include "net/third_party/quic/core/proto/cached_network_parameters.pb.h"
 #include "net/third_party/quic/core/quic_connection.h"
diff --git a/net/third_party/quic/core/quic_server_session_base.h b/net/third_party/quic/core/http/quic_server_session_base.h
similarity index 91%
rename from net/third_party/quic/core/quic_server_session_base.h
rename to net/third_party/quic/core/http/quic_server_session_base.h
index dae10a4..1ff4691 100644
--- a/net/third_party/quic/core/quic_server_session_base.h
+++ b/net/third_party/quic/core/http/quic_server_session_base.h
@@ -4,8 +4,8 @@
 //
 // A server specific QuicSession subclass.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SERVER_SESSION_BASE_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SERVER_SESSION_BASE_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SERVER_SESSION_BASE_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SERVER_SESSION_BASE_H_
 
 #include <cstdint>
 #include <memory>
@@ -14,9 +14,9 @@
 
 #include "base/macros.h"
 #include "net/third_party/quic/core/crypto/quic_compressed_certs_cache.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 #include "net/third_party/quic/core/quic_crypto_server_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_session.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 
@@ -41,6 +41,8 @@
                         QuicCryptoServerStream::Helper* helper,
                         const QuicCryptoServerConfig* crypto_config,
                         QuicCompressedCertsCache* compressed_certs_cache);
+  QuicServerSessionBase(const QuicServerSessionBase&) = delete;
+  QuicServerSessionBase& operator=(const QuicServerSessionBase&) = delete;
 
   // Override the base class to cancel any ongoing asychronous crypto.
   void OnConnectionClosed(QuicErrorCode error,
@@ -129,10 +131,8 @@
   // should go away once we fix http://b//27897982
   int32_t BandwidthToCachedParameterBytesPerSecond(
       const QuicBandwidth& bandwidth);
-
-  DISALLOW_COPY_AND_ASSIGN(QuicServerSessionBase);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SERVER_SESSION_BASE_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SERVER_SESSION_BASE_H_
diff --git a/net/third_party/quic/core/quic_server_session_base_test.cc b/net/third_party/quic/core/http/quic_server_session_base_test.cc
similarity index 98%
rename from net/third_party/quic/core/quic_server_session_base_test.cc
rename to net/third_party/quic/core/http/quic_server_session_base_test.cc
index 8ef5d3a..3bbef179 100644
--- a/net/third_party/quic/core/quic_server_session_base_test.cc
+++ b/net/third_party/quic/core/http/quic_server_session_base_test.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 "net/third_party/quic/core/quic_server_session_base.h"
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
 
 #include <cstdint>
 #include <memory>
@@ -32,7 +32,6 @@
 #include "net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h"
 #include "net/third_party/quic/test_tools/quic_session_peer.h"
 #include "net/third_party/quic/test_tools/quic_spdy_session_peer.h"
-#include "net/third_party/quic/test_tools/quic_spdy_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h"
 #include "net/third_party/quic/test_tools/quic_test_utils.h"
@@ -383,13 +382,13 @@
             GetQuicReloadableFlag(enable_quic_stateless_reject_support),
             session,
             helper) {}
+  MockQuicCryptoServerStream(const MockQuicCryptoServerStream&) = delete;
+  MockQuicCryptoServerStream& operator=(const MockQuicCryptoServerStream&) =
+      delete;
   ~MockQuicCryptoServerStream() override {}
 
   MOCK_METHOD1(SendServerConfigUpdate,
                void(const CachedNetworkParameters* cached_network_parameters));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicCryptoServerStream);
 };
 
 TEST_P(QuicServerSessionBaseTest, BandwidthEstimates) {
diff --git a/net/third_party/quic/core/quic_spdy_client_session.cc b/net/third_party/quic/core/http/quic_spdy_client_session.cc
similarity index 95%
rename from net/third_party/quic/core/quic_spdy_client_session.cc
rename to net/third_party/quic/core/http/quic_spdy_client_session.cc
index a6c11a91..d998cf2 100644
--- a/net/third_party/quic/core/quic_spdy_client_session.cc
+++ b/net/third_party/quic/core/http/quic_spdy_client_session.cc
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
 
 #include "net/log/net_log_with_source.h"
 #include "net/quic/chromium/crypto/proof_verifier_chromium.h"
 #include "net/third_party/quic/core/crypto/crypto_protocol.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/core/quic_spdy_client_session.h b/net/third_party/quic/core/http/quic_spdy_client_session.h
similarity index 87%
rename from net/third_party/quic/core/quic_spdy_client_session.h
rename to net/third_party/quic/core/http/quic_spdy_client_session.h
index 8a3b1ed..06fe99e 100644
--- a/net/third_party/quic/core/quic_spdy_client_session.h
+++ b/net/third_party/quic/core/http/quic_spdy_client_session.h
@@ -4,16 +4,16 @@
 //
 // A client specific QuicSession subclass.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_H_
 
 #include <memory>
 
 #include "base/macros.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_crypto_client_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 
 namespace quic {
@@ -30,6 +30,8 @@
                         const QuicServerId& server_id,
                         QuicCryptoClientConfig* crypto_config,
                         QuicClientPushPromiseIndex* push_promise_index);
+  QuicSpdyClientSession(const QuicSpdyClientSession&) = delete;
+  QuicSpdyClientSession& operator=(const QuicSpdyClientSession&) = delete;
   ~QuicSpdyClientSession() override;
   // Set up the QuicSpdyClientSession. Must be called prior to use.
   void Initialize() override;
@@ -89,10 +91,8 @@
   // If this is set to false, the client will ignore server GOAWAYs and allow
   // the creation of streams regardless of the high chance they will fail.
   bool respect_goaway_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyClientSession);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_H_
diff --git a/net/third_party/quic/core/quic_spdy_client_session_base.cc b/net/third_party/quic/core/http/quic_spdy_client_session_base.cc
similarity index 96%
rename from net/third_party/quic/core/quic_spdy_client_session_base.cc
rename to net/third_party/quic/core/http/quic_spdy_client_session_base.cc
index c1d959f..5080011 100644
--- a/net/third_party/quic/core/quic_spdy_client_session_base.cc
+++ b/net/third_party/quic/core/http/quic_spdy_client_session_base.cc
@@ -2,10 +2,10 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
 
-#include "net/third_party/quic/core/quic_client_promised_info.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
diff --git a/net/third_party/quic/core/quic_spdy_client_session_base.h b/net/third_party/quic/core/http/quic_spdy_client_session_base.h
similarity index 92%
rename from net/third_party/quic/core/quic_spdy_client_session_base.h
rename to net/third_party/quic/core/http/quic_spdy_client_session_base.h
index a97ff63..b40aab9d 100644
--- a/net/third_party/quic/core/quic_spdy_client_session_base.h
+++ b/net/third_party/quic/core/http/quic_spdy_client_session_base.h
@@ -2,12 +2,12 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_BASE_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_BASE_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_BASE_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_BASE_H_
 
 #include "base/macros.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 #include "net/third_party/quic/core/quic_crypto_client_stream.h"
-#include "net/third_party/quic/core/quic_spdy_session.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
@@ -41,6 +41,9 @@
   QuicSpdyClientSessionBase(QuicConnection* connection,
                             QuicClientPushPromiseIndex* push_promise_index,
                             const QuicConfig& config);
+  QuicSpdyClientSessionBase(const QuicSpdyClientSessionBase&) = delete;
+  QuicSpdyClientSessionBase& operator=(const QuicSpdyClientSessionBase&) =
+      delete;
 
   ~QuicSpdyClientSessionBase() override;
 
@@ -129,10 +132,8 @@
   QuicClientPushPromiseIndex* push_promise_index_;
   QuicPromisedByIdMap promised_by_id_;
   QuicStreamId largest_promised_stream_id_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyClientSessionBase);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_SESSION_BASE_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_SESSION_BASE_H_
diff --git a/net/third_party/quic/core/quic_spdy_client_session_test.cc b/net/third_party/quic/core/http/quic_spdy_client_session_test.cc
similarity index 99%
rename from net/third_party/quic/core/quic_spdy_client_session_test.cc
rename to net/third_party/quic/core/http/quic_spdy_client_session_test.cc
index 142fe2af..37ab30a 100644
--- a/net/third_party/quic/core/quic_spdy_client_session_test.cc
+++ b/net/third_party/quic/core/http/quic_spdy_client_session_test.cc
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
 
 #include <memory>
 #include <vector>
 
 #include "net/third_party/quic/core/crypto/aes_128_gcm_12_encrypter.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_arraysize.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/core/quic_spdy_client_stream.cc b/net/third_party/quic/core/http/quic_spdy_client_stream.cc
similarity index 94%
rename from net/third_party/quic/core/quic_spdy_client_stream.cc
rename to net/third_party/quic/core/http/quic_spdy_client_stream.cc
index c03f47d..f6a157b 100644
--- a/net/third_party/quic/core/quic_spdy_client_stream.cc
+++ b/net/third_party/quic/core/http/quic_spdy_client_stream.cc
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 
 #include <utility>
 
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_alarm.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/spdy/core/spdy_protocol.h"
 
diff --git a/net/third_party/quic/core/quic_spdy_client_stream.h b/net/third_party/quic/core/http/quic_spdy_client_stream.h
similarity index 88%
rename from net/third_party/quic/core/quic_spdy_client_stream.h
rename to net/third_party/quic/core/http/quic_spdy_client_stream.h
index 5fe0e36..26d7138 100644
--- a/net/third_party/quic/core/quic_spdy_client_stream.h
+++ b/net/third_party/quic/core/http/quic_spdy_client_stream.h
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_STREAM_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_STREAM_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_STREAM_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_STREAM_H_
 
 #include <cstddef>
 
 #include "base/macros.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/spdy/core/spdy_framer.h"
@@ -23,6 +23,8 @@
 class QuicSpdyClientStream : public QuicSpdyStream {
  public:
   QuicSpdyClientStream(QuicStreamId id, QuicSpdyClientSession* session);
+  QuicSpdyClientStream(const QuicSpdyClientStream&) = delete;
+  QuicSpdyClientStream& operator=(const QuicSpdyClientStream&) = delete;
   ~QuicSpdyClientStream() override;
 
   // Override the base class to parse and store headers.
@@ -87,10 +89,8 @@
   // Expect: 100-continue.
   bool has_preliminary_headers_;
   spdy::SpdyHeaderBlock preliminary_headers_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyClientStream);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_CLIENT_STREAM_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_CLIENT_STREAM_H_
diff --git a/net/third_party/quic/core/quic_spdy_client_stream_test.cc b/net/third_party/quic/core/http/quic_spdy_client_stream_test.cc
similarity index 95%
rename from net/third_party/quic/core/quic_spdy_client_stream_test.cc
rename to net/third_party/quic/core/http/quic_spdy_client_stream_test.cc
index 33c713b..452c164 100644
--- a/net/third_party/quic/core/quic_spdy_client_stream_test.cc
+++ b/net/third_party/quic/core/http/quic_spdy_client_stream_test.cc
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 
 #include <memory>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
@@ -42,14 +42,15 @@
                               push_promise_index),
         crypto_config_(crypto_test_utils::ProofVerifierForTesting(),
                        TlsClientHandshaker::CreateSslCtx()) {}
+  MockQuicSpdyClientSession(const MockQuicSpdyClientSession&) = delete;
+  MockQuicSpdyClientSession& operator=(const MockQuicSpdyClientSession&) =
+      delete;
   ~MockQuicSpdyClientSession() override = default;
 
   MOCK_METHOD1(CloseStream, void(QuicStreamId stream_id));
 
  private:
   QuicCryptoClientConfig crypto_config_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSpdyClientSession);
 };
 
 class QuicSpdyClientStreamTest : public QuicTest {
diff --git a/net/third_party/quic/core/quic_spdy_server_stream_base.cc b/net/third_party/quic/core/http/quic_spdy_server_stream_base.cc
similarity index 94%
rename from net/third_party/quic/core/quic_spdy_server_stream_base.cc
rename to net/third_party/quic/core/http/quic_spdy_server_stream_base.cc
index 8ac9c6a..7da7f89 100644
--- a/net/third_party/quic/core/quic_spdy_server_stream_base.cc
+++ b/net/third_party/quic/core/http/quic_spdy_server_stream_base.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 "net/third_party/quic/core/quic_spdy_server_stream_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_server_stream_base.h"
 
 #include "net/third_party/quic/core/quic_error_codes.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
diff --git a/net/third_party/quic/core/http/quic_spdy_server_stream_base.h b/net/third_party/quic/core/http/quic_spdy_server_stream_base.h
new file mode 100644
index 0000000..56006b0
--- /dev/null
+++ b/net/third_party/quic/core/http/quic_spdy_server_stream_base.h
@@ -0,0 +1,26 @@
+// Copyright (c) 2016 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SERVER_STREAM_BASE_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SERVER_STREAM_BASE_H_
+
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
+
+namespace quic {
+
+class QuicSpdyServerStreamBase : public QuicSpdyStream {
+ public:
+  QuicSpdyServerStreamBase(QuicStreamId id, QuicSpdySession* session);
+  QuicSpdyServerStreamBase(const QuicSpdyServerStreamBase&) = delete;
+  QuicSpdyServerStreamBase& operator=(const QuicSpdyServerStreamBase&) = delete;
+
+  // Override the base class to send QUIC_STREAM_NO_ERROR to the peer
+  // when the stream has not received all the data.
+  void CloseWriteSide() override;
+  void StopReading() override;
+};
+
+}  // namespace quic
+
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SERVER_STREAM_BASE_H_
diff --git a/net/third_party/quic/core/quic_spdy_server_stream_base_test.cc b/net/third_party/quic/core/http/quic_spdy_server_stream_base_test.cc
similarity index 96%
rename from net/third_party/quic/core/quic_spdy_server_stream_base_test.cc
rename to net/third_party/quic/core/http/quic_spdy_server_stream_base_test.cc
index 032914a..44f4039 100644
--- a/net/third_party/quic/core/quic_spdy_server_stream_base_test.cc
+++ b/net/third_party/quic/core/http/quic_spdy_server_stream_base_test.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 "net/third_party/quic/core/quic_spdy_server_stream_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_server_stream_base.h"
 
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/quic/platform/api/quic_test.h"
diff --git a/net/third_party/quic/core/quic_spdy_session.cc b/net/third_party/quic/core/http/quic_spdy_session.cc
similarity index 97%
rename from net/third_party/quic/core/quic_spdy_session.cc
rename to net/third_party/quic/core/http/quic_spdy_session.cc
index 9127014..d600c519 100644
--- a/net/third_party/quic/core/quic_spdy_session.cc
+++ b/net/third_party/quic/core/http/quic_spdy_session.cc
@@ -2,13 +2,13 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 
 #include <algorithm>
 #include <cstdint>
 #include <utility>
 
-#include "net/third_party/quic/core/quic_headers_stream.h"
+#include "net/third_party/quic/core/http/quic_headers_stream.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_fallthrough.h"
 #include "net/third_party/quic/platform/api/quic_flag_utils.h"
@@ -55,6 +55,8 @@
   HeaderTableDebugVisitor(const QuicClock* clock,
                           std::unique_ptr<QuicHpackDebugVisitor> visitor)
       : clock_(clock), headers_stream_hpack_visitor_(std::move(visitor)) {}
+  HeaderTableDebugVisitor(const HeaderTableDebugVisitor&) = delete;
+  HeaderTableDebugVisitor& operator=(const HeaderTableDebugVisitor&) = delete;
 
   int64_t OnNewEntry(const HpackEntry& entry) override {
     QUIC_DVLOG(1) << entry.GetDebugString();
@@ -74,8 +76,6 @@
  private:
   const QuicClock* clock_;
   std::unique_ptr<QuicHpackDebugVisitor> headers_stream_hpack_visitor_;
-
-  DISALLOW_COPY_AND_ASSIGN(HeaderTableDebugVisitor);
 };
 
 }  // namespace
@@ -87,6 +87,8 @@
       public SpdyFramerDebugVisitorInterface {
  public:
   explicit SpdyFramerVisitor(QuicSpdySession* session) : session_(session) {}
+  SpdyFramerVisitor(const SpdyFramerVisitor&) = delete;
+  SpdyFramerVisitor& operator=(const SpdyFramerVisitor&) = delete;
 
   SpdyHeadersHandlerInterface* OnHeaderFrameStart(
       SpdyStreamId /* stream_id */) override {
@@ -289,8 +291,6 @@
  private:
   QuicSpdySession* session_;
   QuicHeaderList header_list_;
-
-  DISALLOW_COPY_AND_ASSIGN(SpdyFramerVisitor);
 };
 
 QuicHpackDebugVisitor::QuicHpackDebugVisitor() {}
diff --git a/net/third_party/quic/core/quic_spdy_session.h b/net/third_party/quic/core/http/quic_spdy_session.h
similarity index 93%
rename from net/third_party/quic/core/quic_spdy_session.h
rename to net/third_party/quic/core/http/quic_spdy_session.h
index 74ed443..9ecf6d0 100644
--- a/net/third_party/quic/core/quic_spdy_session.h
+++ b/net/third_party/quic/core/http/quic_spdy_session.h
@@ -2,17 +2,17 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SESSION_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SESSION_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SESSION_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SESSION_H_
 
 #include <cstddef>
 #include <memory>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_header_list.h"
-#include "net/third_party/quic/core/quic_headers_stream.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_headers_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_session.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
@@ -32,6 +32,8 @@
 class QUIC_EXPORT_PRIVATE QuicHpackDebugVisitor {
  public:
   QuicHpackDebugVisitor();
+  QuicHpackDebugVisitor(const QuicHpackDebugVisitor&) = delete;
+  QuicHpackDebugVisitor& operator=(const QuicHpackDebugVisitor&) = delete;
 
   virtual ~QuicHpackDebugVisitor();
 
@@ -39,9 +41,6 @@
   // the time since the corresponding entry was added to the dynamic
   // table.
   virtual void OnUseEntry(QuicTime::Delta elapsed) = 0;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicHpackDebugVisitor);
 };
 
 // A QUIC session with a headers stream.
@@ -51,6 +50,8 @@
   QuicSpdySession(QuicConnection* connection,
                   QuicSession::Visitor* visitor,
                   const QuicConfig& config);
+  QuicSpdySession(const QuicSpdySession&) = delete;
+  QuicSpdySession& operator=(const QuicSpdySession&) = delete;
 
   ~QuicSpdySession() override;
 
@@ -239,10 +240,8 @@
   spdy::SpdyFramer spdy_framer_;
   http2::Http2DecoderAdapter h2_deframer_;
   std::unique_ptr<SpdyFramerVisitor> spdy_framer_visitor_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdySession);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SESSION_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_SESSION_H_
diff --git a/net/third_party/quic/core/quic_spdy_session_test.cc b/net/third_party/quic/core/http/quic_spdy_session_test.cc
similarity index 99%
rename from net/third_party/quic/core/quic_spdy_session_test.cc
rename to net/third_party/quic/core/http/quic_spdy_session_test.cc
index 5edfb7e..fc9c5be 100644
--- a/net/third_party/quic/core/quic_spdy_session_test.cc
+++ b/net/third_party/quic/core/http/quic_spdy_session_test.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 "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 
 #include <cstdint>
 #include <set>
@@ -28,7 +28,6 @@
 #include "net/third_party/quic/test_tools/quic_flow_controller_peer.h"
 #include "net/third_party/quic/test_tools/quic_session_peer.h"
 #include "net/third_party/quic/test_tools/quic_spdy_session_peer.h"
-#include "net/third_party/quic/test_tools/quic_spdy_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_stream_send_buffer_peer.h"
 #include "net/third_party/quic/test_tools/quic_test_utils.h"
diff --git a/net/third_party/quic/core/quic_spdy_stream.cc b/net/third_party/quic/core/http/quic_spdy_stream.cc
similarity index 97%
rename from net/third_party/quic/core/quic_spdy_stream.cc
rename to net/third_party/quic/core/http/quic_spdy_stream.cc
index 3a8febc4..fe2c14f 100644
--- a/net/third_party/quic/core/quic_spdy_stream.cc
+++ b/net/third_party/quic/core/http/quic_spdy_stream.cc
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 
 #include <utility>
 
-#include "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_utils.h"
 #include "net/third_party/quic/core/quic_write_blocked_list.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
diff --git a/net/third_party/quic/core/quic_spdy_stream.h b/net/third_party/quic/core/http/quic_spdy_stream.h
similarity index 94%
rename from net/third_party/quic/core/quic_spdy_stream.h
rename to net/third_party/quic/core/http/quic_spdy_stream.h
index 0ebe198..a511cbb 100644
--- a/net/third_party/quic/core/quic_spdy_stream.h
+++ b/net/third_party/quic/core/http/quic_spdy_stream.h
@@ -6,8 +6,8 @@
 // In each direction, the data on such a stream first contains compressed
 // headers then body data.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_STREAM_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_STREAM_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_STREAM_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_STREAM_H_
 
 #include <sys/types.h>
 
@@ -16,7 +16,7 @@
 
 #include "base/macros.h"
 #include "net/base/iovec.h"
-#include "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_stream.h"
 #include "net/third_party/quic/core/quic_stream_sequencer.h"
@@ -29,7 +29,6 @@
 namespace quic {
 
 namespace test {
-class QuicSpdyStreamPeer;
 class QuicStreamPeer;
 }  // namespace test
 
@@ -42,6 +41,8 @@
   class QUIC_EXPORT_PRIVATE Visitor {
    public:
     Visitor() {}
+    Visitor(const Visitor&) = delete;
+    Visitor& operator=(const Visitor&) = delete;
 
     // Called when the stream is closed.
     virtual void OnClose(QuicSpdyStream* stream) = 0;
@@ -52,12 +53,11 @@
 
    protected:
     virtual ~Visitor() {}
-
-   private:
-    DISALLOW_COPY_AND_ASSIGN(Visitor);
   };
 
   QuicSpdyStream(QuicStreamId id, QuicSpdySession* spdy_session);
+  QuicSpdyStream(const QuicSpdyStream&) = delete;
+  QuicSpdyStream& operator=(const QuicSpdyStream&) = delete;
   ~QuicSpdyStream() override;
 
   // QuicStream implementation
@@ -181,7 +181,6 @@
   void set_headers_decompressed(bool val) { headers_decompressed_ = val; }
 
  private:
-  friend class test::QuicSpdyStreamPeer;
   friend class test::QuicStreamPeer;
   friend class QuicStreamUtils;
 
@@ -200,10 +199,8 @@
   bool trailers_consumed_;
   // The parsed trailers received from the peer.
   spdy::SpdyHeaderBlock received_trailers_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyStream);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_STREAM_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_QUIC_SPDY_STREAM_H_
diff --git a/net/third_party/quic/core/quic_spdy_stream_test.cc b/net/third_party/quic/core/http/quic_spdy_stream_test.cc
similarity index 99%
rename from net/third_party/quic/core/quic_spdy_stream_test.cc
rename to net/third_party/quic/core/http/quic_spdy_stream_test.cc
index f2a9963..394a180 100644
--- a/net/third_party/quic/core/quic_spdy_stream_test.cc
+++ b/net/third_party/quic/core/http/quic_spdy_stream_test.cc
@@ -2,15 +2,15 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "net/third_party/quic/core/quic_spdy_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 
 #include <memory>
 #include <utility>
 
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_utils.h"
 #include "net/third_party/quic/core/quic_write_blocked_list.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_arraysize.h"
 #include "net/third_party/quic/platform/api/quic_expect_bug.h"
 #include "net/third_party/quic/platform/api/quic_map_util.h"
diff --git a/net/third_party/quic/core/spdy_utils.cc b/net/third_party/quic/core/http/spdy_utils.cc
similarity index 98%
rename from net/third_party/quic/core/spdy_utils.cc
rename to net/third_party/quic/core/http/spdy_utils.cc
index 90ac60c..31e2d09 100644
--- a/net/third_party/quic/core/spdy_utils.cc
+++ b/net/third_party/quic/core/http/spdy_utils.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 "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 
 #include <memory>
 #include <vector>
diff --git a/net/third_party/quic/core/spdy_utils.h b/net/third_party/quic/core/http/spdy_utils.h
similarity index 89%
rename from net/third_party/quic/core/spdy_utils.h
rename to net/third_party/quic/core/http/spdy_utils.h
index b12966d8..ec1bf11 100644
--- a/net/third_party/quic/core/spdy_utils.h
+++ b/net/third_party/quic/core/http/spdy_utils.h
@@ -2,14 +2,14 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#ifndef NET_THIRD_PARTY_QUIC_CORE_SPDY_UTILS_H_
-#define NET_THIRD_PARTY_QUIC_CORE_SPDY_UTILS_H_
+#ifndef NET_THIRD_PARTY_QUIC_CORE_HTTP_SPDY_UTILS_H_
+#define NET_THIRD_PARTY_QUIC_CORE_HTTP_SPDY_UTILS_H_
 
 #include <cstddef>
 #include <cstdint>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/platform/api/quic_export.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
@@ -19,6 +19,8 @@
 
 class QUIC_EXPORT_PRIVATE SpdyUtils {
  public:
+  SpdyUtils() = delete;
+
   // Populate |content length| with the value of the content-length header.
   // Returns true on success, false if parsing fails or content-length header is
   // missing.
@@ -54,11 +56,8 @@
   // which must be fully-qualified.
   static bool PopulateHeaderBlockFromUrl(const QuicString url,
                                          spdy::SpdyHeaderBlock* headers);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(SpdyUtils);
 };
 
 }  // namespace quic
 
-#endif  // NET_THIRD_PARTY_QUIC_CORE_SPDY_UTILS_H_
+#endif  // NET_THIRD_PARTY_QUIC_CORE_HTTP_SPDY_UTILS_H_
diff --git a/net/third_party/quic/core/spdy_utils_test.cc b/net/third_party/quic/core/http/spdy_utils_test.cc
similarity index 99%
rename from net/third_party/quic/core/spdy_utils_test.cc
rename to net/third_party/quic/core/http/spdy_utils_test.cc
index 4910e72..3c6b618 100644
--- a/net/third_party/quic/core/spdy_utils_test.cc
+++ b/net/third_party/quic/core/http/spdy_utils_test.cc
@@ -5,7 +5,7 @@
 #include <memory>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_string.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/quic/platform/api/quic_test.h"
diff --git a/net/third_party/quic/core/quic_alarm.h b/net/third_party/quic/core/quic_alarm.h
index 966bd7c6..e45464108 100644
--- a/net/third_party/quic/core/quic_alarm.h
+++ b/net/third_party/quic/core/quic_alarm.h
@@ -28,6 +28,8 @@
   };
 
   explicit QuicAlarm(QuicArenaScopedPtr<Delegate> delegate);
+  QuicAlarm(const QuicAlarm&) = delete;
+  QuicAlarm& operator=(const QuicAlarm&) = delete;
   virtual ~QuicAlarm();
 
   // Sets the alarm to fire at |deadline|.  Must not be called while
@@ -78,8 +80,6 @@
  private:
   QuicArenaScopedPtr<Delegate> delegate_;
   QuicTime deadline_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicAlarm);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_arena_scoped_ptr.h b/net/third_party/quic/core/quic_arena_scoped_ptr.h
index 4ff6030c..7a524f85 100644
--- a/net/third_party/quic/core/quic_arena_scoped_ptr.h
+++ b/net/third_party/quic/core/quic_arena_scoped_ptr.h
@@ -73,6 +73,8 @@
 
   // Constructs a QuicArenaScopedPtr with the given representation.
   QuicArenaScopedPtr(void* value, ConstructFrom from);
+  QuicArenaScopedPtr(const QuicArenaScopedPtr&) = delete;
+  QuicArenaScopedPtr& operator=(const QuicArenaScopedPtr&) = delete;
 
   // Low-order bits of value_ that determine if the pointer came from an arena.
   static const uintptr_t kFromArenaMask = 0x1;
@@ -80,8 +82,6 @@
   // Every platform we care about has at least 4B aligned integers, so store the
   // is_from_arena bit in the least significant bit.
   void* value_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicArenaScopedPtr);
 };
 
 template <typename T>
diff --git a/net/third_party/quic/core/quic_connection.cc b/net/third_party/quic/core/quic_connection.cc
index d5460ad..b94f16f 100644
--- a/net/third_party/quic/core/quic_connection.cc
+++ b/net/third_party/quic/core/quic_connection.cc
@@ -85,6 +85,8 @@
  public:
   explicit AckAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  AckAlarmDelegate(const AckAlarmDelegate&) = delete;
+  AckAlarmDelegate& operator=(const AckAlarmDelegate&) = delete;
 
   void OnAlarm() override {
     DCHECK(connection_->ack_frame_updated());
@@ -94,8 +96,6 @@
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(AckAlarmDelegate);
 };
 
 // This alarm will be scheduled any time a data-bearing packet is sent out.
@@ -105,13 +105,14 @@
  public:
   explicit RetransmissionAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  RetransmissionAlarmDelegate(const RetransmissionAlarmDelegate&) = delete;
+  RetransmissionAlarmDelegate& operator=(const RetransmissionAlarmDelegate&) =
+      delete;
 
   void OnAlarm() override { connection_->OnRetransmissionTimeout(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(RetransmissionAlarmDelegate);
 };
 
 // An alarm that is scheduled when the SentPacketManager requires a delay
@@ -120,78 +121,82 @@
  public:
   explicit SendAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  SendAlarmDelegate(const SendAlarmDelegate&) = delete;
+  SendAlarmDelegate& operator=(const SendAlarmDelegate&) = delete;
 
   void OnAlarm() override { connection_->WriteAndBundleAcksIfNotBlocked(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(SendAlarmDelegate);
 };
 
 class PathDegradingAlarmDelegate : public QuicAlarm::Delegate {
  public:
   explicit PathDegradingAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  PathDegradingAlarmDelegate(const PathDegradingAlarmDelegate&) = delete;
+  PathDegradingAlarmDelegate& operator=(const PathDegradingAlarmDelegate&) =
+      delete;
 
   void OnAlarm() override { connection_->OnPathDegradingTimeout(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(PathDegradingAlarmDelegate);
 };
 
 class TimeoutAlarmDelegate : public QuicAlarm::Delegate {
  public:
   explicit TimeoutAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  TimeoutAlarmDelegate(const TimeoutAlarmDelegate&) = delete;
+  TimeoutAlarmDelegate& operator=(const TimeoutAlarmDelegate&) = delete;
 
   void OnAlarm() override { connection_->CheckForTimeout(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(TimeoutAlarmDelegate);
 };
 
 class PingAlarmDelegate : public QuicAlarm::Delegate {
  public:
   explicit PingAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  PingAlarmDelegate(const PingAlarmDelegate&) = delete;
+  PingAlarmDelegate& operator=(const PingAlarmDelegate&) = delete;
 
   void OnAlarm() override { connection_->OnPingTimeout(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(PingAlarmDelegate);
 };
 
 class MtuDiscoveryAlarmDelegate : public QuicAlarm::Delegate {
  public:
   explicit MtuDiscoveryAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  MtuDiscoveryAlarmDelegate(const MtuDiscoveryAlarmDelegate&) = delete;
+  MtuDiscoveryAlarmDelegate& operator=(const MtuDiscoveryAlarmDelegate&) =
+      delete;
 
   void OnAlarm() override { connection_->DiscoverMtu(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(MtuDiscoveryAlarmDelegate);
 };
 
 class RetransmittableOnWireAlarmDelegate : public QuicAlarm::Delegate {
  public:
   explicit RetransmittableOnWireAlarmDelegate(QuicConnection* connection)
       : connection_(connection) {}
+  RetransmittableOnWireAlarmDelegate(
+      const RetransmittableOnWireAlarmDelegate&) = delete;
+  RetransmittableOnWireAlarmDelegate& operator=(
+      const RetransmittableOnWireAlarmDelegate&) = delete;
 
   void OnAlarm() override { connection_->OnPingTimeout(); }
 
  private:
   QuicConnection* connection_;
-
-  DISALLOW_COPY_AND_ASSIGN(RetransmittableOnWireAlarmDelegate);
 };
 
 }  // namespace
diff --git a/net/third_party/quic/core/quic_connection.h b/net/third_party/quic/core/quic_connection.h
index a3ab651c..a0d370b 100644
--- a/net/third_party/quic/core/quic_connection.h
+++ b/net/third_party/quic/core/quic_connection.h
@@ -337,6 +337,8 @@
                  bool owns_writer,
                  Perspective perspective,
                  const ParsedQuicVersionVector& supported_versions);
+  QuicConnection(const QuicConnection&) = delete;
+  QuicConnection& operator=(const QuicConnection&) = delete;
   ~QuicConnection() override;
 
   // Sets connection parameters from the supplied |config|.
@@ -1295,8 +1297,6 @@
 
   // Latched value of quic_reloadable_flag_quic_retransmissions_app_limited.
   const bool retransmissions_app_limited_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicConnection);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_connection_test.cc b/net/third_party/quic/core/quic_connection_test.cc
index 714f521a..08b3417 100644
--- a/net/third_party/quic/core/quic_connection_test.cc
+++ b/net/third_party/quic/core/quic_connection_test.cc
@@ -91,6 +91,8 @@
 class TaggingEncrypter : public QuicEncrypter {
  public:
   explicit TaggingEncrypter(uint8_t tag) : tag_(tag) {}
+  TaggingEncrypter(const TaggingEncrypter&) = delete;
+  TaggingEncrypter& operator=(const TaggingEncrypter&) = delete;
 
   ~TaggingEncrypter() override {}
 
@@ -142,8 +144,6 @@
   };
 
   const uint8_t tag_;
-
-  DISALLOW_COPY_AND_ASSIGN(TaggingEncrypter);
 };
 
 // TaggingDecrypter ensures that the final kTagSize bytes of the message all
@@ -237,6 +237,8 @@
       : clock_(clock), random_generator_(random_generator) {
     clock_->AdvanceTime(QuicTime::Delta::FromSeconds(1));
   }
+  TestConnectionHelper(const TestConnectionHelper&) = delete;
+  TestConnectionHelper& operator=(const TestConnectionHelper&) = delete;
 
   // QuicConnectionHelperInterface
   const QuicClock* GetClock() const override { return clock_; }
@@ -251,8 +253,6 @@
   MockClock* clock_;
   MockRandom* random_generator_;
   SimpleBufferAllocator buffer_allocator_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestConnectionHelper);
 };
 
 class TestAlarmFactory : public QuicAlarmFactory {
@@ -268,6 +268,8 @@
   };
 
   TestAlarmFactory() {}
+  TestAlarmFactory(const TestAlarmFactory&) = delete;
+  TestAlarmFactory& operator=(const TestAlarmFactory&) = delete;
 
   QuicAlarm* CreateAlarm(QuicAlarm::Delegate* delegate) override {
     return new TestAlarm(QuicArenaScopedPtr<QuicAlarm::Delegate>(delegate));
@@ -278,9 +280,6 @@
       QuicConnectionArena* arena) override {
     return arena->New<TestAlarm>(std::move(delegate));
   }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(TestAlarmFactory);
 };
 
 class TestPacketWriter : public QuicPacketWriter {
@@ -302,6 +301,8 @@
         clock_(clock),
         write_pause_time_delta_(QuicTime::Delta::Zero()),
         max_packet_size_(kMaxPacketSize) {}
+  TestPacketWriter(const TestPacketWriter&) = delete;
+  TestPacketWriter& operator=(const TestPacketWriter&) = delete;
 
   // QuicPacketWriter interface
   WriteResult WritePacket(const char* buffer,
@@ -492,8 +493,6 @@
   // time.
   QuicTime::Delta write_pause_time_delta_;
   QuicByteCount max_packet_size_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestPacketWriter);
 };
 
 class TestConnection : public QuicConnection {
@@ -519,6 +518,8 @@
                  QuicMakeUnique<NullEncrypter>(perspective));
     SetDataProducer(&producer_);
   }
+  TestConnection(const TestConnection&) = delete;
+  TestConnection& operator=(const TestConnection&) = delete;
 
   void SendAck() { QuicConnectionPeer::SendAck(this); }
 
@@ -713,8 +714,6 @@
   SimpleSessionNotifier* notifier_;
 
   std::unique_ptr<QuicSocketAddress> next_effective_peer_addr_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestConnection);
 };
 
 enum class AckResponse { kDefer, kImmediate };
@@ -851,6 +850,9 @@
         .Times(AnyNumber());
   }
 
+  QuicConnectionTest(const QuicConnectionTest&) = delete;
+  QuicConnectionTest& operator=(const QuicConnectionTest&) = delete;
+
   ParsedQuicVersion version() { return GetParam().version; }
 
   QuicAckFrame* outgoing_ack() {
@@ -1245,9 +1247,6 @@
   QuicConnectionIdLength connection_id_length_;
 
   SimpleSessionNotifier notifier_;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicConnectionTest);
 };
 
 // Run all end to end tests with all supported versions.
diff --git a/net/third_party/quic/core/quic_crypto_client_handshaker.h b/net/third_party/quic/core/quic_crypto_client_handshaker.h
index c2859860c..a7cbb61 100644
--- a/net/third_party/quic/core/quic_crypto_client_handshaker.h
+++ b/net/third_party/quic/core/quic_crypto_client_handshaker.h
@@ -32,6 +32,9 @@
       std::unique_ptr<ProofVerifyContext> verify_context,
       QuicCryptoClientConfig* crypto_config,
       QuicCryptoClientStream::ProofHandler* proof_handler);
+  QuicCryptoClientHandshaker(const QuicCryptoClientHandshaker&) = delete;
+  QuicCryptoClientHandshaker& operator=(const QuicCryptoClientHandshaker&) =
+      delete;
 
   ~QuicCryptoClientHandshaker() override;
 
@@ -233,8 +236,6 @@
   bool handshake_confirmed_;
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters>
       crypto_negotiated_params_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoClientHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_client_stream.h b/net/third_party/quic/core/quic_crypto_client_stream.h
index 5bb85ce..b04047de8 100644
--- a/net/third_party/quic/core/quic_crypto_client_stream.h
+++ b/net/third_party/quic/core/quic_crypto_client_stream.h
@@ -134,6 +134,8 @@
                          std::unique_ptr<ProofVerifyContext> verify_context,
                          QuicCryptoClientConfig* crypto_config,
                          ProofHandler* proof_handler);
+  QuicCryptoClientStream(const QuicCryptoClientStream&) = delete;
+  QuicCryptoClientStream& operator=(const QuicCryptoClientStream&) = delete;
 
   ~QuicCryptoClientStream() override;
 
@@ -162,8 +164,6 @@
 
  private:
   std::unique_ptr<HandshakerDelegate> handshaker_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoClientStream);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_handshaker.h b/net/third_party/quic/core/quic_crypto_handshaker.h
index a88235fc..e5d139c 100644
--- a/net/third_party/quic/core/quic_crypto_handshaker.h
+++ b/net/third_party/quic/core/quic_crypto_handshaker.h
@@ -14,6 +14,8 @@
     : public CryptoFramerVisitorInterface {
  public:
   QuicCryptoHandshaker(QuicCryptoStream* stream, QuicSession* session);
+  QuicCryptoHandshaker(const QuicCryptoHandshaker&) = delete;
+  QuicCryptoHandshaker& operator=(const QuicCryptoHandshaker&) = delete;
 
   ~QuicCryptoHandshaker() override;
 
@@ -41,8 +43,6 @@
 
   // Records last sent crypto handshake message tag.
   QuicTag last_sent_handshake_message_tag_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_server_handshaker.h b/net/third_party/quic/core/quic_crypto_server_handshaker.h
index 3562a92..17a3be6d 100644
--- a/net/third_party/quic/core/quic_crypto_server_handshaker.h
+++ b/net/third_party/quic/core/quic_crypto_server_handshaker.h
@@ -31,6 +31,9 @@
                              QuicCompressedCertsCache* compressed_certs_cache,
                              QuicSession* session,
                              QuicCryptoServerStream::Helper* helper);
+  QuicCryptoServerHandshaker(const QuicCryptoServerHandshaker&) = delete;
+  QuicCryptoServerHandshaker& operator=(const QuicCryptoServerHandshaker&) =
+      delete;
 
   ~QuicCryptoServerHandshaker() override;
 
@@ -79,6 +82,8 @@
   class ValidateCallback : public ValidateClientHelloResultCallback {
    public:
     explicit ValidateCallback(QuicCryptoServerHandshaker* parent);
+    ValidateCallback(const ValidateCallback&) = delete;
+    ValidateCallback& operator=(const ValidateCallback&) = delete;
     // To allow the parent to detach itself from the callback before deletion.
     void Cancel();
 
@@ -88,8 +93,6 @@
 
    private:
     QuicCryptoServerHandshaker* parent_;
-
-    DISALLOW_COPY_AND_ASSIGN(ValidateCallback);
   };
 
   class SendServerConfigUpdateCallback
@@ -221,8 +224,6 @@
   bool handshake_confirmed_;
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters>
       crypto_negotiated_params_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoServerHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_server_stream.h b/net/third_party/quic/core/quic_crypto_server_stream.h
index 5626f25..1e49e48 100644
--- a/net/third_party/quic/core/quic_crypto_server_stream.h
+++ b/net/third_party/quic/core/quic_crypto_server_stream.h
@@ -161,6 +161,8 @@
                          bool use_stateless_rejects_if_peer_supported,
                          QuicSession* session,
                          Helper* helper);
+  QuicCryptoServerStream(const QuicCryptoServerStream&) = delete;
+  QuicCryptoServerStream& operator=(const QuicCryptoServerStream&) = delete;
 
   ~QuicCryptoServerStream() override;
 
@@ -220,8 +222,6 @@
   const QuicCryptoServerConfig* crypto_config_;
   QuicCompressedCertsCache* compressed_certs_cache_;
   Helper* helper_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoServerStream);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_stream.h b/net/third_party/quic/core/quic_crypto_stream.h
index 850502b..1d6a023f 100644
--- a/net/third_party/quic/core/quic_crypto_stream.h
+++ b/net/third_party/quic/core/quic_crypto_stream.h
@@ -34,6 +34,8 @@
 class QUIC_EXPORT_PRIVATE QuicCryptoStream : public QuicStream {
  public:
   explicit QuicCryptoStream(QuicSession* session);
+  QuicCryptoStream(const QuicCryptoStream&) = delete;
+  QuicCryptoStream& operator=(const QuicCryptoStream&) = delete;
 
   ~QuicCryptoStream() override;
 
@@ -108,8 +110,6 @@
   // TODO(fayang): This is not needed once switching from QUIC crypto to
   // TLS 1.3, which never encrypts crypto data.
   QuicIntervalSet<QuicStreamOffset> bytes_consumed_[NUM_ENCRYPTION_LEVELS];
-
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoStream);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_crypto_stream_test.cc b/net/third_party/quic/core/quic_crypto_stream_test.cc
index aa13037..a070950 100644
--- a/net/third_party/quic/core/quic_crypto_stream_test.cc
+++ b/net/third_party/quic/core/quic_crypto_stream_test.cc
@@ -34,6 +34,8 @@
       : QuicCryptoStream(session),
         QuicCryptoHandshaker(this, session),
         params_(new QuicCryptoNegotiatedParameters) {}
+  MockQuicCryptoStream(const MockQuicCryptoStream&) = delete;
+  MockQuicCryptoStream& operator=(const MockQuicCryptoStream&) = delete;
 
   void OnHandshakeMessage(const CryptoHandshakeMessage& message) override {
     messages_.push_back(message);
@@ -60,8 +62,6 @@
  private:
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters> params_;
   std::vector<CryptoHandshakeMessage> messages_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicCryptoStream);
 };
 
 class QuicCryptoStreamTest : public QuicTest {
@@ -79,6 +79,8 @@
     message_.SetStringPiece(2, "def");
     ConstructHandshakeMessage(Perspective::IS_SERVER);
   }
+  QuicCryptoStreamTest(const QuicCryptoStreamTest&) = delete;
+  QuicCryptoStreamTest& operator=(const QuicCryptoStreamTest&) = delete;
 
   void ConstructHandshakeMessage(Perspective perspective) {
     CryptoFramer framer;
@@ -94,9 +96,6 @@
   MockQuicCryptoStream* stream_;
   CryptoHandshakeMessage message_;
   std::unique_ptr<QuicData> message_data_;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicCryptoStreamTest);
 };
 
 TEST_F(QuicCryptoStreamTest, NotInitiallyConected) {
diff --git a/net/third_party/quic/core/quic_data_reader.h b/net/third_party/quic/core/quic_data_reader.h
index 256ed1b..81eaba1 100644
--- a/net/third_party/quic/core/quic_data_reader.h
+++ b/net/third_party/quic/core/quic_data_reader.h
@@ -34,6 +34,8 @@
  public:
   // Caller must provide an underlying buffer to work on.
   QuicDataReader(const char* data, const size_t len, Endianness endianness);
+  QuicDataReader(const QuicDataReader&) = delete;
+  QuicDataReader& operator=(const QuicDataReader&) = delete;
 
   // Empty destructor.
   ~QuicDataReader() {}
@@ -159,8 +161,6 @@
 
   // The endianness to read integers and floating numbers.
   Endianness endianness_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicDataReader);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_data_writer.h b/net/third_party/quic/core/quic_data_writer.h
index 98dbc68..7d7faaa1 100644
--- a/net/third_party/quic/core/quic_data_writer.h
+++ b/net/third_party/quic/core/quic_data_writer.h
@@ -38,6 +38,8 @@
  public:
   // Creates a QuicDataWriter where |buffer| is not owned.
   QuicDataWriter(size_t size, char* buffer, Endianness endianness);
+  QuicDataWriter(const QuicDataWriter&) = delete;
+  QuicDataWriter& operator=(const QuicDataWriter&) = delete;
 
   ~QuicDataWriter();
 
@@ -127,8 +129,6 @@
 
   // The endianness to write integers and floating numbers.
   Endianness endianness_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicDataWriter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_default_packet_writer.h b/net/third_party/quic/core/quic_default_packet_writer.h
index b72daa1..3500f91 100644
--- a/net/third_party/quic/core/quic_default_packet_writer.h
+++ b/net/third_party/quic/core/quic_default_packet_writer.h
@@ -20,6 +20,8 @@
 class QUIC_EXPORT_PRIVATE QuicDefaultPacketWriter : public QuicPacketWriter {
  public:
   explicit QuicDefaultPacketWriter(int fd);
+  QuicDefaultPacketWriter(const QuicDefaultPacketWriter&) = delete;
+  QuicDefaultPacketWriter& operator=(const QuicDefaultPacketWriter&) = delete;
   ~QuicDefaultPacketWriter() override;
 
   // QuicPacketWriter
@@ -47,8 +49,6 @@
  private:
   int fd_;
   bool write_blocked_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicDefaultPacketWriter);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_dispatcher.cc b/net/third_party/quic/core/quic_dispatcher.cc
index b863b78..79430b88 100644
--- a/net/third_party/quic/core/quic_dispatcher.cc
+++ b/net/third_party/quic/core/quic_dispatcher.cc
@@ -35,14 +35,14 @@
  public:
   explicit DeleteSessionsAlarm(QuicDispatcher* dispatcher)
       : dispatcher_(dispatcher) {}
+  DeleteSessionsAlarm(const DeleteSessionsAlarm&) = delete;
+  DeleteSessionsAlarm& operator=(const DeleteSessionsAlarm&) = delete;
 
   void OnAlarm() override { dispatcher_->DeleteSessions(); }
 
  private:
   // Not owned.
   QuicDispatcher* dispatcher_;
-
-  DISALLOW_COPY_AND_ASSIGN(DeleteSessionsAlarm);
 };
 
 // Collects packets serialized by a QuicPacketCreator in order
diff --git a/net/third_party/quic/core/quic_dispatcher.h b/net/third_party/quic/core/quic_dispatcher.h
index 2b5402a..61c6392 100644
--- a/net/third_party/quic/core/quic_dispatcher.h
+++ b/net/third_party/quic/core/quic_dispatcher.h
@@ -50,6 +50,8 @@
                  std::unique_ptr<QuicConnectionHelperInterface> helper,
                  std::unique_ptr<QuicCryptoServerStream::Helper> session_helper,
                  std::unique_ptr<QuicAlarmFactory> alarm_factory);
+  QuicDispatcher(const QuicDispatcher&) = delete;
+  QuicDispatcher& operator=(const QuicDispatcher&) = delete;
 
   ~QuicDispatcher() override;
 
@@ -446,8 +448,6 @@
 
   // True if this dispatcher is not draining.
   bool accept_new_connections_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicDispatcher);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_dispatcher_test.cc b/net/third_party/quic/core/quic_dispatcher_test.cc
index 2f9e3d1..2bbdf16 100644
--- a/net/third_party/quic/core/quic_dispatcher_test.cc
+++ b/net/third_party/quic/core/quic_dispatcher_test.cc
@@ -65,6 +65,9 @@
                               crypto_config,
                               compressed_certs_cache),
         crypto_stream_(QuicServerSessionBase::GetMutableCryptoStream()) {}
+  TestQuicSpdyServerSession(const TestQuicSpdyServerSession&) = delete;
+  TestQuicSpdyServerSession& operator=(const TestQuicSpdyServerSession&) =
+      delete;
 
   ~TestQuicSpdyServerSession() override { delete connection(); };
 
@@ -102,8 +105,6 @@
 
  private:
   QuicCryptoServerStreamBase* crypto_stream_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestQuicSpdyServerSession);
 };
 
 class TestDispatcher : public QuicDispatcher {
@@ -805,6 +806,9 @@
             session,
             helper),
         handshake_confirmed_(false) {}
+  MockQuicCryptoServerStream(const MockQuicCryptoServerStream&) = delete;
+  MockQuicCryptoServerStream& operator=(const MockQuicCryptoServerStream&) =
+      delete;
 
   void set_handshake_confirmed_for_testing(bool handshake_confirmed) {
     handshake_confirmed_ = handshake_confirmed;
@@ -814,7 +818,6 @@
 
  private:
   bool handshake_confirmed_;
-  DISALLOW_COPY_AND_ASSIGN(MockQuicCryptoServerStream);
 };
 
 struct StatelessRejectTestParams {
diff --git a/net/third_party/quic/core/quic_epoll_alarm_factory.h b/net/third_party/quic/core/quic_epoll_alarm_factory.h
index 87e8ccda..9fe432d 100644
--- a/net/third_party/quic/core/quic_epoll_alarm_factory.h
+++ b/net/third_party/quic/core/quic_epoll_alarm_factory.h
@@ -18,6 +18,8 @@
 class QuicEpollAlarmFactory : public QuicAlarmFactory {
  public:
   explicit QuicEpollAlarmFactory(net::EpollServer* epoll_server);
+  QuicEpollAlarmFactory(const QuicEpollAlarmFactory&) = delete;
+  QuicEpollAlarmFactory& operator=(const QuicEpollAlarmFactory&) = delete;
   ~QuicEpollAlarmFactory() override;
 
   // QuicAlarmFactory interface.
@@ -28,8 +30,6 @@
 
  private:
   net::EpollServer* epoll_server_;  // Not owned.
-
-  DISALLOW_COPY_AND_ASSIGN(QuicEpollAlarmFactory);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_epoll_connection_helper.h b/net/third_party/quic/core/quic_epoll_connection_helper.h
index f55be65..d996e08 100644
--- a/net/third_party/quic/core/quic_epoll_connection_helper.h
+++ b/net/third_party/quic/core/quic_epoll_connection_helper.h
@@ -34,6 +34,9 @@
 class QuicEpollConnectionHelper : public QuicConnectionHelperInterface {
  public:
   QuicEpollConnectionHelper(net::EpollServer* eps, QuicAllocator allocator);
+  QuicEpollConnectionHelper(const QuicEpollConnectionHelper&) = delete;
+  QuicEpollConnectionHelper& operator=(const QuicEpollConnectionHelper&) =
+      delete;
   ~QuicEpollConnectionHelper() override;
 
   // QuicEpollConnectionHelperInterface
@@ -51,8 +54,6 @@
   QuicStreamBufferAllocator stream_buffer_allocator_;
   SimpleBufferAllocator simple_buffer_allocator_;
   QuicAllocator allocator_type_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicEpollConnectionHelper);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_flow_controller.h b/net/third_party/quic/core/quic_flow_controller.h
index 6092f42..fc5deab 100644
--- a/net/third_party/quic/core/quic_flow_controller.h
+++ b/net/third_party/quic/core/quic_flow_controller.h
@@ -48,6 +48,8 @@
                      QuicStreamOffset receive_window_offset,
                      bool should_auto_tune_receive_window,
                      QuicFlowControllerInterface* session_flow_controller);
+  QuicFlowController(const QuicFlowController&) = delete;
+  QuicFlowController& operator=(const QuicFlowController&) = delete;
 
   ~QuicFlowController() override {}
 
@@ -195,8 +197,6 @@
   // Keep time of the last time a window update was sent.  We use this
   // as part of the receive window auto tuning.
   QuicTime prev_window_update_time_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicFlowController);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_flow_controller_test.cc b/net/third_party/quic/core/quic_flow_controller_test.cc
index b4a201b9..b59fac99 100644
--- a/net/third_party/quic/core/quic_flow_controller_test.cc
+++ b/net/third_party/quic/core/quic_flow_controller_test.cc
@@ -26,12 +26,11 @@
 class MockFlowController : public QuicFlowControllerInterface {
  public:
   MockFlowController() {}
+  MockFlowController(const MockFlowController&) = delete;
+  MockFlowController& operator=(const MockFlowController&) = delete;
   ~MockFlowController() override {}
 
   MOCK_METHOD1(EnsureWindowAtLeast, void(QuicByteCount));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockFlowController);
 };
 
 class QuicFlowControllerTest : public QuicTest {
diff --git a/net/third_party/quic/core/quic_framer.h b/net/third_party/quic/core/quic_framer.h
index 95420ef9..74dca318 100644
--- a/net/third_party/quic/core/quic_framer.h
+++ b/net/third_party/quic/core/quic_framer.h
@@ -195,6 +195,8 @@
   QuicFramer(const ParsedQuicVersionVector& supported_versions,
              QuicTime creation_time,
              Perspective perspective);
+  QuicFramer(const QuicFramer&) = delete;
+  QuicFramer& operator=(const QuicFramer&) = delete;
 
   virtual ~QuicFramer();
 
@@ -763,8 +765,6 @@
   // If not null, framer asks data_producer_ to write stream frame data. Not
   // owned. TODO(fayang): Consider add data producer to framer's constructor.
   QuicStreamFrameDataProducer* data_producer_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicFramer);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_one_block_arena.h b/net/third_party/quic/core/quic_one_block_arena.h
index 494331ee..c07c2cc 100644
--- a/net/third_party/quic/core/quic_one_block_arena.h
+++ b/net/third_party/quic/core/quic_one_block_arena.h
@@ -26,6 +26,8 @@
 
  public:
   QuicOneBlockArena();
+  QuicOneBlockArena(const QuicOneBlockArena&) = delete;
+  QuicOneBlockArena& operator=(const QuicOneBlockArena&) = delete;
 
   // Instantiates an object of type |T| with |args|. |args| are perfectly
   // forwarded to |T|'s constructor. The returned pointer's lifetime is
@@ -46,8 +48,6 @@
   QUIC_ALIGNED(8) char storage_[ArenaSize];
   // Current offset into the storage.
   uint32_t offset_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicOneBlockArena);
 };
 
 template <uint32_t ArenaSize>
diff --git a/net/third_party/quic/core/quic_packet_creator.h b/net/third_party/quic/core/quic_packet_creator.h
index 22859e9..2e670a07 100644
--- a/net/third_party/quic/core/quic_packet_creator.h
+++ b/net/third_party/quic/core/quic_packet_creator.h
@@ -55,6 +55,8 @@
   QuicPacketCreator(QuicConnectionId connection_id,
                     QuicFramer* framer,
                     DelegateInterface* delegate);
+  QuicPacketCreator(const QuicPacketCreator&) = delete;
+  QuicPacketCreator& operator=(const QuicPacketCreator&) = delete;
 
   ~QuicPacketCreator();
 
@@ -328,8 +330,6 @@
   // If true, packet_'s transmission type is only set by
   // SetPacketTransmissionType and does not get cleared in ClearPacket.
   bool can_set_transmission_type_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketCreator);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_packet_generator.h b/net/third_party/quic/core/quic_packet_generator.h
index 50a4077..44fb7eba 100644
--- a/net/third_party/quic/core/quic_packet_generator.h
+++ b/net/third_party/quic/core/quic_packet_generator.h
@@ -75,6 +75,8 @@
                       QuicFramer* framer,
                       QuicRandom* random_generator,
                       DelegateInterface* delegate);
+  QuicPacketGenerator(const QuicPacketGenerator&) = delete;
+  QuicPacketGenerator& operator=(const QuicPacketGenerator&) = delete;
 
   ~QuicPacketGenerator();
 
@@ -237,8 +239,6 @@
   QuicStopWaitingFrame pending_stop_waiting_frame_;
 
   QuicRandom* random_generator_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketGenerator);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_packet_generator_test.cc b/net/third_party/quic/core/quic_packet_generator_test.cc
index 4b11b4c..721039d3 100644
--- a/net/third_party/quic/core/quic_packet_generator_test.cc
+++ b/net/third_party/quic/core/quic_packet_generator_test.cc
@@ -40,6 +40,8 @@
 class MockDelegate : public QuicPacketGenerator::DelegateInterface {
  public:
   MockDelegate() {}
+  MockDelegate(const MockDelegate&) = delete;
+  MockDelegate& operator=(const MockDelegate&) = delete;
   ~MockDelegate() override {}
 
   MOCK_METHOD2(ShouldGeneratePacket,
@@ -72,9 +74,6 @@
     EXPECT_CALL(*this, ShouldGeneratePacket(NO_RETRANSMITTABLE_DATA, _))
         .WillRepeatedly(Return(true));
   }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockDelegate);
 };
 
 // Simple struct for describing the contents of a packet.
diff --git a/net/third_party/quic/core/quic_packet_reader.h b/net/third_party/quic/core/quic_packet_reader.h
index 5fb7abb..f1353379 100644
--- a/net/third_party/quic/core/quic_packet_reader.h
+++ b/net/third_party/quic/core/quic_packet_reader.h
@@ -31,6 +31,8 @@
 class QuicPacketReader {
  public:
   QuicPacketReader();
+  QuicPacketReader(const QuicPacketReader&) = delete;
+  QuicPacketReader& operator=(const QuicPacketReader&) = delete;
 
   virtual ~QuicPacketReader();
 
@@ -84,8 +86,6 @@
   PacketData packets_[kNumPacketsPerReadMmsgCall];
   mmsghdr mmsg_hdr_[kNumPacketsPerReadMmsgCall];
 #endif
-
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketReader);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_packet_writer.h b/net/third_party/quic/core/quic_packet_writer.h
index 46e0860..0399a8b9a 100644
--- a/net/third_party/quic/core/quic_packet_writer.h
+++ b/net/third_party/quic/core/quic_packet_writer.h
@@ -28,9 +28,9 @@
 
  private:
   PerPacketOptions(PerPacketOptions&& other) = delete;
+  PerPacketOptions(const PerPacketOptions&) = delete;
+  PerPacketOptions& operator=(const PerPacketOptions&) = delete;
   PerPacketOptions& operator=(PerPacketOptions&& other) = delete;
-
-  DISALLOW_COPY_AND_ASSIGN(PerPacketOptions);
 };
 
 // An interface between writers and the entity managing the
diff --git a/net/third_party/quic/core/quic_packet_writer_wrapper.h b/net/third_party/quic/core/quic_packet_writer_wrapper.h
index 10820f5d..17a5b7b4 100644
--- a/net/third_party/quic/core/quic_packet_writer_wrapper.h
+++ b/net/third_party/quic/core/quic_packet_writer_wrapper.h
@@ -19,6 +19,8 @@
 class QuicPacketWriterWrapper : public QuicPacketWriter {
  public:
   QuicPacketWriterWrapper();
+  QuicPacketWriterWrapper(const QuicPacketWriterWrapper&) = delete;
+  QuicPacketWriterWrapper& operator=(const QuicPacketWriterWrapper&) = delete;
   ~QuicPacketWriterWrapper() override;
 
   // Default implementation of the QuicPacketWriter interface. Passes everything
@@ -53,8 +55,6 @@
 
   QuicPacketWriter* writer_ = nullptr;
   bool owns_writer_ = false;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketWriterWrapper);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_packets.h b/net/third_party/quic/core/quic_packets.h
index 7b23cb0..e558340 100644
--- a/net/third_party/quic/core/quic_packets.h
+++ b/net/third_party/quic/core/quic_packets.h
@@ -127,6 +127,8 @@
  public:
   QuicData(const char* buffer, size_t length);
   QuicData(const char* buffer, size_t length, bool owns_buffer);
+  QuicData(const QuicData&) = delete;
+  QuicData& operator=(const QuicData&) = delete;
   virtual ~QuicData();
 
   QuicStringPiece AsStringPiece() const {
@@ -140,8 +142,6 @@
   const char* buffer_;
   size_t length_;
   bool owns_buffer_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicData);
 };
 
 class QUIC_EXPORT_PRIVATE QuicPacket : public QuicData {
@@ -157,6 +157,8 @@
              bool includes_version,
              bool includes_diversification_nonce,
              QuicPacketNumberLength packet_number_length);
+  QuicPacket(const QuicPacket&) = delete;
+  QuicPacket& operator=(const QuicPacket&) = delete;
 
   QuicStringPiece AssociatedData(QuicTransportVersion version) const;
   QuicStringPiece Plaintext(QuicTransportVersion version) const;
@@ -170,14 +172,14 @@
   const bool includes_version_;
   const bool includes_diversification_nonce_;
   const QuicPacketNumberLength packet_number_length_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicPacket);
 };
 
 class QUIC_EXPORT_PRIVATE QuicEncryptedPacket : public QuicData {
  public:
   QuicEncryptedPacket(const char* buffer, size_t length);
   QuicEncryptedPacket(const char* buffer, size_t length, bool owns_buffer);
+  QuicEncryptedPacket(const QuicEncryptedPacket&) = delete;
+  QuicEncryptedPacket& operator=(const QuicEncryptedPacket&) = delete;
 
   // Clones the packet into a new packet which owns the buffer.
   std::unique_ptr<QuicEncryptedPacket> Clone() const;
@@ -189,9 +191,6 @@
   QUIC_EXPORT_PRIVATE friend std::ostream& operator<<(
       std::ostream& os,
       const QuicEncryptedPacket& s);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicEncryptedPacket);
 };
 
 // A received encrypted QUIC packet, with a recorded time of receipt.
@@ -208,6 +207,8 @@
                      bool owns_buffer,
                      int ttl,
                      bool ttl_valid);
+  QuicReceivedPacket(const QuicReceivedPacket&) = delete;
+  QuicReceivedPacket& operator=(const QuicReceivedPacket&) = delete;
 
   // Clones the packet into a new packet which owns the buffer.
   std::unique_ptr<QuicReceivedPacket> Clone() const;
@@ -229,8 +230,6 @@
  private:
   const QuicTime receipt_time_;
   int ttl_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicReceivedPacket);
 };
 
 struct QUIC_EXPORT_PRIVATE SerializedPacket {
diff --git a/net/third_party/quic/core/quic_received_packet_manager.h b/net/third_party/quic/core/quic_received_packet_manager.h
index 7c615004..4438194 100644
--- a/net/third_party/quic/core/quic_received_packet_manager.h
+++ b/net/third_party/quic/core/quic_received_packet_manager.h
@@ -23,6 +23,9 @@
 class QUIC_EXPORT_PRIVATE QuicReceivedPacketManager {
  public:
   explicit QuicReceivedPacketManager(QuicConnectionStats* stats);
+  QuicReceivedPacketManager(const QuicReceivedPacketManager&) = delete;
+  QuicReceivedPacketManager& operator=(const QuicReceivedPacketManager&) =
+      delete;
   virtual ~QuicReceivedPacketManager();
 
   // Updates the internal state concerning which packets have been received.
@@ -92,8 +95,6 @@
   QuicTime time_largest_observed_;
 
   QuicConnectionStats* stats_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicReceivedPacketManager);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_sent_packet_manager.h b/net/third_party/quic/core/quic_sent_packet_manager.h
index ef8e796..85e7b48 100644
--- a/net/third_party/quic/core/quic_sent_packet_manager.h
+++ b/net/third_party/quic/core/quic_sent_packet_manager.h
@@ -89,6 +89,8 @@
                         QuicConnectionStats* stats,
                         CongestionControlType congestion_control_type,
                         LossDetectionType loss_type);
+  QuicSentPacketManager(const QuicSentPacketManager&) = delete;
+  QuicSentPacketManager& operator=(const QuicSentPacketManager&) = delete;
   virtual ~QuicSentPacketManager();
 
   virtual void SetFromConfig(const QuicConfig& config);
@@ -551,8 +553,6 @@
   // A reverse iterator of last_ack_frame_.packets. This is reset in
   // OnAckRangeStart, and gradually moves in OnAckRange..
   PacketNumberQueue::const_reverse_iterator acked_packets_iter_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSentPacketManager);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_session.h b/net/third_party/quic/core/quic_session.h
index ce9f92db..b5093e6e 100644
--- a/net/third_party/quic/core/quic_session.h
+++ b/net/third_party/quic/core/quic_session.h
@@ -82,6 +82,8 @@
   QuicSession(QuicConnection* connection,
               Visitor* owner,
               const QuicConfig& config);
+  QuicSession(const QuicSession&) = delete;
+  QuicSession& operator=(const QuicSession&) = delete;
 
   ~QuicSession() override;
 
@@ -551,8 +553,6 @@
   // is not used here.
   // List of streams with pending retransmissions.
   QuicLinkedHashMap<QuicStreamId, bool> streams_with_pending_retransmission_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSession);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_socket_address_coder.h b/net/third_party/quic/core/quic_socket_address_coder.h
index 6894079d..dbb84ad 100644
--- a/net/third_party/quic/core/quic_socket_address_coder.h
+++ b/net/third_party/quic/core/quic_socket_address_coder.h
@@ -21,6 +21,8 @@
  public:
   QuicSocketAddressCoder();
   explicit QuicSocketAddressCoder(const QuicSocketAddress& address);
+  QuicSocketAddressCoder(const QuicSocketAddressCoder&) = delete;
+  QuicSocketAddressCoder& operator=(const QuicSocketAddressCoder&) = delete;
   ~QuicSocketAddressCoder();
 
   QuicString Encode() const;
@@ -33,8 +35,6 @@
 
  private:
   QuicSocketAddress address_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSocketAddressCoder);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_spdy_server_stream_base.h b/net/third_party/quic/core/quic_spdy_server_stream_base.h
deleted file mode 100644
index 74c7fa4..0000000
--- a/net/third_party/quic/core/quic_spdy_server_stream_base.h
+++ /dev/null
@@ -1,27 +0,0 @@
-// Copyright (c) 2016 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SERVER_STREAM_BASE_H_
-#define NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SERVER_STREAM_BASE_H_
-
-#include "net/third_party/quic/core/quic_spdy_stream.h"
-
-namespace quic {
-
-class QuicSpdyServerStreamBase : public QuicSpdyStream {
- public:
-  QuicSpdyServerStreamBase(QuicStreamId id, QuicSpdySession* session);
-
-  // Override the base class to send QUIC_STREAM_NO_ERROR to the peer
-  // when the stream has not received all the data.
-  void CloseWriteSide() override;
-  void StopReading() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyServerStreamBase);
-};
-
-}  // namespace quic
-
-#endif  // NET_THIRD_PARTY_QUIC_CORE_QUIC_SPDY_SERVER_STREAM_BASE_H_
diff --git a/net/third_party/quic/core/quic_stream.h b/net/third_party/quic/core/quic_stream.h
index 75e0d17..daa5ff49 100644
--- a/net/third_party/quic/core/quic_stream.h
+++ b/net/third_party/quic/core/quic_stream.h
@@ -59,6 +59,8 @@
   // |is_static| is true, then the stream will be given precedence
   // over other streams when determing what streams should write next.
   QuicStream(QuicStreamId id, QuicSession* session, bool is_static);
+  QuicStream(const QuicStream&) = delete;
+  QuicStream& operator=(const QuicStream&) = delete;
 
   virtual ~QuicStream();
 
@@ -433,8 +435,6 @@
   // If true, then this stream has precedence over other streams for write
   // scheduling.
   const bool is_static_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicStream);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_stream_sequencer.h b/net/third_party/quic/core/quic_stream_sequencer.h
index e93d11b..f536f45a 100644
--- a/net/third_party/quic/core/quic_stream_sequencer.h
+++ b/net/third_party/quic/core/quic_stream_sequencer.h
@@ -27,6 +27,8 @@
 class QUIC_EXPORT_PRIVATE QuicStreamSequencer {
  public:
   explicit QuicStreamSequencer(QuicStream* quic_stream);
+  QuicStreamSequencer(const QuicStreamSequencer&) = delete;
+  QuicStreamSequencer& operator=(const QuicStreamSequencer&) = delete;
   virtual ~QuicStreamSequencer();
 
   // If the frame is the next one we need in order to process in-order data,
@@ -152,8 +154,6 @@
   // If false, only call OnDataAvailable() when it becomes newly unblocked.
   // Otherwise, call OnDataAvailable() when number of readable bytes changes.
   bool level_triggered_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamSequencer);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_stream_sequencer_buffer.h b/net/third_party/quic/core/quic_stream_sequencer_buffer.h
index e76be81..45368dd 100644
--- a/net/third_party/quic/core/quic_stream_sequencer_buffer.h
+++ b/net/third_party/quic/core/quic_stream_sequencer_buffer.h
@@ -88,6 +88,9 @@
   };
 
   explicit QuicStreamSequencerBuffer(size_t max_capacity_bytes);
+  QuicStreamSequencerBuffer(const QuicStreamSequencerBuffer&) = delete;
+  QuicStreamSequencerBuffer& operator=(const QuicStreamSequencerBuffer&) =
+      delete;
   ~QuicStreamSequencerBuffer();
 
   // Free the space used to buffer data.
@@ -225,8 +228,6 @@
 
   // Currently received data.
   QuicIntervalSet<QuicStreamOffset> bytes_received_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamSequencerBuffer);
 };
 }  // namespace quic
 
diff --git a/net/third_party/quic/core/quic_sustained_bandwidth_recorder.h b/net/third_party/quic/core/quic_sustained_bandwidth_recorder.h
index 73baa1d0..123d193 100644
--- a/net/third_party/quic/core/quic_sustained_bandwidth_recorder.h
+++ b/net/third_party/quic/core/quic_sustained_bandwidth_recorder.h
@@ -26,6 +26,10 @@
 class QUIC_EXPORT_PRIVATE QuicSustainedBandwidthRecorder {
  public:
   QuicSustainedBandwidthRecorder();
+  QuicSustainedBandwidthRecorder(const QuicSustainedBandwidthRecorder&) =
+      delete;
+  QuicSustainedBandwidthRecorder& operator=(
+      const QuicSustainedBandwidthRecorder&) = delete;
 
   // As long as |in_recovery| is consistently false, multiple calls to this
   // method over a 3 * srtt period results in storage of a valid sustained
@@ -85,8 +89,6 @@
 
   // Timestamp marking the beginning of the latest recording period.
   QuicTime start_time_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSustainedBandwidthRecorder);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_time_wait_list_manager.cc b/net/third_party/quic/core/quic_time_wait_list_manager.cc
index 95bbecf..00aa6d5 100644
--- a/net/third_party/quic/core/quic_time_wait_list_manager.cc
+++ b/net/third_party/quic/core/quic_time_wait_list_manager.cc
@@ -32,6 +32,8 @@
   explicit ConnectionIdCleanUpAlarm(
       QuicTimeWaitListManager* time_wait_list_manager)
       : time_wait_list_manager_(time_wait_list_manager) {}
+  ConnectionIdCleanUpAlarm(const ConnectionIdCleanUpAlarm&) = delete;
+  ConnectionIdCleanUpAlarm& operator=(const ConnectionIdCleanUpAlarm&) = delete;
 
   void OnAlarm() override {
     time_wait_list_manager_->CleanUpOldConnectionIds();
@@ -40,8 +42,6 @@
  private:
   // Not owned.
   QuicTimeWaitListManager* time_wait_list_manager_;
-
-  DISALLOW_COPY_AND_ASSIGN(ConnectionIdCleanUpAlarm);
 };
 
 // This class stores pending public reset packets to be sent to clients.
@@ -59,6 +59,8 @@
       : server_address_(server_address),
         client_address_(client_address),
         packet_(std::move(packet)) {}
+  QueuedPacket(const QueuedPacket&) = delete;
+  QueuedPacket& operator=(const QueuedPacket&) = delete;
 
   const QuicSocketAddress& server_address() const { return server_address_; }
   const QuicSocketAddress& client_address() const { return client_address_; }
@@ -68,8 +70,6 @@
   const QuicSocketAddress server_address_;
   const QuicSocketAddress client_address_;
   std::unique_ptr<QuicEncryptedPacket> packet_;
-
-  DISALLOW_COPY_AND_ASSIGN(QueuedPacket);
 };
 
 QuicTimeWaitListManager::QuicTimeWaitListManager(
diff --git a/net/third_party/quic/core/quic_time_wait_list_manager.h b/net/third_party/quic/core/quic_time_wait_list_manager.h
index bae6a2c..45f559f 100644
--- a/net/third_party/quic/core/quic_time_wait_list_manager.h
+++ b/net/third_party/quic/core/quic_time_wait_list_manager.h
@@ -66,6 +66,8 @@
                           Visitor* visitor,
                           const QuicClock* clock,
                           QuicAlarmFactory* alarm_factory);
+  QuicTimeWaitListManager(const QuicTimeWaitListManager&) = delete;
+  QuicTimeWaitListManager& operator=(const QuicTimeWaitListManager&) = delete;
   ~QuicTimeWaitListManager() override;
 
   // Adds the given connection_id to time wait state for time_wait_period_.
@@ -215,8 +217,6 @@
 
   // Interface that manages blocked writers.
   Visitor* visitor_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicTimeWaitListManager);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_unacked_packet_map.h b/net/third_party/quic/core/quic_unacked_packet_map.h
index ef14e46..656fbca 100644
--- a/net/third_party/quic/core/quic_unacked_packet_map.h
+++ b/net/third_party/quic/core/quic_unacked_packet_map.h
@@ -23,6 +23,8 @@
 class QUIC_EXPORT_PRIVATE QuicUnackedPacketMap {
  public:
   QuicUnackedPacketMap();
+  QuicUnackedPacketMap(const QuicUnackedPacketMap&) = delete;
+  QuicUnackedPacketMap& operator=(const QuicUnackedPacketMap&) = delete;
   ~QuicUnackedPacketMap();
 
   // Adds |serialized_packet| to the map and marks it as sent at |sent_time|.
@@ -231,8 +233,6 @@
 
   // If true, let session decides what to write.
   bool session_decides_what_to_write_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicUnackedPacketMap);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_utils.h b/net/third_party/quic/core/quic_utils.h
index 5038faf..ba539a8 100644
--- a/net/third_party/quic/core/quic_utils.h
+++ b/net/third_party/quic/core/quic_utils.h
@@ -22,6 +22,8 @@
 
 class QUIC_EXPORT_PRIVATE QuicUtils {
  public:
+  QuicUtils() = delete;
+
   // Returns the 64 bit FNV1a hash of the data.  See
   // http://www.isthe.com/chongo/tech/comp/fnv/index.html#FNV-param
   static uint64_t FNV1a_64_Hash(QuicStringPiece data);
@@ -87,9 +89,6 @@
   // Returns true if header with |first_byte| is considered as an IETF QUIC
   // packet header.
   static bool IsIetfPacketHeader(uint8_t first_byte);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicUtils);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/quic_write_blocked_list.h b/net/third_party/quic/core/quic_write_blocked_list.h
index 793f8a1..d592123b 100644
--- a/net/third_party/quic/core/quic_write_blocked_list.h
+++ b/net/third_party/quic/core/quic_write_blocked_list.h
@@ -25,6 +25,8 @@
 
  public:
   explicit QuicWriteBlockedList();
+  QuicWriteBlockedList(const QuicWriteBlockedList&) = delete;
+  QuicWriteBlockedList& operator=(const QuicWriteBlockedList&) = delete;
   ~QuicWriteBlockedList();
 
   bool HasWriteBlockedDataStreams() const {
@@ -340,8 +342,6 @@
 
   // Latched value of quic_use_static_stream_collection_in_write_blocked_list.
   const bool use_static_stream_collection_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicWriteBlockedList);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/stateless_rejector.h b/net/third_party/quic/core/stateless_rejector.h
index 925d63f..6904bb8 100644
--- a/net/third_party/quic/core/stateless_rejector.h
+++ b/net/third_party/quic/core/stateless_rejector.h
@@ -33,6 +33,8 @@
                     QuicByteCount chlo_packet_size,
                     const QuicSocketAddress& client_address,
                     const QuicSocketAddress& server_address);
+  StatelessRejector(const StatelessRejector&) = delete;
+  StatelessRejector& operator=(const StatelessRejector&) = delete;
 
   ~StatelessRejector();
 
@@ -110,8 +112,6 @@
   CryptoFramer crypto_framer_;
   QuicReferenceCountedPointer<QuicSignedServerConfig> signed_config_;
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters> params_;
-
-  DISALLOW_COPY_AND_ASSIGN(StatelessRejector);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/tls_client_handshaker.h b/net/third_party/quic/core/tls_client_handshaker.h
index c950b59..9a51d432 100644
--- a/net/third_party/quic/core/tls_client_handshaker.h
+++ b/net/third_party/quic/core/tls_client_handshaker.h
@@ -29,6 +29,8 @@
                       SSL_CTX* ssl_ctx,
                       std::unique_ptr<ProofVerifyContext> verify_context,
                       const QuicString& user_agent_id);
+  TlsClientHandshaker(const TlsClientHandshaker&) = delete;
+  TlsClientHandshaker& operator=(const TlsClientHandshaker&) = delete;
 
   ~TlsClientHandshaker() override;
 
@@ -119,8 +121,6 @@
   bool handshake_confirmed_ = false;
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters>
       crypto_negotiated_params_;
-
-  DISALLOW_COPY_AND_ASSIGN(TlsClientHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/tls_handshaker.cc b/net/third_party/quic/core/tls_handshaker.cc
index 08f105e..ad4f1838 100644
--- a/net/third_party/quic/core/tls_handshaker.cc
+++ b/net/third_party/quic/core/tls_handshaker.cc
@@ -38,11 +38,12 @@
     CHECK_LE(0, ssl_ex_data_index_handshaker_);
   }
 
+  SslIndexSingleton(const SslIndexSingleton&) = delete;
+  SslIndexSingleton& operator=(const SslIndexSingleton&) = delete;
+
   friend QuicSingletonFriend<SslIndexSingleton>;
 
   int ssl_ex_data_index_handshaker_;
-
-  DISALLOW_COPY_AND_ASSIGN(SslIndexSingleton);
 };
 
 }  // namespace
diff --git a/net/third_party/quic/core/tls_handshaker.h b/net/third_party/quic/core/tls_handshaker.h
index d6fdea4..4975a11 100644
--- a/net/third_party/quic/core/tls_handshaker.h
+++ b/net/third_party/quic/core/tls_handshaker.h
@@ -28,6 +28,8 @@
   TlsHandshaker(QuicCryptoStream* stream,
                 QuicSession* session,
                 SSL_CTX* ssl_ctx);
+  TlsHandshaker(const TlsHandshaker&) = delete;
+  TlsHandshaker& operator=(const TlsHandshaker&) = delete;
 
   ~TlsHandshaker() override;
 
@@ -81,8 +83,6 @@
   QuicSession* session_;
   QuicTlsAdapter bio_adapter_;
   bssl::UniquePtr<SSL> ssl_;
-
-  DISALLOW_COPY_AND_ASSIGN(TlsHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/core/tls_server_handshaker.h b/net/third_party/quic/core/tls_server_handshaker.h
index 13fbdbb..b0a57a5 100644
--- a/net/third_party/quic/core/tls_server_handshaker.h
+++ b/net/third_party/quic/core/tls_server_handshaker.h
@@ -27,6 +27,8 @@
                       QuicSession* session,
                       SSL_CTX* ssl_ctx,
                       ProofSource* proof_source);
+  TlsServerHandshaker(const TlsServerHandshaker&) = delete;
+  TlsServerHandshaker& operator=(const TlsServerHandshaker&) = delete;
 
   ~TlsServerHandshaker() override;
 
@@ -159,8 +161,6 @@
   bool handshake_confirmed_ = false;
   QuicReferenceCountedPointer<QuicCryptoNegotiatedParameters>
       crypto_negotiated_params_;
-
-  DISALLOW_COPY_AND_ASSIGN(TlsServerHandshaker);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/platform/api/quic_hostname_utils.h b/net/third_party/quic/platform/api/quic_hostname_utils.h
index 3fa4ffea..5f8d53fa 100644
--- a/net/third_party/quic/platform/api/quic_hostname_utils.h
+++ b/net/third_party/quic/platform/api/quic_hostname_utils.h
@@ -14,6 +14,8 @@
 
 class QUIC_EXPORT_PRIVATE QuicHostnameUtils {
  public:
+  QuicHostnameUtils() = delete;
+
   // Returns true if the sni is valid, false otherwise.
   //  (1) disallow IP addresses;
   //  (2) check that the hostname contains valid characters only; and
@@ -23,9 +25,6 @@
   // Convert hostname to lowercase and remove the trailing '.'.
   // WARNING: mutates |hostname| in place and returns |hostname|.
   static char* NormalizeHostname(char* hostname);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicHostnameUtils);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/platform/api/quic_lru_cache.h b/net/third_party/quic/platform/api/quic_lru_cache.h
index facedb2..771a2cff 100644
--- a/net/third_party/quic/platform/api/quic_lru_cache.h
+++ b/net/third_party/quic/platform/api/quic_lru_cache.h
@@ -19,6 +19,8 @@
 class QuicLRUCache {
  public:
   explicit QuicLRUCache(int64_t total_units) : impl_(total_units) {}
+  QuicLRUCache(const QuicLRUCache&) = delete;
+  QuicLRUCache& operator=(const QuicLRUCache&) = delete;
 
   // Inserts one unit of |key|, |value| pair to the cache. Cache takes ownership
   // of inserted |value|.
@@ -43,8 +45,6 @@
 
  private:
   QuicLRUCacheImpl<K, V> impl_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicLRUCache);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/platform/api/quic_mutex.h b/net/third_party/quic/platform/api/quic_mutex.h
index 280bed2..3637ce59 100644
--- a/net/third_party/quic/platform/api/quic_mutex.h
+++ b/net/third_party/quic/platform/api/quic_mutex.h
@@ -14,6 +14,8 @@
 class QUIC_EXPORT_PRIVATE LOCKABLE QuicMutex {
  public:
   QuicMutex() = default;
+  QuicMutex(const QuicMutex&) = delete;
+  QuicMutex& operator=(const QuicMutex&) = delete;
 
   // Block until this Mutex is free, then acquire it exclusively.
   void WriterLock() EXCLUSIVE_LOCK_FUNCTION();
@@ -34,8 +36,6 @@
 
  private:
   QuicLockImpl impl_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicMutex);
 };
 
 // A helper class that acquires the given QuicMutex shared lock while the
@@ -43,13 +43,13 @@
 class QUIC_EXPORT_PRIVATE SCOPED_LOCKABLE QuicReaderMutexLock {
  public:
   explicit QuicReaderMutexLock(QuicMutex* lock) SHARED_LOCK_FUNCTION(lock);
+  QuicReaderMutexLock(const QuicReaderMutexLock&) = delete;
+  QuicReaderMutexLock& operator=(const QuicReaderMutexLock&) = delete;
 
   ~QuicReaderMutexLock() UNLOCK_FUNCTION();
 
  private:
   QuicMutex* const lock_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicReaderMutexLock);
 };
 
 // A helper class that acquires the given QuicMutex exclusive lock while the
@@ -57,13 +57,13 @@
 class QUIC_EXPORT_PRIVATE SCOPED_LOCKABLE QuicWriterMutexLock {
  public:
   explicit QuicWriterMutexLock(QuicMutex* lock) EXCLUSIVE_LOCK_FUNCTION(lock);
+  QuicWriterMutexLock(const QuicWriterMutexLock&) = delete;
+  QuicWriterMutexLock& operator=(const QuicWriterMutexLock&) = delete;
 
   ~QuicWriterMutexLock() UNLOCK_FUNCTION();
 
  private:
   QuicMutex* const lock_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicWriterMutexLock);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/platform/api/quic_url_utils.h b/net/third_party/quic/platform/api/quic_url_utils.h
index 5580c39..65a1467 100644
--- a/net/third_party/quic/platform/api/quic_url_utils.h
+++ b/net/third_party/quic/platform/api/quic_url_utils.h
@@ -15,6 +15,8 @@
 
 class QUIC_EXPORT_PRIVATE QuicUrlUtils {
  public:
+  QuicUrlUtils() = delete;
+
   // Returns hostname, or empty std::string if missing.
   static QuicString HostName(QuicStringPiece url);
 
@@ -29,9 +31,6 @@
   static QuicString GetPushPromiseUrl(QuicStringPiece scheme,
                                       QuicStringPiece authority,
                                       QuicStringPiece path);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicUrlUtils);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/test_tools/limited_mtu_test_writer.h b/net/third_party/quic/test_tools/limited_mtu_test_writer.h
index 105252b..4b9734d 100644
--- a/net/third_party/quic/test_tools/limited_mtu_test_writer.h
+++ b/net/third_party/quic/test_tools/limited_mtu_test_writer.h
@@ -17,6 +17,8 @@
 class LimitedMtuTestWriter : public QuicPacketWriterWrapper {
  public:
   explicit LimitedMtuTestWriter(QuicByteCount mtu);
+  LimitedMtuTestWriter(const LimitedMtuTestWriter&) = delete;
+  LimitedMtuTestWriter& operator=(const LimitedMtuTestWriter&) = delete;
   ~LimitedMtuTestWriter() override;
 
   // Inherited from QuicPacketWriterWrapper.
@@ -28,8 +30,6 @@
 
  private:
   QuicByteCount mtu_;
-
-  DISALLOW_COPY_AND_ASSIGN(LimitedMtuTestWriter);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/mock_clock.h b/net/third_party/quic/test_tools/mock_clock.h
index 3eba3c8..8c94e9a 100644
--- a/net/third_party/quic/test_tools/mock_clock.h
+++ b/net/third_party/quic/test_tools/mock_clock.h
@@ -16,6 +16,8 @@
 class MockClock : public QuicClock {
  public:
   MockClock();
+  MockClock(const MockClock&) = delete;
+  MockClock& operator=(const MockClock&) = delete;
   ~MockClock() override;
 
   // QuicClock implementation:
@@ -31,8 +33,6 @@
 
  private:
   QuicTime now_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockClock);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/test_tools/mock_crypto_client_stream.cc b/net/third_party/quic/test_tools/mock_crypto_client_stream.cc
index 61b0209..ce761ec3 100644
--- a/net/third_party/quic/test_tools/mock_crypto_client_stream.cc
+++ b/net/third_party/quic/test_tools/mock_crypto_client_stream.cc
@@ -8,7 +8,7 @@
 #include "net/third_party/quic/core/crypto/null_encrypter.h"
 #include "net/third_party/quic/core/crypto/quic_decrypter.h"
 #include "net/third_party/quic/core/crypto/quic_encrypter.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/quic/test_tools/mock_decrypter.h"
 #include "net/third_party/quic/test_tools/mock_encrypter.h"
diff --git a/net/third_party/quic/test_tools/mock_crypto_client_stream.h b/net/third_party/quic/test_tools/mock_crypto_client_stream.h
index e258be1..61cbd42 100644
--- a/net/third_party/quic/test_tools/mock_crypto_client_stream.h
+++ b/net/third_party/quic/test_tools/mock_crypto_client_stream.h
@@ -11,10 +11,10 @@
 #include "net/quic/chromium/crypto/proof_verifier_chromium.h"
 #include "net/third_party/quic/core/crypto/crypto_handshake.h"
 #include "net/third_party/quic/core/crypto/crypto_protocol.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session_base.h"
 #include "net/third_party/quic/core/quic_crypto_client_stream.h"
 #include "net/third_party/quic/core/quic_server_id.h"
 #include "net/third_party/quic/core/quic_session.h"
-#include "net/third_party/quic/core/quic_spdy_client_session_base.h"
 
 namespace quic {
 
diff --git a/net/third_party/quic/test_tools/mock_quic_client_promised_info.h b/net/third_party/quic/test_tools/mock_quic_client_promised_info.h
index e52acd06..20cfbac 100644
--- a/net/third_party/quic/test_tools/mock_quic_client_promised_info.h
+++ b/net/third_party/quic/test_tools/mock_quic_client_promised_info.h
@@ -8,7 +8,7 @@
 #include <string>
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "testing/gmock/include/gmock/gmock.h"
 
diff --git a/net/third_party/quic/test_tools/mock_quic_dispatcher.h b/net/third_party/quic/test_tools/mock_quic_dispatcher.h
index d6637f9..03096371 100644
--- a/net/third_party/quic/test_tools/mock_quic_dispatcher.h
+++ b/net/third_party/quic/test_tools/mock_quic_dispatcher.h
@@ -26,6 +26,8 @@
       std::unique_ptr<QuicCryptoServerStream::Helper> session_helper,
       std::unique_ptr<QuicAlarmFactory> alarm_factory,
       QuicSimpleServerBackend* quic_simple_server_backend);
+  MockQuicDispatcher(const MockQuicDispatcher&) = delete;
+  MockQuicDispatcher& operator=(const MockQuicDispatcher&) = delete;
 
   ~MockQuicDispatcher() override;
 
@@ -33,9 +35,6 @@
                void(const QuicSocketAddress& server_address,
                     const QuicSocketAddress& client_address,
                     const QuicReceivedPacket& packet));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicDispatcher);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/mock_quic_session_visitor.h b/net/third_party/quic/test_tools/mock_quic_session_visitor.h
index 3ac911b3..b207375c 100644
--- a/net/third_party/quic/test_tools/mock_quic_session_visitor.h
+++ b/net/third_party/quic/test_tools/mock_quic_session_visitor.h
@@ -16,6 +16,8 @@
 class MockQuicSessionVisitor : public QuicTimeWaitListManager::Visitor {
  public:
   MockQuicSessionVisitor();
+  MockQuicSessionVisitor(const MockQuicSessionVisitor&) = delete;
+  MockQuicSessionVisitor& operator=(const MockQuicSessionVisitor&) = delete;
   ~MockQuicSessionVisitor() override;
   MOCK_METHOD3(OnConnectionClosed,
                void(QuicConnectionId connection_id,
@@ -26,14 +28,15 @@
   MOCK_METHOD1(OnRstStreamReceived, void(const QuicRstStreamFrame& frame));
   MOCK_METHOD1(OnConnectionAddedToTimeWaitList,
                void(QuicConnectionId connection_id));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSessionVisitor);
 };
 
 class MockQuicCryptoServerStreamHelper : public QuicCryptoServerStream::Helper {
  public:
   MockQuicCryptoServerStreamHelper();
+  MockQuicCryptoServerStreamHelper(const MockQuicCryptoServerStreamHelper&) =
+      delete;
+  MockQuicCryptoServerStreamHelper& operator=(
+      const MockQuicCryptoServerStreamHelper&) = delete;
   ~MockQuicCryptoServerStreamHelper() override;
   MOCK_CONST_METHOD1(GenerateConnectionIdForReject,
                      QuicConnectionId(QuicConnectionId connection_id));
@@ -43,9 +46,6 @@
                           const QuicSocketAddress& peer_address,
                           const QuicSocketAddress& self_address,
                           QuicString* error_details));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicCryptoServerStreamHelper);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/mock_quic_spdy_client_stream.h b/net/third_party/quic/test_tools/mock_quic_spdy_client_stream.h
index e1c8531..6a0a4d89 100644
--- a/net/third_party/quic/test_tools/mock_quic_spdy_client_stream.h
+++ b/net/third_party/quic/test_tools/mock_quic_spdy_client_stream.h
@@ -6,9 +6,9 @@
 #define NET_THIRD_PARTY_QUIC_TEST_TOOLS_MOCK_QUIC_SPDY_CLIENT_STREAM_H_
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_header_list.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "testing/gmock/include/gmock/gmock.h"
 
 namespace quic {
diff --git a/net/third_party/quic/test_tools/mock_random.h b/net/third_party/quic/test_tools/mock_random.h
index 090bab6a..727c8240 100644
--- a/net/third_party/quic/test_tools/mock_random.h
+++ b/net/third_party/quic/test_tools/mock_random.h
@@ -17,6 +17,8 @@
   // Initializes base_ to 0xDEADBEEF.
   MockRandom();
   explicit MockRandom(uint32_t base);
+  MockRandom(const MockRandom&) = delete;
+  MockRandom& operator=(const MockRandom&) = delete;
 
   // QuicRandom:
   // Fills the |data| buffer with a repeating byte, initially 'r'.
@@ -31,8 +33,6 @@
  private:
   uint32_t base_;
   uint8_t increment_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockRandom);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/packet_dropping_test_writer.h b/net/third_party/quic/test_tools/packet_dropping_test_writer.h
index a9c29c3..e4c47ed 100644
--- a/net/third_party/quic/test_tools/packet_dropping_test_writer.h
+++ b/net/third_party/quic/test_tools/packet_dropping_test_writer.h
@@ -37,6 +37,8 @@
   };
 
   PacketDroppingTestWriter();
+  PacketDroppingTestWriter(const PacketDroppingTestWriter&) = delete;
+  PacketDroppingTestWriter& operator=(const PacketDroppingTestWriter&) = delete;
 
   ~PacketDroppingTestWriter() override;
 
@@ -138,6 +140,8 @@
                  const QuicSocketAddress& peer_address,
                  std::unique_ptr<PerPacketOptions> options,
                  QuicTime send_time);
+    DelayedWrite(const DelayedWrite&) = delete;
+    DelayedWrite& operator=(const DelayedWrite&) = delete;
     // TODO(rtenneti): on windows RValue reference gives errors.
     DelayedWrite(DelayedWrite&& other);
     // TODO(rtenneti): on windows RValue reference gives errors.
@@ -149,9 +153,6 @@
     const QuicSocketAddress peer_address;
     std::unique_ptr<PerPacketOptions> options;
     QuicTime send_time;
-
-   private:
-    DISALLOW_COPY_AND_ASSIGN(DelayedWrite);
   };
 
   typedef std::list<DelayedWrite> DelayedPacketList;
@@ -175,8 +176,6 @@
   QuicBandwidth fake_bandwidth_ GUARDED_BY(config_mutex_);
   QuicByteCount buffer_size_ GUARDED_BY(config_mutex_);
   int32_t num_consecutive_packet_lost_ GUARDED_BY(config_mutex_);
-
-  DISALLOW_COPY_AND_ASSIGN(PacketDroppingTestWriter);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_buffered_packet_store_peer.h b/net/third_party/quic/test_tools/quic_buffered_packet_store_peer.h
index f2ee116..5450b48 100644
--- a/net/third_party/quic/test_tools/quic_buffered_packet_store_peer.h
+++ b/net/third_party/quic/test_tools/quic_buffered_packet_store_peer.h
@@ -18,12 +18,11 @@
 
 class QuicBufferedPacketStorePeer {
  public:
+  QuicBufferedPacketStorePeer() = delete;
+
   static QuicAlarm* expiration_alarm(QuicBufferedPacketStore* store);
 
   static void set_clock(QuicBufferedPacketStore* store, const QuicClock* clock);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicBufferedPacketStorePeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_client_peer.h b/net/third_party/quic/test_tools/quic_client_peer.h
index c0061062..59d2d6e 100644
--- a/net/third_party/quic/test_tools/quic_client_peer.h
+++ b/net/third_party/quic/test_tools/quic_client_peer.h
@@ -16,13 +16,12 @@
 
 class QuicClientPeer {
  public:
+  QuicClientPeer() = delete;
+
   static bool CreateUDPSocketAndBind(QuicClient* client);
   static void CleanUpUDPSocket(QuicClient* client, int fd);
   static void SetClientPort(QuicClient* client, int port);
   static void SetWriter(QuicClient* client, QuicPacketWriter* writer);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicClientPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_client_promised_info_peer.h b/net/third_party/quic/test_tools/quic_client_promised_info_peer.h
index d169531..693904e 100644
--- a/net/third_party/quic/test_tools/quic_client_promised_info_peer.h
+++ b/net/third_party/quic/test_tools/quic_client_promised_info_peer.h
@@ -6,17 +6,16 @@
 #define NET_THIRD_PARTY_QUIC_TEST_TOOLS_QUIC_CLIENT_PROMISED_INFO_PEER_H_
 
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_client_promised_info.h"
+#include "net/third_party/quic/core/http/quic_client_promised_info.h"
 
 namespace quic {
 namespace test {
 
 class QuicClientPromisedInfoPeer {
  public:
-  static QuicAlarm* GetAlarm(QuicClientPromisedInfo* promised_stream);
+  QuicClientPromisedInfoPeer() = delete;
 
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicClientPromisedInfoPeer);
+  static QuicAlarm* GetAlarm(QuicClientPromisedInfo* promised_stream);
 };
 }  // namespace test
 }  // namespace quic
diff --git a/net/third_party/quic/test_tools/quic_config_peer.h b/net/third_party/quic/test_tools/quic_config_peer.h
index 33a802d..2164fd30 100644
--- a/net/third_party/quic/test_tools/quic_config_peer.h
+++ b/net/third_party/quic/test_tools/quic_config_peer.h
@@ -18,6 +18,8 @@
 
 class QuicConfigPeer {
  public:
+  QuicConfigPeer() = delete;
+
   static void SetReceivedInitialStreamFlowControlWindow(QuicConfig* config,
                                                         uint32_t window_bytes);
 
@@ -40,9 +42,6 @@
 
   static void SetReceivedStatelessResetToken(QuicConfig* config,
                                              QuicUint128 token);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicConfigPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_connection_peer.h b/net/third_party/quic/test_tools/quic_connection_peer.h
index fe43104..b5ad7ae 100644
--- a/net/third_party/quic/test_tools/quic_connection_peer.h
+++ b/net/third_party/quic/test_tools/quic_connection_peer.h
@@ -31,6 +31,8 @@
 // Peer to make public a number of otherwise private QuicConnection methods.
 class QuicConnectionPeer {
  public:
+  QuicConnectionPeer() = delete;
+
   static void SendAck(QuicConnection* connection);
 
   static void SetSendAlgorithm(QuicConnection* connection,
@@ -128,9 +130,6 @@
                                    QuicPacketCount max_tracked_packets);
   static void SetSessionDecidesWhatToWrite(QuicConnection* connection);
   static void SetNegotiatedVersion(QuicConnection* connection);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicConnectionPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_dispatcher_peer.h b/net/third_party/quic/test_tools/quic_dispatcher_peer.h
index b97aa8f2..7494d32 100644
--- a/net/third_party/quic/test_tools/quic_dispatcher_peer.h
+++ b/net/third_party/quic/test_tools/quic_dispatcher_peer.h
@@ -16,6 +16,8 @@
 
 class QuicDispatcherPeer {
  public:
+  QuicDispatcherPeer() = delete;
+
   static void SetTimeWaitListManager(
       QuicDispatcher* dispatcher,
       QuicTimeWaitListManager* time_wait_list_manager);
@@ -60,9 +62,6 @@
   static void RestorePerPacketContext(
       QuicDispatcher* dispatcher,
       std::unique_ptr<QuicDispatcher::PerPacketContext>);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicDispatcherPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_flow_controller_peer.h b/net/third_party/quic/test_tools/quic_flow_controller_peer.h
index f423f36..2f99b28 100644
--- a/net/third_party/quic/test_tools/quic_flow_controller_peer.h
+++ b/net/third_party/quic/test_tools/quic_flow_controller_peer.h
@@ -16,6 +16,8 @@
 
 class QuicFlowControllerPeer {
  public:
+  QuicFlowControllerPeer() = delete;
+
   static void SetSendWindowOffset(QuicFlowController* flow_controller,
                                   QuicStreamOffset offset);
 
@@ -36,9 +38,6 @@
 
   static QuicByteCount WindowUpdateThreshold(
       QuicFlowController* flow_controller);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicFlowControllerPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_framer_peer.h b/net/third_party/quic/test_tools/quic_framer_peer.h
index 71255db..d7903bb 100644
--- a/net/third_party/quic/test_tools/quic_framer_peer.h
+++ b/net/third_party/quic/test_tools/quic_framer_peer.h
@@ -16,6 +16,8 @@
 
 class QuicFramerPeer {
  public:
+  QuicFramerPeer() = delete;
+
   static QuicPacketNumber CalculatePacketNumberFromWire(
       QuicFramer* framer,
       QuicPacketNumberLength packet_number_length,
@@ -147,9 +149,6 @@
                                    const QuicFrame& frame,
                                    bool last_frame_in_packet,
                                    QuicPacketNumberLength packet_number_length);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicFramerPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_packet_creator_peer.h b/net/third_party/quic/test_tools/quic_packet_creator_peer.h
index a94cf8d0..35a4f869 100644
--- a/net/third_party/quic/test_tools/quic_packet_creator_peer.h
+++ b/net/third_party/quic/test_tools/quic_packet_creator_peer.h
@@ -16,6 +16,8 @@
 
 class QuicPacketCreatorPeer {
  public:
+  QuicPacketCreatorPeer() = delete;
+
   static bool SendVersionInPacket(QuicPacketCreator* creator);
 
   static void SetSendVersionInPacket(QuicPacketCreator* creator,
@@ -43,9 +45,6 @@
       QuicPacketCreator* creator);
   static EncryptionLevel GetEncryptionLevel(QuicPacketCreator* creator);
   static QuicFramer* framer(QuicPacketCreator* creator);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketCreatorPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_packet_generator_peer.h b/net/third_party/quic/test_tools/quic_packet_generator_peer.h
index fbf853b1..d9f22a5 100644
--- a/net/third_party/quic/test_tools/quic_packet_generator_peer.h
+++ b/net/third_party/quic/test_tools/quic_packet_generator_peer.h
@@ -17,10 +17,9 @@
 
 class QuicPacketGeneratorPeer {
  public:
-  static QuicPacketCreator* GetPacketCreator(QuicPacketGenerator* generator);
+  QuicPacketGeneratorPeer() = delete;
 
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicPacketGeneratorPeer);
+  static QuicPacketCreator* GetPacketCreator(QuicPacketGenerator* generator);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h b/net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h
index b75b195a..4aa02b8 100644
--- a/net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h
+++ b/net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h
@@ -17,6 +17,8 @@
 
 class QuicSentPacketManagerPeer {
  public:
+  QuicSentPacketManagerPeer() = delete;
+
   static size_t GetMaxTailLossProbes(
       QuicSentPacketManager* sent_packet_manager);
 
@@ -104,9 +106,6 @@
 
   static void SetNextPacedPacketTime(QuicSentPacketManager* sent_packet_manager,
                                      QuicTime time);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSentPacketManagerPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_server_peer.h b/net/third_party/quic/test_tools/quic_server_peer.h
index 5759a3b..bf4bff5 100644
--- a/net/third_party/quic/test_tools/quic_server_peer.h
+++ b/net/third_party/quic/test_tools/quic_server_peer.h
@@ -17,12 +17,11 @@
 
 class QuicServerPeer {
  public:
+  QuicServerPeer() = delete;
+
   static bool SetSmallSocket(QuicServer* server);
   static QuicDispatcher* GetDispatcher(QuicServer* server);
   static void SetReader(QuicServer* server, QuicPacketReader* reader);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicServerPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_session_peer.h b/net/third_party/quic/test_tools/quic_session_peer.h
index a3a57310..e0e510c 100644
--- a/net/third_party/quic/test_tools/quic_session_peer.h
+++ b/net/third_party/quic/test_tools/quic_session_peer.h
@@ -25,6 +25,8 @@
 
 class QuicSessionPeer {
  public:
+  QuicSessionPeer() = delete;
+
   static QuicStreamId GetNextOutgoingStreamId(QuicSession* session);
   static void SetNextOutgoingStreamId(QuicSession* session, QuicStreamId id);
   static void SetMaxOpenIncomingStreams(QuicSession* session,
@@ -55,9 +57,6 @@
 
   static QuicStream* GetStream(QuicSession* session, QuicStreamId id);
   static bool IsStreamWriteBlocked(QuicSession* session, QuicStreamId id);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSessionPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_spdy_session_peer.cc b/net/third_party/quic/test_tools/quic_spdy_session_peer.cc
index ae763a14..60b0359 100644
--- a/net/third_party/quic/test_tools/quic_spdy_session_peer.cc
+++ b/net/third_party/quic/test_tools/quic_spdy_session_peer.cc
@@ -4,7 +4,7 @@
 
 #include "net/third_party/quic/test_tools/quic_spdy_session_peer.h"
 
-#include "net/third_party/quic/core/quic_spdy_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 
 namespace quic {
 namespace test {
diff --git a/net/third_party/quic/test_tools/quic_spdy_session_peer.h b/net/third_party/quic/test_tools/quic_spdy_session_peer.h
index c98986b8..488300d 100644
--- a/net/third_party/quic/test_tools/quic_spdy_session_peer.h
+++ b/net/third_party/quic/test_tools/quic_spdy_session_peer.h
@@ -20,6 +20,8 @@
 
 class QuicSpdySessionPeer {
  public:
+  QuicSpdySessionPeer() = delete;
+
   static QuicHeadersStream* GetHeadersStream(QuicSpdySession* session);
   static void SetHeadersStream(QuicSpdySession* session,
                                QuicHeadersStream* headers_stream);
@@ -54,9 +56,6 @@
   static QuicStreamId GetNthServerInitiatedStreamId(
       const QuicSpdySession& session,
       int n);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdySessionPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_spdy_stream_peer.cc b/net/third_party/quic/test_tools/quic_spdy_stream_peer.cc
deleted file mode 100644
index a491cf5..0000000
--- a/net/third_party/quic/test_tools/quic_spdy_stream_peer.cc
+++ /dev/null
@@ -1,7 +0,0 @@
-// Copyright 2013 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "net/third_party/quic/test_tools/quic_spdy_stream_peer.h"
-
-#include "net/third_party/quic/core/quic_spdy_stream.h"
diff --git a/net/third_party/quic/test_tools/quic_spdy_stream_peer.h b/net/third_party/quic/test_tools/quic_spdy_stream_peer.h
deleted file mode 100644
index 88cb151e..0000000
--- a/net/third_party/quic/test_tools/quic_spdy_stream_peer.h
+++ /dev/null
@@ -1,24 +0,0 @@
-// Copyright 2013 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef NET_THIRD_PARTY_QUIC_TEST_TOOLS_QUIC_SPDY_STREAM_PEER_H_
-#define NET_THIRD_PARTY_QUIC_TEST_TOOLS_QUIC_SPDY_STREAM_PEER_H_
-
-#include "base/macros.h"
-
-namespace quic {
-
-namespace test {
-
-class QuicSpdyStreamPeer {
- public:
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyStreamPeer);
-};
-
-}  // namespace test
-
-}  // namespace quic
-
-#endif  // NET_THIRD_PARTY_QUIC_TEST_TOOLS_QUIC_SPDY_STREAM_PEER_H_
diff --git a/net/third_party/quic/test_tools/quic_stream_peer.h b/net/third_party/quic/test_tools/quic_stream_peer.h
index fad55440..407aeb7 100644
--- a/net/third_party/quic/test_tools/quic_stream_peer.h
+++ b/net/third_party/quic/test_tools/quic_stream_peer.h
@@ -22,6 +22,8 @@
 
 class QuicStreamPeer {
  public:
+  QuicStreamPeer() = delete;
+
   static void SetWriteSideClosed(bool value, QuicStream* stream);
   static void SetStreamBytesWritten(QuicStreamOffset stream_bytes_written,
                                     QuicStream* stream);
@@ -49,9 +51,6 @@
   static void set_ack_listener(
       QuicStream* stream,
       QuicReferenceCountedPointer<QuicAckListenerInterface> ack_listener);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_stream_sequencer_buffer_peer.h b/net/third_party/quic/test_tools/quic_stream_sequencer_buffer_peer.h
index 7937766..7e4def1 100644
--- a/net/third_party/quic/test_tools/quic_stream_sequencer_buffer_peer.h
+++ b/net/third_party/quic/test_tools/quic_stream_sequencer_buffer_peer.h
@@ -14,6 +14,9 @@
 class QuicStreamSequencerBufferPeer {
  public:
   explicit QuicStreamSequencerBufferPeer(QuicStreamSequencerBuffer* buffer);
+  QuicStreamSequencerBufferPeer(const QuicStreamSequencerBufferPeer&) = delete;
+  QuicStreamSequencerBufferPeer& operator=(
+      const QuicStreamSequencerBufferPeer&) = delete;
 
   // Read from this buffer_ into the given destination buffer_ up to the
   // size of the destination. Returns the number of bytes read. Reading from
@@ -52,7 +55,6 @@
 
  private:
   QuicStreamSequencerBuffer* buffer_;
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamSequencerBufferPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_stream_sequencer_peer.h b/net/third_party/quic/test_tools/quic_stream_sequencer_peer.h
index aa97588d..67ef054 100644
--- a/net/third_party/quic/test_tools/quic_stream_sequencer_peer.h
+++ b/net/third_party/quic/test_tools/quic_stream_sequencer_peer.h
@@ -16,6 +16,8 @@
 
 class QuicStreamSequencerPeer {
  public:
+  QuicStreamSequencerPeer() = delete;
+
   static size_t GetNumBufferedBytes(QuicStreamSequencer* sequencer);
 
   static QuicStreamOffset GetCloseOffset(QuicStreamSequencer* sequencer);
@@ -24,9 +26,6 @@
 
   static void SetFrameBufferTotalBytesRead(QuicStreamSequencer* sequencer,
                                            QuicStreamOffset total_bytes_read);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicStreamSequencerPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h b/net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h
index e398b7d..355237a 100644
--- a/net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h
+++ b/net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h
@@ -18,6 +18,8 @@
 
 class QuicSustainedBandwidthRecorderPeer {
  public:
+  QuicSustainedBandwidthRecorderPeer() = delete;
+
   static void SetBandwidthEstimate(
       QuicSustainedBandwidthRecorder* bandwidth_recorder,
       int32_t bandwidth_estimate_kbytes_per_second);
@@ -26,9 +28,6 @@
       QuicSustainedBandwidthRecorder* bandwidth_recorder,
       int32_t max_bandwidth_estimate_kbytes_per_second,
       int32_t max_bandwidth_timestamp);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(QuicSustainedBandwidthRecorderPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_test_client.cc b/net/third_party/quic/test_tools/quic_test_client.cc
index 38f3f3c..10bb8ea2 100644
--- a/net/third_party/quic/test_tools/quic_test_client.cc
+++ b/net/third_party/quic/test_tools/quic_test_client.cc
@@ -9,12 +9,12 @@
 #include <vector>
 
 #include "net/third_party/quic/core/crypto/proof_verifier.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_epoll_connection_helper.h"
 #include "net/third_party/quic/core/quic_packet_writer_wrapper.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/test_tools/quic_test_client.h b/net/third_party/quic/test_tools/quic_test_client.h
index 4c09e07..903c7e7 100644
--- a/net/third_party/quic/test_tools/quic_test_client.h
+++ b/net/third_party/quic/test_tools/quic_test_client.h
@@ -49,6 +49,8 @@
                      const ParsedQuicVersionVector& supported_versions,
                      net::EpollServer* epoll_server,
                      std::unique_ptr<ProofVerifier> proof_verifier);
+  MockableQuicClient(const MockableQuicClient&) = delete;
+  MockableQuicClient& operator=(const MockableQuicClient&) = delete;
 
   ~MockableQuicClient() override;
 
@@ -69,8 +71,6 @@
  private:
   QuicConnectionId override_connection_id_;  // ConnectionId to use, if nonzero
   CachedNetworkParameters cached_network_paramaters_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockableQuicClient);
 };
 
 // A toy QUIC client used for testing.
@@ -318,6 +318,8 @@
 
  protected:
   QuicTestClient();
+  QuicTestClient(const QuicTestClient&) = delete;
+  QuicTestClient& operator=(const QuicTestClient&) = delete;
 
  private:
   class TestClientDataToResend : public QuicClient::QuicDataToResend {
@@ -395,8 +397,6 @@
   // logic which extracts the SNI from the request URL.
   bool override_sni_set_ = false;
   QuicString override_sni_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicTestClient);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/quic_test_utils.h b/net/third_party/quic/test_tools/quic_test_utils.h
index ef0d0fb..231d60ba 100644
--- a/net/third_party/quic/test_tools/quic_test_utils.h
+++ b/net/third_party/quic/test_tools/quic_test_utils.h
@@ -18,12 +18,12 @@
 #include "net/test/gtest_util.h"
 #include "net/third_party/quic/core/congestion_control/loss_detection_interface.h"
 #include "net/third_party/quic/core/congestion_control/send_algorithm_interface.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_connection_close_delegate_interface.h"
 #include "net/third_party/quic/core/quic_framer.h"
 #include "net/third_party/quic/core/quic_sent_packet_manager.h"
-#include "net/third_party/quic/core/quic_server_session_base.h"
 #include "net/third_party/quic/core/quic_simple_buffer_allocator.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
@@ -218,6 +218,8 @@
 class SimpleRandom : public QuicRandom {
  public:
   SimpleRandom() : seed_(0) {}
+  SimpleRandom(const SimpleRandom&) = delete;
+  SimpleRandom& operator=(const SimpleRandom&) = delete;
   ~SimpleRandom() override {}
 
   // Returns a random number in the range [0, kuint64max].
@@ -229,13 +231,13 @@
 
  private:
   uint64_t seed_;
-
-  DISALLOW_COPY_AND_ASSIGN(SimpleRandom);
 };
 
 class MockFramerVisitor : public QuicFramerVisitorInterface {
  public:
   MockFramerVisitor();
+  MockFramerVisitor(const MockFramerVisitor&) = delete;
+  MockFramerVisitor& operator=(const MockFramerVisitor&) = delete;
   ~MockFramerVisitor() override;
 
   MOCK_METHOD1(OnError, void(QuicFramer* framer));
@@ -278,14 +280,13 @@
   MOCK_CONST_METHOD1(IsValidStatelessResetToken, bool(QuicUint128));
   MOCK_METHOD1(OnAuthenticatedIetfStatelessResetPacket,
                void(const QuicIetfStatelessResetPacket&));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockFramerVisitor);
 };
 
 class NoOpFramerVisitor : public QuicFramerVisitorInterface {
  public:
   NoOpFramerVisitor() {}
+  NoOpFramerVisitor(const NoOpFramerVisitor&) = delete;
+  NoOpFramerVisitor& operator=(const NoOpFramerVisitor&) = delete;
 
   void OnError(QuicFramer* framer) override {}
   void OnPacket() override {}
@@ -322,14 +323,14 @@
   bool IsValidStatelessResetToken(QuicUint128 token) const override;
   void OnAuthenticatedIetfStatelessResetPacket(
       const QuicIetfStatelessResetPacket& packet) override {}
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(NoOpFramerVisitor);
 };
 
 class MockQuicConnectionVisitor : public QuicConnectionVisitorInterface {
  public:
   MockQuicConnectionVisitor();
+  MockQuicConnectionVisitor(const MockQuicConnectionVisitor&) = delete;
+  MockQuicConnectionVisitor& operator=(const MockQuicConnectionVisitor&) =
+      delete;
   ~MockQuicConnectionVisitor() override;
 
   MOCK_METHOD1(OnStreamFrame, void(const QuicStreamFrame& frame));
@@ -360,14 +361,13 @@
   MOCK_METHOD0(SendPing, void());
   MOCK_CONST_METHOD0(AllowSelfAddressChange, bool());
   MOCK_METHOD0(OnForwardProgressConfirmed, void());
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicConnectionVisitor);
 };
 
 class MockQuicConnectionHelper : public QuicConnectionHelperInterface {
  public:
   MockQuicConnectionHelper();
+  MockQuicConnectionHelper(const MockQuicConnectionHelper&) = delete;
+  MockQuicConnectionHelper& operator=(const MockQuicConnectionHelper&) = delete;
   ~MockQuicConnectionHelper() override;
   const QuicClock* GetClock() const override;
   QuicRandom* GetRandomGenerator() override;
@@ -378,8 +378,6 @@
   MockClock clock_;
   MockRandom random_generator_;
   SimpleBufferAllocator buffer_allocator_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicConnectionHelper);
 };
 
 class MockAlarmFactory : public QuicAlarmFactory {
@@ -437,6 +435,8 @@
                      MockAlarmFactory* alarm_factory,
                      Perspective perspective,
                      const ParsedQuicVersionVector& supported_versions);
+  MockQuicConnection(const MockQuicConnection&) = delete;
+  MockQuicConnection& operator=(const MockQuicConnection&) = delete;
 
   ~MockQuicConnection() override;
 
@@ -505,9 +505,6 @@
     return QuicConnection::SendConnectivityProbingPacket(probing_writer,
                                                          peer_address);
   }
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicConnection);
 };
 
 class PacketSavingConnection : public MockQuicConnection {
@@ -520,15 +517,14 @@
                          MockAlarmFactory* alarm_factory,
                          Perspective perspective,
                          const ParsedQuicVersionVector& supported_versions);
+  PacketSavingConnection(const PacketSavingConnection&) = delete;
+  PacketSavingConnection& operator=(const PacketSavingConnection&) = delete;
 
   ~PacketSavingConnection() override;
 
   void SendOrQueuePacket(SerializedPacket* packet) override;
 
   std::vector<std::unique_ptr<QuicEncryptedPacket>> encrypted_packets_;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(PacketSavingConnection);
 };
 
 class MockQuicSession : public QuicSession {
@@ -538,6 +534,8 @@
 
   // Takes ownership of |connection|.
   explicit MockQuicSession(QuicConnection* connection);
+  MockQuicSession(const MockQuicSession&) = delete;
+  MockQuicSession& operator=(const MockQuicSession&) = delete;
   ~MockQuicSession() override;
 
   QuicCryptoStream* GetMutableCryptoStream() override;
@@ -584,8 +582,6 @@
 
  private:
   std::unique_ptr<QuicCryptoStream> crypto_stream_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSession);
 };
 
 class MockQuicCryptoStream : public QuicCryptoStream {
@@ -613,6 +609,8 @@
   // Takes ownership of |connection|.
   MockQuicSpdySession(QuicConnection* connection,
                       bool create_mock_crypto_stream);
+  MockQuicSpdySession(const MockQuicSpdySession&) = delete;
+  MockQuicSpdySession& operator=(const MockQuicSpdySession&) = delete;
   ~MockQuicSpdySession() override;
 
   QuicCryptoStream* GetMutableCryptoStream() override;
@@ -693,8 +691,6 @@
  private:
   std::unique_ptr<QuicCryptoStream> crypto_stream_;
   spdy::SpdyHeaderBlock write_headers_;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSpdySession);
 };
 
 class TestQuicSpdyServerSession : public QuicServerSessionBase {
@@ -704,6 +700,9 @@
                             const QuicConfig& config,
                             const QuicCryptoServerConfig* crypto_config,
                             QuicCompressedCertsCache* compressed_certs_cache);
+  TestQuicSpdyServerSession(const TestQuicSpdyServerSession&) = delete;
+  TestQuicSpdyServerSession& operator=(const TestQuicSpdyServerSession&) =
+      delete;
   ~TestQuicSpdyServerSession() override;
 
   MOCK_METHOD1(CreateIncomingDynamicStream, QuicSpdyStream*(QuicStreamId id));
@@ -724,8 +723,6 @@
  private:
   MockQuicSessionVisitor visitor_;
   MockQuicCryptoServerStreamHelper helper_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestQuicSpdyServerSession);
 };
 
 // A test implementation of QuicClientPushPromiseIndex::Delegate.
@@ -756,6 +753,9 @@
                             const QuicConfig& config,
                             const QuicServerId& server_id,
                             QuicCryptoClientConfig* crypto_config);
+  TestQuicSpdyClientSession(const TestQuicSpdyClientSession&) = delete;
+  TestQuicSpdyClientSession& operator=(const TestQuicSpdyClientSession&) =
+      delete;
   ~TestQuicSpdyClientSession() override;
 
   bool IsAuthorized(const QuicString& authority) override;
@@ -780,13 +780,13 @@
  private:
   std::unique_ptr<QuicCryptoClientStream> crypto_stream_;
   QuicClientPushPromiseIndex push_promise_index_;
-
-  DISALLOW_COPY_AND_ASSIGN(TestQuicSpdyClientSession);
 };
 
 class MockPacketWriter : public QuicPacketWriter {
  public:
   MockPacketWriter();
+  MockPacketWriter(const MockPacketWriter&) = delete;
+  MockPacketWriter& operator=(const MockPacketWriter&) = delete;
   ~MockPacketWriter() override;
 
   MOCK_METHOD5(WritePacket,
@@ -804,14 +804,13 @@
   MOCK_CONST_METHOD0(IsBatchMode, bool());
   MOCK_CONST_METHOD0(GetNextWriteLocation, char*());
   MOCK_METHOD0(Flush, WriteResult());
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockPacketWriter);
 };
 
 class MockSendAlgorithm : public SendAlgorithmInterface {
  public:
   MockSendAlgorithm();
+  MockSendAlgorithm(const MockSendAlgorithm&) = delete;
+  MockSendAlgorithm& operator=(const MockSendAlgorithm&) = delete;
   ~MockSendAlgorithm() override;
 
   MOCK_METHOD2(SetFromConfig,
@@ -850,14 +849,13 @@
   MOCK_CONST_METHOD0(GetCongestionControlType, CongestionControlType());
   MOCK_METHOD2(AdjustNetworkParameters, void(QuicBandwidth, QuicTime::Delta));
   MOCK_METHOD1(OnApplicationLimited, void(QuicByteCount));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockSendAlgorithm);
 };
 
 class MockLossAlgorithm : public LossDetectionInterface {
  public:
   MockLossAlgorithm();
+  MockLossAlgorithm(const MockLossAlgorithm&) = delete;
+  MockLossAlgorithm& operator=(const MockLossAlgorithm&) = delete;
   ~MockLossAlgorithm() override;
 
   MOCK_CONST_METHOD0(GetLossDetectionType, LossDetectionType());
@@ -873,14 +871,13 @@
                     QuicTime,
                     const RttStats&,
                     QuicPacketNumber));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockLossAlgorithm);
 };
 
 class MockAckListener : public QuicAckListenerInterface {
  public:
   MockAckListener();
+  MockAckListener(const MockAckListener&) = delete;
+  MockAckListener& operator=(const MockAckListener&) = delete;
 
   MOCK_METHOD2(OnPacketAcked,
                void(int acked_bytes, QuicTime::Delta ack_delay_time));
@@ -890,22 +887,18 @@
  protected:
   // Object is ref counted.
   ~MockAckListener() override;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockAckListener);
 };
 
 class MockNetworkChangeVisitor
     : public QuicSentPacketManager::NetworkChangeVisitor {
  public:
   MockNetworkChangeVisitor();
+  MockNetworkChangeVisitor(const MockNetworkChangeVisitor&) = delete;
+  MockNetworkChangeVisitor& operator=(const MockNetworkChangeVisitor&) = delete;
   ~MockNetworkChangeVisitor() override;
 
   MOCK_METHOD0(OnCongestionChange, void());
   MOCK_METHOD1(OnPathMtuIncreased, void(QuicPacketLength));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockNetworkChangeVisitor);
 };
 
 class MockQuicConnectionDebugVisitor : public QuicConnectionDebugVisitor {
@@ -990,6 +983,9 @@
 class MockPacketCreatorDelegate : public QuicPacketCreator::DelegateInterface {
  public:
   MockPacketCreatorDelegate();
+  MockPacketCreatorDelegate(const MockPacketCreatorDelegate&) = delete;
+  MockPacketCreatorDelegate& operator=(const MockPacketCreatorDelegate&) =
+      delete;
   ~MockPacketCreatorDelegate() override;
 
   MOCK_METHOD0(GetPacketBuffer, char*());
@@ -998,9 +994,6 @@
                void(QuicErrorCode,
                     const QuicString&,
                     ConnectionCloseSource source));
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockPacketCreatorDelegate);
 };
 
 class MockSessionNotifier : public SessionNotifierInterface {
diff --git a/net/third_party/quic/test_tools/rtt_stats_peer.h b/net/third_party/quic/test_tools/rtt_stats_peer.h
index b508c84..aba2f23 100644
--- a/net/third_party/quic/test_tools/rtt_stats_peer.h
+++ b/net/third_party/quic/test_tools/rtt_stats_peer.h
@@ -14,12 +14,11 @@
 
 class RttStatsPeer {
  public:
+  RttStatsPeer() = delete;
+
   static void SetSmoothedRtt(RttStats* rtt_stats, QuicTime::Delta rtt_ms);
 
   static void SetMinRtt(RttStats* rtt_stats, QuicTime::Delta rtt_ms);
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(RttStatsPeer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/server_thread.h b/net/third_party/quic/test_tools/server_thread.h
index e75e64b..4046e8ac 100644
--- a/net/third_party/quic/test_tools/server_thread.h
+++ b/net/third_party/quic/test_tools/server_thread.h
@@ -23,6 +23,8 @@
 class ServerThread : public base::SimpleThread {
  public:
   ServerThread(QuicServer* server, const QuicSocketAddress& address);
+  ServerThread(const ServerThread&) = delete;
+  ServerThread& operator=(const ServerThread&) = delete;
 
   ~ServerThread() override;
 
@@ -80,8 +82,6 @@
   QuicMutex scheduled_actions_lock_;
   QuicDeque<std::function<void()>> scheduled_actions_
       GUARDED_BY(scheduled_actions_lock_);
-
-  DISALLOW_COPY_AND_ASSIGN(ServerThread);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/simple_quic_framer.cc b/net/third_party/quic/test_tools/simple_quic_framer.cc
index 7d33411..c84585d 100644
--- a/net/third_party/quic/test_tools/simple_quic_framer.cc
+++ b/net/third_party/quic/test_tools/simple_quic_framer.cc
@@ -18,6 +18,8 @@
 class SimpleFramerVisitor : public QuicFramerVisitorInterface {
  public:
   SimpleFramerVisitor() : error_(QUIC_NO_ERROR) {}
+  SimpleFramerVisitor(const SimpleFramerVisitor&) = delete;
+  SimpleFramerVisitor& operator=(const SimpleFramerVisitor&) = delete;
 
   ~SimpleFramerVisitor() override {}
 
@@ -230,8 +232,6 @@
   std::vector<QuicBlockedFrame> blocked_frames_;
   std::vector<QuicNewConnectionIdFrame> new_connection_id_frames_;
   std::vector<std::unique_ptr<QuicString>> stream_data_;
-
-  DISALLOW_COPY_AND_ASSIGN(SimpleFramerVisitor);
 };
 
 SimpleQuicFramer::SimpleQuicFramer()
diff --git a/net/third_party/quic/test_tools/simple_quic_framer.h b/net/third_party/quic/test_tools/simple_quic_framer.h
index 886facd9..37d20f1 100644
--- a/net/third_party/quic/test_tools/simple_quic_framer.h
+++ b/net/third_party/quic/test_tools/simple_quic_framer.h
@@ -27,6 +27,8 @@
   explicit SimpleQuicFramer(const ParsedQuicVersionVector& supported_versions);
   SimpleQuicFramer(const ParsedQuicVersionVector& supported_versions,
                    Perspective perspective);
+  SimpleQuicFramer(const SimpleQuicFramer&) = delete;
+  SimpleQuicFramer& operator=(const SimpleQuicFramer&) = delete;
   ~SimpleQuicFramer();
 
   bool ProcessPacket(const QuicEncryptedPacket& packet);
@@ -54,7 +56,6 @@
  private:
   QuicFramer framer_;
   std::unique_ptr<SimpleFramerVisitor> visitor_;
-  DISALLOW_COPY_AND_ASSIGN(SimpleQuicFramer);
 };
 
 }  // namespace test
diff --git a/net/third_party/quic/test_tools/simulator/actor.h b/net/third_party/quic/test_tools/simulator/actor.h
index d2698f7d..514e8f0 100644
--- a/net/third_party/quic/test_tools/simulator/actor.h
+++ b/net/third_party/quic/test_tools/simulator/actor.h
@@ -55,9 +55,9 @@
   // Since the Actor object registers itself with a simulator using a pointer to
   // itself, do not allow it to be moved.
   Actor(Actor&&) = delete;
+  Actor(const Actor&) = delete;
+  Actor& operator=(const Actor&) = delete;
   Actor& operator=(Actor&&) = delete;
-
-  DISALLOW_COPY_AND_ASSIGN(Actor);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/alarm_factory.h b/net/third_party/quic/test_tools/simulator/alarm_factory.h
index d10cd47..33d7e099 100644
--- a/net/third_party/quic/test_tools/simulator/alarm_factory.h
+++ b/net/third_party/quic/test_tools/simulator/alarm_factory.h
@@ -15,6 +15,8 @@
 class AlarmFactory : public QuicAlarmFactory {
  public:
   AlarmFactory(Simulator* simulator, std::string name);
+  AlarmFactory(const AlarmFactory&) = delete;
+  AlarmFactory& operator=(const AlarmFactory&) = delete;
   ~AlarmFactory() override;
 
   QuicAlarm* CreateAlarm(QuicAlarm::Delegate* delegate) override;
@@ -29,8 +31,6 @@
   Simulator* simulator_;
   std::string name_;
   int counter_;
-
-  DISALLOW_COPY_AND_ASSIGN(AlarmFactory);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/link.h b/net/third_party/quic/test_tools/simulator/link.h
index 8636ee5..64dfabdc 100644
--- a/net/third_party/quic/test_tools/simulator/link.h
+++ b/net/third_party/quic/test_tools/simulator/link.h
@@ -24,6 +24,8 @@
              UnconstrainedPortInterface* sink,
              QuicBandwidth bandwidth,
              QuicTime::Delta propagation_delay);
+  OneWayLink(const OneWayLink&) = delete;
+  OneWayLink& operator=(const OneWayLink&) = delete;
   ~OneWayLink() override;
 
   void AcceptPacket(std::unique_ptr<Packet> packet) override;
@@ -57,8 +59,6 @@
   const QuicTime::Delta propagation_delay_;
 
   QuicTime next_write_at_;
-
-  DISALLOW_COPY_AND_ASSIGN(OneWayLink);
 };
 
 // A full-duplex link between two endpoints, functionally equivalent to two
@@ -75,14 +75,14 @@
                 Endpoint* endpoint_b,
                 QuicBandwidth bandwidth,
                 QuicTime::Delta propagation_delay);
+  SymmetricLink(const SymmetricLink&) = delete;
+  SymmetricLink& operator=(const SymmetricLink&) = delete;
 
   inline QuicBandwidth bandwidth() { return a_to_b_link_.bandwidth(); }
 
  private:
   OneWayLink a_to_b_link_;
   OneWayLink b_to_a_link_;
-
-  DISALLOW_COPY_AND_ASSIGN(SymmetricLink);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/packet_filter.h b/net/third_party/quic/test_tools/simulator/packet_filter.h
index 4c4878e3..e769db1 100644
--- a/net/third_party/quic/test_tools/simulator/packet_filter.h
+++ b/net/third_party/quic/test_tools/simulator/packet_filter.h
@@ -40,6 +40,8 @@
   // Initialize the filter by wrapping around |input|.  Does not take the
   // ownership of |input|.
   PacketFilter(Simulator* simulator, std::string name, Endpoint* input);
+  PacketFilter(const PacketFilter&) = delete;
+  PacketFilter& operator=(const PacketFilter&) = delete;
   ~PacketFilter() override;
 
   // Implementation of ConstrainedPortInterface.
@@ -66,8 +68,6 @@
 
   // The original network endpoint wrapped by the class.
   Endpoint* input_;
-
-  DISALLOW_COPY_AND_ASSIGN(PacketFilter);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/queue.h b/net/third_party/quic/test_tools/simulator/queue.h
index 2d57fee..e9641d30 100644
--- a/net/third_party/quic/test_tools/simulator/queue.h
+++ b/net/third_party/quic/test_tools/simulator/queue.h
@@ -24,6 +24,8 @@
   };
 
   Queue(Simulator* simulator, std::string name, QuicByteCount capacity);
+  Queue(const Queue&) = delete;
+  Queue& operator=(const Queue&) = delete;
   ~Queue() override;
 
   void set_tx_port(ConstrainedPortInterface* port);
@@ -110,8 +112,6 @@
   QuicQueue<EnqueuedPacket> queue_;
 
   ListenerInterface* listener_;
-
-  DISALLOW_COPY_AND_ASSIGN(Queue);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/simulator.h b/net/third_party/quic/test_tools/simulator/simulator.h
index ebb882c..a086eb2 100644
--- a/net/third_party/quic/test_tools/simulator/simulator.h
+++ b/net/third_party/quic/test_tools/simulator/simulator.h
@@ -22,6 +22,8 @@
 class Simulator : public QuicConnectionHelperInterface {
  public:
   Simulator();
+  Simulator(const Simulator&) = delete;
+  Simulator& operator=(const Simulator&) = delete;
   ~Simulator() override;
 
   // Register an actor with the simulator.  Returns a handle which the actor can
@@ -125,8 +127,6 @@
   // unscheduled actors is QuicTime::Infinite().
   QuicUnorderedMap<Actor*, QuicTime> scheduled_times_;
   QuicUnorderedSet<std::string> actor_names_;
-
-  DISALLOW_COPY_AND_ASSIGN(Simulator);
 };
 
 template <class TerminationPredicate>
diff --git a/net/third_party/quic/test_tools/simulator/switch.h b/net/third_party/quic/test_tools/simulator/switch.h
index f344a425..0b272f3 100644
--- a/net/third_party/quic/test_tools/simulator/switch.h
+++ b/net/third_party/quic/test_tools/simulator/switch.h
@@ -23,6 +23,8 @@
          std::string name,
          SwitchPortNumber port_count,
          QuicByteCount queue_capacity);
+  Switch(const Switch&) = delete;
+  Switch& operator=(const Switch&) = delete;
   ~Switch();
 
   // Returns Endpoint associated with the port under number |port_number|.  Just
@@ -45,6 +47,8 @@
          SwitchPortNumber port_number,
          QuicByteCount queue_capacity);
     Port(Port&&) = delete;
+    Port(const Port&) = delete;
+    Port& operator=(const Port&) = delete;
     ~Port() override {}
 
     // Accepts packet to be routed into the switch.
@@ -66,8 +70,6 @@
     bool connected_;
 
     Queue queue_;
-
-    DISALLOW_COPY_AND_ASSIGN(Port);
   };
 
   // Sends the packet to the appropriate port, or to all ports if the
@@ -79,8 +81,6 @@
   // assumed to be stable.
   std::deque<Port> ports_;
   QuicUnorderedMap<std::string, Port*> switching_table_;
-
-  DISALLOW_COPY_AND_ASSIGN(Switch);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/test_tools/simulator/traffic_policer.h b/net/third_party/quic/test_tools/simulator/traffic_policer.h
index 8e389de9..fd38fcf 100644
--- a/net/third_party/quic/test_tools/simulator/traffic_policer.h
+++ b/net/third_party/quic/test_tools/simulator/traffic_policer.h
@@ -25,6 +25,8 @@
                  QuicByteCount max_bucket_size,
                  QuicBandwidth target_bandwidth,
                  Endpoint* input);
+  TrafficPolicer(const TrafficPolicer&) = delete;
+  TrafficPolicer& operator=(const TrafficPolicer&) = delete;
   ~TrafficPolicer() override;
 
  protected:
@@ -44,8 +46,6 @@
 
   // Maps each destination to the number of tokens it has left.
   QuicUnorderedMap<std::string, QuicByteCount> token_buckets_;
-
-  DISALLOW_COPY_AND_ASSIGN(TrafficPolicer);
 };
 
 }  // namespace simulator
diff --git a/net/third_party/quic/tools/quic_backend_response.h b/net/third_party/quic/tools/quic_backend_response.h
index 3d898f8..81698d0 100644
--- a/net/third_party/quic/tools/quic_backend_response.h
+++ b/net/third_party/quic/tools/quic_backend_response.h
@@ -5,7 +5,7 @@
 #ifndef NET_THIRD_PARTY_QUIC_TOOLS_QUIC_BACKEND_RESPONSE_H_
 #define NET_THIRD_PARTY_QUIC_TOOLS_QUIC_BACKEND_RESPONSE_H_
 
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_url.h"
 
 namespace quic {
diff --git a/net/third_party/quic/tools/quic_client.cc b/net/third_party/quic/tools/quic_client.cc
index 97b3feec4..2b00375 100644
--- a/net/third_party/quic/tools/quic_client.cc
+++ b/net/third_party/quic/tools/quic_client.cc
@@ -13,13 +13,13 @@
 
 #include "base/run_loop.h"
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_data_reader.h"
 #include "net/third_party/quic/core/quic_epoll_alarm_factory.h"
 #include "net/third_party/quic/core/quic_epoll_connection_helper.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/tools/quic_client.h b/net/third_party/quic/tools/quic_client.h
index 14747f4..6778176 100644
--- a/net/third_party/quic/tools/quic_client.h
+++ b/net/third_party/quic/tools/quic_client.h
@@ -14,12 +14,12 @@
 
 #include "base/command_line.h"
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
 #include "net/third_party/quic/core/quic_packet_reader.h"
 #include "net/third_party/quic/core/quic_process_packet_interface.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
 #include "net/third_party/quic/tools/quic_client_base.h"
 #include "net/third_party/quic/tools/quic_client_epoll_network_helper.h"
@@ -56,6 +56,8 @@
              net::EpollServer* epoll_server,
              std::unique_ptr<QuicClientEpollNetworkHelper> network_helper,
              std::unique_ptr<ProofVerifier> proof_verifier);
+  QuicClient(const QuicClient&) = delete;
+  QuicClient& operator=(const QuicClient&) = delete;
 
   ~QuicClient() override;
 
@@ -75,8 +77,6 @@
  private:
   friend class test::QuicClientPeer;
   bool drop_response_body_ = false;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicClient);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_client_base.cc b/net/third_party/quic/tools/quic_client_base.cc
index 8166c4e..8b7958d 100644
--- a/net/third_party/quic/tools/quic_client_base.cc
+++ b/net/third_party/quic/tools/quic_client_base.cc
@@ -5,8 +5,8 @@
 #include "net/third_party/quic/tools/quic_client_base.h"
 
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_client_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
diff --git a/net/third_party/quic/tools/quic_client_base.h b/net/third_party/quic/tools/quic_client_base.h
index 4503085..e722180 100644
--- a/net/third_party/quic/tools/quic_client_base.h
+++ b/net/third_party/quic/tools/quic_client_base.h
@@ -12,10 +12,10 @@
 
 #include "base/macros.h"
 #include "net/third_party/quic/core/crypto/crypto_handshake.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "net/third_party/quic/platform/api/quic_socket_address.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 
@@ -65,6 +65,8 @@
                  QuicAlarmFactory* alarm_factory,
                  std::unique_ptr<NetworkHelper> network_helper,
                  std::unique_ptr<ProofVerifier> proof_verifier);
+  QuicClientBase(const QuicClientBase&) = delete;
+  QuicClientBase& operator=(const QuicClientBase&) = delete;
 
   virtual ~QuicClientBase();
 
@@ -355,8 +357,6 @@
   // The network helper used to create sockets and manage the event loop.
   // Not owned by this class.
   std::unique_ptr<NetworkHelper> network_helper_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicClientBase);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_client_epoll_network_helper.cc b/net/third_party/quic/tools/quic_client_epoll_network_helper.cc
index ae144130..784282f 100644
--- a/net/third_party/quic/tools/quic_client_epoll_network_helper.cc
+++ b/net/third_party/quic/tools/quic_client_epoll_network_helper.cc
@@ -13,13 +13,13 @@
 
 #include "base/run_loop.h"
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_data_reader.h"
 #include "net/third_party/quic/core/quic_epoll_alarm_factory.h"
 #include "net/third_party/quic/core/quic_epoll_connection_helper.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/tools/quic_client_epoll_network_helper.h b/net/third_party/quic/tools/quic_client_epoll_network_helper.h
index 1b20afd..2a6d22e3 100644
--- a/net/third_party/quic/tools/quic_client_epoll_network_helper.h
+++ b/net/third_party/quic/tools/quic_client_epoll_network_helper.h
@@ -14,12 +14,12 @@
 
 #include "base/command_line.h"
 #include "base/macros.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
 #include "net/third_party/quic/core/quic_packet_reader.h"
 #include "net/third_party/quic/core/quic_process_packet_interface.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
 #include "net/third_party/quic/tools/quic_client_base.h"
 #include "net/third_party/quic/tools/quic_spdy_client_base.h"
@@ -41,6 +41,9 @@
   // net::EpollServer.
   QuicClientEpollNetworkHelper(net::EpollServer* epoll_server,
                                QuicClientBase* client);
+  QuicClientEpollNetworkHelper(const QuicClientEpollNetworkHelper&) = delete;
+  QuicClientEpollNetworkHelper& operator=(const QuicClientEpollNetworkHelper&) =
+      delete;
 
   ~QuicClientEpollNetworkHelper() override;
 
@@ -130,8 +133,6 @@
   QuicClientBase* client_;
 
   int max_reads_per_epoll_loop_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicClientEpollNetworkHelper);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_memory_cache_backend.cc b/net/third_party/quic/tools/quic_memory_cache_backend.cc
index f6825dd..167fbd93 100644
--- a/net/third_party/quic/tools/quic_memory_cache_backend.cc
+++ b/net/third_party/quic/tools/quic_memory_cache_backend.cc
@@ -6,7 +6,7 @@
 
 #include <utility>
 
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_file_utils.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
diff --git a/net/third_party/quic/tools/quic_memory_cache_backend.h b/net/third_party/quic/tools/quic_memory_cache_backend.h
index 50d03423..ee605a4 100644
--- a/net/third_party/quic/tools/quic_memory_cache_backend.h
+++ b/net/third_party/quic/tools/quic_memory_cache_backend.h
@@ -10,7 +10,7 @@
 #include <memory>
 #include <vector>
 
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
 #include "net/third_party/quic/platform/api/quic_mutex.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
@@ -33,6 +33,8 @@
   class ResourceFile {
    public:
     explicit ResourceFile(const QuicString& file_name);
+    ResourceFile(const ResourceFile&) = delete;
+    ResourceFile& operator=(const ResourceFile&) = delete;
     virtual ~ResourceFile();
 
     void Read();
@@ -68,11 +70,11 @@
     QuicStringPiece host_;
     QuicStringPiece path_;
     QuicMemoryCacheBackend* cache_;
-
-    DISALLOW_COPY_AND_ASSIGN(ResourceFile);
   };
 
   QuicMemoryCacheBackend();
+  QuicMemoryCacheBackend(const QuicMemoryCacheBackend&) = delete;
+  QuicMemoryCacheBackend& operator=(const QuicMemoryCacheBackend&) = delete;
   ~QuicMemoryCacheBackend() override;
 
   // Retrieve a response from this cache for a given host and path..
@@ -184,8 +186,6 @@
   // server threads accessing those responses.
   mutable QuicMutex response_mutex_;
   bool cache_initialized_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicMemoryCacheBackend);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_server.h b/net/third_party/quic/tools/quic_server.h
index 1d5ffa1..322c4ab 100644
--- a/net/third_party/quic/tools/quic_server.h
+++ b/net/third_party/quic/tools/quic_server.h
@@ -42,6 +42,8 @@
              const QuicCryptoServerConfig::ConfigOptions& server_config_options,
              const ParsedQuicVersionVector& supported_versions,
              QuicSimpleServerBackend* quic_simple_server_backend);
+  QuicServer(const QuicServer&) = delete;
+  QuicServer& operator=(const QuicServer&) = delete;
 
   ~QuicServer() override;
 
@@ -148,8 +150,6 @@
   QuicSimpleServerBackend* quic_simple_server_backend_;  // unowned.
 
   base::WeakPtrFactory<QuicServer> weak_factory_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicServer);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_simple_client_session.h b/net/third_party/quic/tools/quic_simple_client_session.h
index 195b40a..05d9121 100644
--- a/net/third_party/quic/tools/quic_simple_client_session.h
+++ b/net/third_party/quic/tools/quic_simple_client_session.h
@@ -5,7 +5,7 @@
 #ifndef NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_CLIENT_SESSION_H_
 #define NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_CLIENT_SESSION_H_
 
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
 #include "net/third_party/quic/tools/quic_simple_client_stream.h"
 
 namespace quic {
diff --git a/net/third_party/quic/tools/quic_simple_client_stream.h b/net/third_party/quic/tools/quic_simple_client_stream.h
index d50c107..dc6abf4c 100644
--- a/net/third_party/quic/tools/quic_simple_client_stream.h
+++ b/net/third_party/quic/tools/quic_simple_client_stream.h
@@ -5,7 +5,7 @@
 #ifndef NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_CLIENT_STREAM_H_
 #define NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_CLIENT_STREAM_H_
 
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 
 namespace quic {
 
diff --git a/net/third_party/quic/tools/quic_simple_dispatcher.h b/net/third_party/quic/tools/quic_simple_dispatcher.h
index fa460723..25414a1 100644
--- a/net/third_party/quic/tools/quic_simple_dispatcher.h
+++ b/net/third_party/quic/tools/quic_simple_dispatcher.h
@@ -5,8 +5,8 @@
 #ifndef NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_DISPATCHER_H_
 #define NET_THIRD_PARTY_QUIC_TOOLS_QUIC_SIMPLE_DISPATCHER_H_
 
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
 #include "net/third_party/quic/core/quic_dispatcher.h"
-#include "net/third_party/quic/core/quic_server_session_base.h"
 #include "net/third_party/quic/tools/quic_simple_server_backend.h"
 
 namespace quic {
diff --git a/net/third_party/quic/tools/quic_simple_server_session.h b/net/third_party/quic/tools/quic_simple_server_session.h
index 423e433..2551bc43 100644
--- a/net/third_party/quic/tools/quic_simple_server_session.h
+++ b/net/third_party/quic/tools/quic_simple_server_session.h
@@ -17,10 +17,10 @@
 #include <vector>
 
 #include "base/macros.h"
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_session.h"
 #include "net/third_party/quic/core/quic_crypto_server_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_server_session_base.h"
-#include "net/third_party/quic/core/quic_spdy_session.h"
 #include "net/third_party/quic/platform/api/quic_containers.h"
 #include "net/third_party/quic/tools/quic_backend_response.h"
 #include "net/third_party/quic/tools/quic_simple_server_backend.h"
@@ -60,6 +60,8 @@
                           const QuicCryptoServerConfig* crypto_config,
                           QuicCompressedCertsCache* compressed_certs_cache,
                           QuicSimpleServerBackend* quic_simple_server_backend);
+  QuicSimpleServerSession(const QuicSimpleServerSession&) = delete;
+  QuicSimpleServerSession& operator=(const QuicSimpleServerSession&) = delete;
 
   ~QuicSimpleServerSession() override;
 
@@ -149,8 +151,6 @@
   QuicDeque<PromisedStreamInfo> promised_streams_;
 
   QuicSimpleServerBackend* quic_simple_server_backend_;  // Not owned.
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSimpleServerSession);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_simple_server_session_test.cc b/net/third_party/quic/tools/quic_simple_server_session_test.cc
index 8352e96a..c9c76aa 100644
--- a/net/third_party/quic/tools/quic_simple_server_session_test.cc
+++ b/net/third_party/quic/tools/quic_simple_server_session_test.cc
@@ -31,7 +31,6 @@
 #include "net/third_party/quic/test_tools/quic_sent_packet_manager_peer.h"
 #include "net/third_party/quic/test_tools/quic_session_peer.h"
 #include "net/third_party/quic/test_tools/quic_spdy_session_peer.h"
-#include "net/third_party/quic/test_tools/quic_spdy_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_stream_peer.h"
 #include "net/third_party/quic/test_tools/quic_sustained_bandwidth_recorder_peer.h"
 #include "net/third_party/quic/test_tools/quic_test_utils.h"
@@ -88,7 +87,10 @@
                 enable_quic_stateless_reject_support),  // NOLINT
             session,
             helper) {}
-  ~MockQuicCryptoServerStream() override = default;
+  MockQuicCryptoServerStream(const MockQuicCryptoServerStream&) = delete;
+  MockQuicCryptoServerStream& operator=(const MockQuicCryptoServerStream&) =
+      delete;
+  ~MockQuicCryptoServerStream() override {}
 
   MOCK_METHOD1(SendServerConfigUpdate,
                void(const CachedNetworkParameters* cached_network_parameters));
@@ -104,8 +106,6 @@
 
  private:
   bool encryption_established_override_ = false;
-
-  DISALLOW_COPY_AND_ASSIGN(MockQuicCryptoServerStream);
 };
 
 class MockQuicConnectionWithSendStreamData : public MockQuicConnection {
diff --git a/net/third_party/quic/tools/quic_simple_server_stream.cc b/net/third_party/quic/tools/quic_simple_server_stream.cc
index ee33460..eac3e0c 100644
--- a/net/third_party/quic/tools/quic_simple_server_stream.cc
+++ b/net/third_party/quic/tools/quic_simple_server_stream.cc
@@ -7,8 +7,8 @@
 #include <list>
 #include <utility>
 
-#include "net/third_party/quic/core/quic_spdy_stream.h"
-#include "net/third_party/quic/core/spdy_utils.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_bug_tracker.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
diff --git a/net/third_party/quic/tools/quic_simple_server_stream.h b/net/third_party/quic/tools/quic_simple_server_stream.h
index 8dd096a..226210b 100644
--- a/net/third_party/quic/tools/quic_simple_server_stream.h
+++ b/net/third_party/quic/tools/quic_simple_server_stream.h
@@ -7,9 +7,9 @@
 
 #include "base/macros.h"
 #include "net/http/http_response_headers.h"
+#include "net/third_party/quic/core/http/quic_spdy_server_stream_base.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_packets.h"
-#include "net/third_party/quic/core/quic_spdy_server_stream_base.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/quic/tools/quic_backend_response.h"
 #include "net/third_party/quic/tools/quic_simple_server_backend.h"
@@ -29,6 +29,8 @@
   QuicSimpleServerStream(QuicStreamId id,
                          QuicSpdySession* session,
                          QuicSimpleServerBackend* quic_simple_server_backend);
+  QuicSimpleServerStream(const QuicSimpleServerStream&) = delete;
+  QuicSimpleServerStream& operator=(const QuicSimpleServerStream&) = delete;
   ~QuicSimpleServerStream() override;
 
   // QuicSpdyStream
@@ -97,8 +99,6 @@
   QuicString body_;
 
   QuicSimpleServerBackend* quic_simple_server_backend_;  // Not owned.
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSimpleServerStream);
 };
 
 }  // namespace quic
diff --git a/net/third_party/quic/tools/quic_simple_server_stream_test.cc b/net/third_party/quic/tools/quic_simple_server_stream_test.cc
index 771a7c4..8fcd565 100644
--- a/net/third_party/quic/tools/quic_simple_server_stream_test.cc
+++ b/net/third_party/quic/tools/quic_simple_server_stream_test.cc
@@ -9,8 +9,8 @@
 #include <utility>
 
 #include "net/test/gtest_util.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_utils.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/core/tls_server_handshaker.h"
 #include "net/third_party/quic/platform/api/quic_arraysize.h"
 #include "net/third_party/quic/platform/api/quic_expect_bug.h"
@@ -103,6 +103,9 @@
         .WillByDefault(testing::Return(QuicConsumedData(0, false)));
   }
 
+  MockQuicSimpleServerSession(const MockQuicSimpleServerSession&) = delete;
+  MockQuicSimpleServerSession& operator=(const MockQuicSimpleServerSession&) =
+      delete;
   ~MockQuicSimpleServerSession() override = default;
 
   MOCK_METHOD3(OnConnectionClosed,
@@ -165,9 +168,6 @@
   using QuicSession::ActivateStream;
 
   spdy::SpdyHeaderBlock original_request_headers_;
-
- private:
-  DISALLOW_COPY_AND_ASSIGN(MockQuicSimpleServerSession);
 };
 
 class QuicSimpleServerStreamTest : public QuicTestWithParam<ParsedQuicVersion> {
diff --git a/net/third_party/quic/tools/quic_spdy_client_base.cc b/net/third_party/quic/tools/quic_spdy_client_base.cc
index 0d34d46..b88fd79 100644
--- a/net/third_party/quic/tools/quic_spdy_client_base.cc
+++ b/net/third_party/quic/tools/quic_spdy_client_base.cc
@@ -5,8 +5,8 @@
 #include "net/third_party/quic/tools/quic_spdy_client_base.h"
 
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_logging.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
diff --git a/net/third_party/quic/tools/quic_spdy_client_base.h b/net/third_party/quic/tools/quic_spdy_client_base.h
index c896447..edb2003 100644
--- a/net/third_party/quic/tools/quic_spdy_client_base.h
+++ b/net/third_party/quic/tools/quic_spdy_client_base.h
@@ -12,10 +12,10 @@
 
 #include "base/macros.h"
 #include "net/third_party/quic/core/crypto/crypto_handshake.h"
-#include "net/third_party/quic/core/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_client_push_promise_index.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_session.h"
+#include "net/third_party/quic/core/http/quic_spdy_client_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
-#include "net/third_party/quic/core/quic_spdy_client_session.h"
-#include "net/third_party/quic/core/quic_spdy_client_stream.h"
 #include "net/third_party/quic/platform/api/quic_socket_address.h"
 #include "net/third_party/quic/platform/api/quic_string_piece.h"
 #include "net/third_party/quic/tools/quic_client_base.h"
@@ -53,6 +53,8 @@
     QuicDataToResend(std::unique_ptr<spdy::SpdyHeaderBlock> headers,
                      QuicStringPiece body,
                      bool fin);
+    QuicDataToResend(const QuicDataToResend&) = delete;
+    QuicDataToResend& operator=(const QuicDataToResend&) = delete;
 
     virtual ~QuicDataToResend();
 
@@ -64,9 +66,6 @@
     std::unique_ptr<spdy::SpdyHeaderBlock> headers_;
     QuicStringPiece body_;
     bool fin_;
-
-   private:
-    DISALLOW_COPY_AND_ASSIGN(QuicDataToResend);
   };
 
   QuicSpdyClientBase(const QuicServerId& server_id,
@@ -76,6 +75,8 @@
                      QuicAlarmFactory* alarm_factory,
                      std::unique_ptr<NetworkHelper> network_helper,
                      std::unique_ptr<ProofVerifier> proof_verifier);
+  QuicSpdyClientBase(const QuicSpdyClientBase&) = delete;
+  QuicSpdyClientBase& operator=(const QuicSpdyClientBase&) = delete;
 
   ~QuicSpdyClientBase() override;
 
@@ -171,14 +172,14 @@
       DCHECK(client);
     }
 
+    ClientQuicDataToResend(const ClientQuicDataToResend&) = delete;
+    ClientQuicDataToResend& operator=(const ClientQuicDataToResend&) = delete;
     ~ClientQuicDataToResend() override {}
 
     void Resend() override;
 
    private:
     QuicSpdyClientBase* client_;
-
-    DISALLOW_COPY_AND_ASSIGN(ClientQuicDataToResend);
   };
 
   // Index of pending promised streams. Must outlive |session_|.
@@ -207,8 +208,6 @@
   std::vector<std::unique_ptr<QuicDataToResend>> data_to_resend_on_connect_;
 
   std::unique_ptr<ClientQuicDataToResend> push_promise_data_to_resend_;
-
-  DISALLOW_COPY_AND_ASSIGN(QuicSpdyClientBase);
 };
 
 }  // namespace quic
diff --git a/net/tools/quic/quic_client_message_loop_network_helper.cc b/net/tools/quic/quic_client_message_loop_network_helper.cc
index c6154de..25a00eb 100644
--- a/net/tools/quic/quic_client_message_loop_network_helper.cc
+++ b/net/tools/quic/quic_client_message_loop_network_helper.cc
@@ -21,10 +21,10 @@
 #include "net/socket/udp_client_socket.h"
 #include "net/spdy/spdy_http_utils.h"
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/spdy/core/spdy_header_block.h"
diff --git a/net/tools/quic/quic_client_message_loop_network_helper.h b/net/tools/quic/quic_client_message_loop_network_helper.h
index a41f7fd3..a4e7d42 100644
--- a/net/tools/quic/quic_client_message_loop_network_helper.h
+++ b/net/tools/quic/quic_client_message_loop_network_helper.h
@@ -20,8 +20,8 @@
 #include "net/http/http_response_headers.h"
 #include "net/log/net_log.h"
 #include "net/quic/chromium/quic_chromium_packet_reader.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/impl/quic_chromium_clock.h"
 #include "net/third_party/quic/tools/quic_spdy_client_base.h"
 
diff --git a/net/tools/quic/quic_simple_client.cc b/net/tools/quic/quic_simple_client.cc
index ac17ee6c..d743f723 100644
--- a/net/tools/quic/quic_simple_client.cc
+++ b/net/tools/quic/quic_simple_client.cc
@@ -21,10 +21,10 @@
 #include "net/socket/udp_client_socket.h"
 #include "net/spdy/spdy_http_utils.h"
 #include "net/third_party/quic/core/crypto/quic_random.h"
+#include "net/third_party/quic/core/http/spdy_utils.h"
 #include "net/third_party/quic/core/quic_connection.h"
 #include "net/third_party/quic/core/quic_packets.h"
 #include "net/third_party/quic/core/quic_server_id.h"
-#include "net/third_party/quic/core/spdy_utils.h"
 #include "net/third_party/quic/platform/api/quic_flags.h"
 #include "net/third_party/quic/platform/api/quic_ptr_util.h"
 #include "net/third_party/spdy/core/spdy_header_block.h"
diff --git a/net/tools/quic/quic_simple_client.h b/net/tools/quic/quic_simple_client.h
index d9144b36..4d73cda8 100644
--- a/net/tools/quic/quic_simple_client.h
+++ b/net/tools/quic/quic_simple_client.h
@@ -20,8 +20,8 @@
 #include "net/http/http_response_headers.h"
 #include "net/log/net_log.h"
 #include "net/quic/chromium/quic_chromium_packet_reader.h"
+#include "net/third_party/quic/core/http/quic_spdy_stream.h"
 #include "net/third_party/quic/core/quic_config.h"
-#include "net/third_party/quic/core/quic_spdy_stream.h"
 #include "net/third_party/quic/platform/impl/quic_chromium_clock.h"
 #include "net/third_party/quic/tools/quic_spdy_client_base.h"
 #include "net/tools/quic/quic_client_message_loop_network_helper.h"
diff --git a/net/tools/quic/quic_simple_server_session_helper.h b/net/tools/quic/quic_simple_server_session_helper.h
index b176e26..46f3b65 100644
--- a/net/tools/quic/quic_simple_server_session_helper.h
+++ b/net/tools/quic/quic_simple_server_session_helper.h
@@ -6,7 +6,7 @@
 #define NET_TOOLS_QUIC_QUIC_SIMPLE_SERVER_SESSION_HELPER_H_
 
 #include "net/third_party/quic/core/crypto/quic_random.h"
-#include "net/third_party/quic/core/quic_server_session_base.h"
+#include "net/third_party/quic/core/http/quic_server_session_base.h"
 
 namespace net {
 
diff --git a/ppapi/generators/idl_ast.py b/ppapi/generators/idl_ast.py
index 515dbb0..e95e94e 100644
--- a/ppapi/generators/idl_ast.py
+++ b/ppapi/generators/idl_ast.py
@@ -178,5 +178,3 @@
       errors = filenode.GetProperty('ERRORS')
       if errors:
         self.errors += errors
-
-
diff --git a/ppapi/generators/idl_c_header.py b/ppapi/generators/idl_c_header.py
index b4b98227..9b4513ac 100755
--- a/ppapi/generators/idl_c_header.py
+++ b/ppapi/generators/idl_c_header.py
@@ -371,4 +371,3 @@
 
 if __name__ == '__main__':
   sys.exit(main(sys.argv[1:]))
-
diff --git a/ppapi/generators/idl_namespace.py b/ppapi/generators/idl_namespace.py
index b81411e..21a643c 100755
--- a/ppapi/generators/idl_namespace.py
+++ b/ppapi/generators/idl_namespace.py
@@ -246,4 +246,3 @@
 
 if __name__ == '__main__':
   sys.exit(Main(sys.argv[1:]))
-
diff --git a/ppapi/generators/idl_node.py b/ppapi/generators/idl_node.py
index 55b24d1..feeb430 100755
--- a/ppapi/generators/idl_node.py
+++ b/ppapi/generators/idl_node.py
@@ -444,4 +444,3 @@
 
 if __name__ == '__main__':
   sys.exit(Main())
-
diff --git a/ppapi/generators/idl_parser.py b/ppapi/generators/idl_parser.py
index 823dcf5a..fdd5e88c 100755
--- a/ppapi/generators/idl_parser.py
+++ b/ppapi/generators/idl_parser.py
@@ -1293,4 +1293,3 @@
 
 if __name__ == '__main__':
   sys.exit(Main(sys.argv[1:]))
-
diff --git a/ppapi/generators/idl_propertynode.py b/ppapi/generators/idl_propertynode.py
index e15354a..9152ed7 100755
--- a/ppapi/generators/idl_propertynode.py
+++ b/ppapi/generators/idl_propertynode.py
@@ -111,4 +111,3 @@
 
 if __name__ == '__main__':
   sys.exit(Main())
-
diff --git a/ppapi/generators/idl_release.py b/ppapi/generators/idl_release.py
index ff4aa01d..44068548 100755
--- a/ppapi/generators/idl_release.py
+++ b/ppapi/generators/idl_release.py
@@ -353,4 +353,3 @@
 
 if __name__ == '__main__':
   sys.exit(Main(sys.argv[1:]))
-
diff --git a/ppapi/generators/idl_tests.py b/ppapi/generators/idl_tests.py
index bf6d8a1..b41bb0f66 100755
--- a/ppapi/generators/idl_tests.py
+++ b/ppapi/generators/idl_tests.py
@@ -42,4 +42,3 @@
 
 if __name__ == '__main__':
   sys.exit(main(sys.argv[1:]))
-
diff --git a/services/identity/public/cpp/identity_manager.cc b/services/identity/public/cpp/identity_manager.cc
index dfc727dd..285f46b 100644
--- a/services/identity/public/cpp/identity_manager.cc
+++ b/services/identity/public/cpp/identity_manager.cc
@@ -260,9 +260,9 @@
   // TODO(843510): Consider setting this info and notifying observers
   // asynchronously in response to GoogleSigninSucceeded() once there are no
   // direct clients of SigninManager.
-  DCHECK(account_info.account_id == primary_account_info_.account_id);
-  DCHECK(account_info.gaia == primary_account_info_.gaia);
-  DCHECK(account_info.email == primary_account_info_.email);
+  DCHECK_EQ(account_info.account_id, primary_account_info_.account_id);
+  DCHECK_EQ(account_info.gaia, primary_account_info_.gaia);
+  DCHECK(gaia::AreEmailsSame(account_info.email, primary_account_info_.email));
   primary_account_info_ = AccountInfo();
 }
 #endif
diff --git a/services/identity/public/cpp/identity_test_utils.cc b/services/identity/public/cpp/identity_test_utils.cc
index 309f856..f1704081 100644
--- a/services/identity/public/cpp/identity_test_utils.cc
+++ b/services/identity/public/cpp/identity_test_utils.cc
@@ -194,7 +194,7 @@
   return account_info;
 }
 
-void ClearPrimaryAccount(SigninManagerForTest* signin_manager,
+void ClearPrimaryAccount(SigninManagerBase* signin_manager,
                          IdentityManager* identity_manager) {
 #if defined(OS_CHROMEOS)
   // TODO(blundell): If we ever need this functionality on ChromeOS (which seems
@@ -209,7 +209,10 @@
       identity_manager, run_loop.QuitClosure(),
       IdentityManagerEvent::PRIMARY_ACCOUNT_CLEARED);
 
-  signin_manager->ForceSignOut();
+  SigninManager* real_signin_manager =
+      SigninManager::FromSigninManagerBase(signin_manager);
+  real_signin_manager->SignOut(signin_metrics::SIGNOUT_TEST,
+                               signin_metrics::SignoutDelete::IGNORE_METRIC);
 
   run_loop.Run();
 #endif
diff --git a/services/identity/public/cpp/identity_test_utils.h b/services/identity/public/cpp/identity_test_utils.h
index ac550f4..e72264fc 100644
--- a/services/identity/public/cpp/identity_test_utils.h
+++ b/services/identity/public/cpp/identity_test_utils.h
@@ -79,11 +79,7 @@
 // Clears the primary account. On non-ChromeOS, results in the firing of the
 // IdentityManager and SigninManager callbacks for signout. Blocks until the
 // primary account is cleared.
-// Note that this function requires FakeSigninManager, as it internally invokes
-// functionality of the fake. If a use case emerges for invoking this
-// functionality with a production SigninManager, contact blundell@chromium.org.
-// NOTE: See disclaimer at top of file re: direct usage.
-void ClearPrimaryAccount(SigninManagerForTest* signin_manager,
+void ClearPrimaryAccount(SigninManagerBase* signin_manager,
                          IdentityManager* identity_manager);
 
 // Makes an account available for the given email address, generating a GAIA ID
diff --git a/storage/OWNERS b/storage/OWNERS
index 4cc173f..ca4682d 100644
--- a/storage/OWNERS
+++ b/storage/OWNERS
@@ -1,5 +1,6 @@
 dmurph@chromium.org
 jsbell@chromium.org
+pwnall@chromium.org
 kinuko@chromium.org
 
 # TEAM: storage-dev@chromium.org
diff --git a/testing/libfuzzer/fuzzers/v8_fuzzer.cc b/testing/libfuzzer/fuzzers/v8_fuzzer.cc
index 3f95f34..ec2c0b6 100644
--- a/testing/libfuzzer/fuzzers/v8_fuzzer.cc
+++ b/testing/libfuzzer/fuzzers/v8_fuzzer.cc
@@ -113,7 +113,11 @@
   bool is_running;
 };
 
-extern "C" int LLVMFuzzerInitialize(int* argc, char*** argv) {
+// Explicitly specify some attributes to avoid issues with the linker dead-
+// stripping the following function on macOS, as it is not called directly
+// by fuzz target. LibFuzzer runtime uses dlsym() to resolve that function.
+extern "C" __attribute__((used)) __attribute__((visibility("default"))) int
+LLVMFuzzerInitialize(int* argc, char*** argv) {
   v8::V8::InitializeICUDefaultLocation((*argv)[0]);
   v8::V8::InitializeExternalStartupData((*argv)[0]);
   v8::V8::SetFlagsFromCommandLine(argc, *argv, true);
diff --git a/third_party/WebKit/LayoutTests/FlagExpectations/enable-blink-features=LayoutNG b/third_party/WebKit/LayoutTests/FlagExpectations/enable-blink-features=LayoutNG
index 3fb4f9e..1afbf9fd 100644
--- a/third_party/WebKit/LayoutTests/FlagExpectations/enable-blink-features=LayoutNG
+++ b/third_party/WebKit/LayoutTests/FlagExpectations/enable-blink-features=LayoutNG
@@ -488,7 +488,6 @@
 crbug.com/591099 fast/block/positioning/positioned-child-inside-relative-positioned-anonymous-block.html [ Failure ]
 crbug.com/591099 fast/borders/bidi-002.html [ Failure ]
 crbug.com/859497 fast/borders/bidi-009a.html [ Failure ]
-crbug.com/591099 fast/borders/border-image-border-radius.html [ Failure ]
 crbug.com/591099 fast/borders/inline-mask-overlay-image-outset-vertical-rl.html [ Failure ]
 crbug.com/591099 fast/box-shadow/box-shadow.html [ Failure ]
 crbug.com/591099 fast/box-sizing/replaced.html [ Failure ]
diff --git a/third_party/WebKit/LayoutTests/TestExpectations b/third_party/WebKit/LayoutTests/TestExpectations
index 4c29df8..34dea75 100644
--- a/third_party/WebKit/LayoutTests/TestExpectations
+++ b/third_party/WebKit/LayoutTests/TestExpectations
@@ -2657,6 +2657,15 @@
 crbug.com/849859 external/wpt/web-animations/timing-model/animations/pausing-an-animation.html [ Failure ]
 
 # ====== New tests from wpt-importer added here ======
+crbug.com/626703 external/wpt/svg/render/reftests/blending-002.svg [ Failure ]
+crbug.com/626703 external/wpt/svg/render/reftests/blending-001.svg [ Failure ]
+crbug.com/626703 [ Win7 ] external/wpt/css/css-lists/content-property/marker-text-matches-armenian.html [ Failure ]
+crbug.com/626703 [ Mac ] external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin.html [ Failure ]
+crbug.com/626703 external/wpt/css/css-lists/content-property/marker-text-matches-disc.html [ Failure ]
+crbug.com/626703 [ Win ] external/wpt/css/css-lists/content-property/marker-text-matches-georgian.html [ Failure ]
+crbug.com/626703 external/wpt/css/css-lists/content-property/marker-text-matches-square.html [ Failure ]
+crbug.com/626703 external/wpt/svg/linking/reftests/use-descendant-combinator-003.html [ Failure ]
+crbug.com/626703 external/wpt/css/css-lists/content-property/marker-text-matches-circle.html [ Failure ]
 crbug.com/626703 external/wpt/css/vendor-imports/mozilla/mozilla-central-reftests/flexbox/flexbox-safe-overflow-position-001.html [ Failure ]
 crbug.com/626703 external/wpt/content-security-policy/securitypolicyviolation/inside-service-worker.https.html [ Timeout ]
 crbug.com/626703 external/wpt/content-security-policy/securitypolicyviolation/inside-shared-worker.html [ Timeout ]
@@ -3745,7 +3754,8 @@
 
 crbug.com/689781 external/wpt/media-source/mediasource-duration.html [ Failure Pass ]
 
-# When WebAssembly is exposed in V8 (soon), this test has the wrong number of expected Object.getOwnPropertyNames() for global object.
+crbug.com/860637 http/tests/wasm_streaming/regression860637.html [ Skip ]
+crbug.com/860637 virtual/enable_wasm_streaming/http/tests/wasm_streaming/regression860637.html [ Skip ]
 
 crbug.com/681468 fast/forms/suggestion-picker/date-suggestion-picker-appearance-zoom125.html [ Failure Pass ]
 crbug.com/681468 fast/forms/suggestion-picker/date-suggestion-picker-appearance-zoom200.html [ Failure Pass ]
@@ -4612,14 +4622,6 @@
 crbug.com/856601 [ Linux ] external/wpt/touch-events/idlharness.window.html [ Timeout Pass ]
 crbug.com/856601 [ Linux ] http/tests/event-timing/event-timing-retrievability.html [ Timeout Pass ]
 
-### fast/canvas/color-space/ + virtual/gpu/..., color management and transparency.
-crbug.com/859102 fast/canvas/color-space/canvas-drawImage-linear-rgb.html [ Pass Failure ]
-crbug.com/859102 fast/canvas/color-space/canvas-drawImage-p3.html [ Pass Failure ]
-crbug.com/859102 fast/canvas/color-space/canvas-drawImage-rec2020.html [ Pass Failure ]
-crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-linear-rgb.html [ Pass Failure ]
-crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-p3.html [ Pass Failure ]
-crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-rec2020.html [ Pass Failure ]
-
 # Sheriff 2018-07-11
 # Test timeouts following wpt-import@89ccb892...
 crbug.com/862588 [ Linux ] external/wpt/cookie-store/idlharness.tentative.https.html [ Timeout Pass ]
@@ -4627,7 +4629,6 @@
 crbug.com/862588 [ Linux ] external/wpt/push-api/idlharness.https.any.worker.html [ Timeout Pass ]
 crbug.com/862588 [ Linux ] external/wpt/resource-timing/idlharness.any.html [ Timeout Pass ]
 crbug.com/862588 [ Linux ] external/wpt/resource-timing/idlharness.any.worker.html [ Timeout Pass ]
-crbug.com/862588 [ Linux ] external/wpt/selection/interfaces.html [ Timeout Pass ]
 crbug.com/862588 [ Linux ] external/wpt/service-workers/service-worker/interfaces-sw.https.html [ Timeout Pass ]
 crbug.com/862588 [ Linux ] virtual/feature-policy-permissions/external/wpt/mediacapture-streams/idlharness.https.window.html [ Timeout Pass ]
 
@@ -4658,3 +4659,21 @@
 crbug.com/866785 [ Mac ] svg/as-border-image/svg-as-border-image-2.html [ Failure Pass ]
 crbug.com/866785 [ Mac ] svg/as-border-image/svg-as-border-image.html [ Failure Pass ]
 crbug.com/866785 [ Mac ] virtual/new-remote-playback-pipeline/media/controls/text-track-menu-pointer-selection.html [ Failure Pass ]
+
+### fast/canvas/color-space/ + virtual/gpu/..., color management and transparency.
+crbug.com/859102 fast/canvas/color-space/canvas-drawImage-linear-rgb.html [ Pass Failure ]
+crbug.com/859102 fast/canvas/color-space/canvas-drawImage-p3.html [ Pass Failure ]
+crbug.com/859102 fast/canvas/color-space/canvas-drawImage-rec2020.html [ Pass Failure ]
+crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-linear-rgb.html [ Pass Failure ]
+crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-p3.html [ Pass Failure ]
+crbug.com/859102 virtual/gpu/fast/canvas/color-space/canvas-drawImage-rec2020.html [ Pass Failure ]
+
+### virtual/gpu/fast/canvas/ blending layout tests timeout
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blend-image.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blending-gradient-over-image.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blending-image-over-image.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blending-pattern-over-color.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blending-pattern-over-gradient.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/canvas-blending-pattern-over-image.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/color-space/canvas-createImageBitmap-p3.html [ Pass Timeout ]
+crbug.com/866850 [ Linux Mac ] virtual/gpu/fast/canvas/color-space/canvas-createImageBitmap-rec2020.html [ Pass Timeout ]
diff --git a/third_party/WebKit/LayoutTests/css-parser/unitless-length-quirk.html b/third_party/WebKit/LayoutTests/css-parser/unitless-length-quirk.html
index c0afd56b..1bd667db 100644
--- a/third_party/WebKit/LayoutTests/css-parser/unitless-length-quirk.html
+++ b/third_party/WebKit/LayoutTests/css-parser/unitless-length-quirk.html
@@ -9,6 +9,6 @@
 
 assert_invalid_value("outlineWidth", "50");
 assert_invalid_value("offsetDistance", "60");
-assert_invalid_value("webkitPaddingStart", "12");
-assert_invalid_value("webkitMarginBefore", "24");
+assert_invalid_value("paddingInlineStart", "12");
+assert_invalid_value("marginBlockStart", "24");
 </script>
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-2.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-2.html
index 4117697d..f6a040e 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-2.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-2.html
@@ -66,57 +66,57 @@
 
 <div class="flexbox">
   <div class="first" data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div class="second" data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="275"></div></div>
-  <div class="third" data-expected-width="75" data-offset-x="425" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="275"></div></div>
+  <div class="third" data-expected-width="75" data-offset-x="425" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox rtl">
   <div class="first" data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div class="second" data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="175"></div></div>
-  <div class="third" data-expected-width="75" data-offset-x="100" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="175"></div></div>
+  <div class="third" data-expected-width="75" data-offset-x="100" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox row-reverse">
   <div class="first" data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div class="second" data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="375"></div></div>
-  <div class="third" data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="375"></div></div>
+  <div class="third" data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox rtl row-reverse">
   <div class="first" data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div class="second" data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="75"></div></div>
-  <div class="third" data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="75"></div></div>
+  <div class="third" data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column">
   <div class="first" data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; margin: auto 200px auto 150px;"></div>
-  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
-  <div class="third" data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
+  <div class="third" data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column-reverse">
   <div class="first" data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; margin: auto 200px auto 150px;"></div>
-  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
-  <div class="third" data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
+  <div class="third" data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column rtl">
   <div class="first" data-expected-height="150" data-offset-y="0" data-offset-x="480" style="flex: 1 0 0; margin: auto 100px auto 50px;"></div>
-  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
-  <div class="third" data-expected-height="150" data-offset-y="450" data-offset-x="580" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
+  <div class="third" data-expected-height="150" data-offset-y="450" data-offset-x="580" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column-reverse rtl">
   <div class="first" data-expected-height="150" data-offset-y="450" data-offset-x="480" style="flex: 1 0 0; margin: auto 100px auto 50px;"></div>
-  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
-  <div class="third" data-expected-height="150" data-offset-y="0" data-offset-x="580" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div class="second" data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
+  <div class="third" data-expected-height="150" data-offset-y="0" data-offset-x="580" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-auto-margins.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-auto-margins.html
index fcb7c51..3979711 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-auto-margins.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-auto-margins.html
@@ -53,16 +53,16 @@
     margin: 13px auto 17px auto;
 }
 .logical {
-    -webkit-margin-before: 10px;
-    -webkit-margin-after: 30px;
-    -webkit-margin-start: 40px;
-    -webkit-margin-end: 20px;
+    margin-block-start: 10px;
+    margin-block-end: 30px;
+    margin-inline-start: 40px;
+    margin-inline-end: 20px;
 }
 .logical > div {
-    -webkit-margin-before: auto;
-    -webkit-margin-after: 17px;
-    -webkit-margin-start: auto;
-    -webkit-margin-end: 2px;
+    margin-block-start: auto;
+    margin-block-end: 17px;
+    margin-inline-start: auto;
+    margin-inline-end: 2px;
 }
 .flexbox > :nth-child(1) {
     background-color: blue;
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-border.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-border.html
index eba0bb4..cc86577 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-border.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-border.html
@@ -55,16 +55,16 @@
     border-width: 13px 2px 17px 8px;
 }
 .logical {
-    -webkit-border-before-width: 10px;
-    -webkit-border-after-width: 30px;
-    -webkit-border-start-width: 40px;
-    -webkit-border-end-width: 20px;
+    border-block-start-width: 10px;
+    border-block-end-width: 30px;
+    border-inline-start-width: 40px;
+    border-inline-end-width: 20px;
 }
 .logical > div {
-    -webkit-border-before-width: 13px;
-    -webkit-border-after-width: 17px;
-    -webkit-border-start-width: 8px;
-    -webkit-border-end-width: 2px;
+    border-block-start-width: 13px;
+    border-block-end-width: 17px;
+    border-inline-start-width: 8px;
+    border-inline-end-width: 2px;
 }
 .flexbox > :nth-child(1) {
     background-color: blue;
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-margins.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-margins.html
index cb3f930f..2bc8a06 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-margins.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-margins.html
@@ -50,16 +50,16 @@
     margin: 13px 2px 17px 8px;
 }
 .logical {
-    -webkit-margin-before: 10px;
-    -webkit-margin-after: 30px;
-    -webkit-margin-start: 40px;
-    -webkit-margin-end: 20px;
+    margin-block-start: 10px;
+    margin-block-end: 30px;
+    margin-inline-start: 40px;
+    margin-inline-end: 20px;
 }
 .logical > div {
-    -webkit-margin-before: 13px;
-    -webkit-margin-after: 17px;
-    -webkit-margin-start: 8px;
-    -webkit-margin-end: 2px;
+    margin-block-start: 13px;
+    margin-block-end: 17px;
+    margin-inline-start: 8px;
+    margin-inline-end: 2px;
 }
 .flexbox > :nth-child(1) {
     background-color: blue;
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-padding.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-padding.html
index bf0119c..f9cd465 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-padding.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow-padding.html
@@ -50,16 +50,16 @@
     padding: 13px 2px 17px 8px;
 }
 .logical {
-    -webkit-padding-before: 10px;
-    -webkit-padding-after: 30px;
-    -webkit-padding-start: 40px;
-    -webkit-padding-end: 20px;
+    padding-block-start: 10px;
+    padding-block-end: 30px;
+    padding-inline-start: 40px;
+    padding-inline-end: 20px;
 }
 .logical > div {
-    -webkit-padding-before: 13px;
-    -webkit-padding-after: 17px;
-    -webkit-padding-start: 8px;
-    -webkit-padding-end: 2px;
+    padding-block-start: 13px;
+    padding-block-end: 17px;
+    padding-inline-start: 8px;
+    padding-inline-end: 2px;
 }
 .flexbox > :nth-child(1) {
     background-color: blue;
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow.html b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow.html
index b61463a..32938a28 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/flex-flow.html
@@ -66,57 +66,57 @@
 
 <div class="flexbox">
   <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="275"></div></div>
-  <div data-expected-width="75" data-offset-x="425" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="275"></div></div>
+  <div data-expected-width="75" data-offset-x="425" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox rtl">
   <div data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="175"></div></div>
-  <div data-expected-width="75" data-offset-x="100" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="175"></div></div>
+  <div data-expected-width="75" data-offset-x="100" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox row-reverse">
   <div data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="375"></div></div>
-  <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="375"></div></div>
+  <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div class="flexbox rtl row-reverse">
   <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin: 0 auto;"></div>
-  <div data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-x="75"></div></div>
-  <div data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-width="350" data-offset-x="75" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-x="75"></div></div>
+  <div data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column">
   <div data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; margin: auto 200px auto 150px;"></div>
-  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
-  <div data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
+  <div data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column-reverse">
   <div data-expected-height="150" data-offset-y="450" style="flex: 1 0 0; margin: auto 200px auto 150px;"></div>
-  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
-  <div data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="200"></div></div>
+  <div data-expected-height="150" data-offset-y="0" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column rtl">
   <div data-expected-height="150" data-offset-y="0" data-offset-x="480" style="flex: 1 0 0; margin: auto 100px auto 50px;"></div>
-  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
-  <div data-expected-height="150" data-offset-y="450" data-offset-x="580" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
+  <div data-expected-height="150" data-offset-y="450" data-offset-x="580" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
 <div style="position: relative;">
 <div class="flexbox column-reverse rtl">
   <div data-expected-height="150" data-offset-y="450" data-offset-x="480" style="flex: 1 0 0; margin: auto 100px auto 50px;"></div>
-  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; -webkit-padding-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
-  <div data-expected-height="150" data-offset-y="0" data-offset-x="580" style="flex: 1 0 0; -webkit-margin-end: 100px;"></div>
+  <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0; padding-inline-start: 200px"><div data-offset-y="150" data-offset-x="380"></div></div>
+  <div data-expected-height="150" data-offset-y="0" data-offset-x="580" style="flex: 1 0 0; margin-inline-end: 100px;"></div>
 </div>
 </div>
 
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/multiline-align-self.html b/third_party/WebKit/LayoutTests/css3/flexbox/multiline-align-self.html
index 6ccfdd55..09d2054 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/multiline-align-self.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/multiline-align-self.html
@@ -1037,7 +1037,7 @@
                 flexbox.className = 'flexbox ' + flexboxClassName;
                 flexbox.setAttribute('style', mainAxis + ': 70px');
 
-                var baselineMargin = (flexDirection.indexOf('row') != -1) ? '-webkit-margin-before: 5px;' : '-webkit-margin-start: 5px;';
+                var baselineMargin = (flexDirection.indexOf('row') != -1) ? 'margin-block-start: 5px;' : 'margin-inline-start: 5px;';
 
                 var testExpectations = expectations[writingMode][flexDirection][direction][wrap];
                 addChild(flexbox, mainAxis, crossAxis, 10, '10px', 'flex-start', testExpectations['child1']);
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/multiline-justify-content.html b/third_party/WebKit/LayoutTests/css3/flexbox/multiline-justify-content.html
index dd91223f..a5e18c1 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/multiline-justify-content.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/multiline-justify-content.html
@@ -1985,7 +1985,7 @@
                     flexbox.className = 'flexbox ' + flexboxClassName;
                     flexbox.setAttribute('style', mainAxis + ': 80px');
 
-                    var baselineMargin = (flexDirection.indexOf('row') != -1) ? '-webkit-margin-before: 5px' : '-webkit-margin-start: 5px';
+                    var baselineMargin = (flexDirection.indexOf('row') != -1) ? 'margin-block-start: 5px' : 'margin-inline-start: 5px';
 
                     var testExpectations = expectations[writingMode][flexDirection][direction][wrap][justifyContent];
                     addChild(flexbox, mainAxis, crossAxis, 40, '10', testExpectations['child1']);
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/negative-flex-rounding-assert.html b/third_party/WebKit/LayoutTests/css3/flexbox/negative-flex-rounding-assert.html
index 1daafb9..12abe30 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/negative-flex-rounding-assert.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/negative-flex-rounding-assert.html
@@ -12,10 +12,10 @@
     flex-direction: column;
 }
 .mfrac > :first-child {
-    -webkit-margin-after: 0.2em;
+    margin-block-end: 0.2em;
 }
 .mfrac > :last-child {
-    -webkit-margin-before: 0.2em;
+    margin-block-start: 0.2em;
 }
 .x {
     line-height: 9px;
diff --git a/third_party/WebKit/LayoutTests/css3/flexbox/writing-modes.html b/third_party/WebKit/LayoutTests/css3/flexbox/writing-modes.html
index edfc353c..a819051 100644
--- a/third_party/WebKit/LayoutTests/css3/flexbox/writing-modes.html
+++ b/third_party/WebKit/LayoutTests/css3/flexbox/writing-modes.html
@@ -63,13 +63,13 @@
 <div class="flexbox rtl">
   <div data-expected-width="75" data-offset-x="525" style="flex: 1 0 0; margin: 0 auto;"></div>
   <div data-expected-width="350" data-offset-x="175" style="flex: 2 0 0; padding: 0 100px;"></div>
-  <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; -webkit-margin-start: 100px;"></div>
+  <div data-expected-width="75" data-offset-x="0" style="flex: 1 0 0; margin-inline-start: 100px;"></div>
 </div>
 
-<div class="flexbox rtl" style="-webkit-margin-start: 20px;-webkit-margin-end: 50px;">
+<div class="flexbox rtl" style="margin-inline-start: 20px;margin-inline-end: 50px;">
   <div data-expected-width="75" data-offset-x="575" style="flex: 1 0 0; margin: 0 auto;"></div>
   <div data-expected-width="350" data-offset-x="225" style="flex: 2 0 0; padding: 0 100px;"></div>
-  <div data-expected-width="75" data-offset-x="50" style="flex: 1 0 0; -webkit-margin-start: 100px;"></div>
+  <div data-expected-width="75" data-offset-x="50" style="flex: 1 0 0; margin-inline-start: 100px;"></div>
 </div>
 
 <div style="position:relative">
@@ -98,7 +98,7 @@
 
 <div style="position:relative">
 <div class="flexbox vertical-lr">
-  <div data-expected-height="450" data-offset-y="0" style="flex: 1 0 0;-webkit-padding-start:200px;-webkit-padding-end:200px;-webkit-padding-before:100px;-webkit-padding-after:100px;"></div>
+  <div data-expected-height="450" data-offset-y="0" style="flex: 1 0 0;padding-inline-start:200px;padding-inline-end:200px;padding-block-start:100px;padding-block-end:100px;"></div>
   <div data-expected-height="100" data-offset-y="450" style="flex: 2 0 0;"></div>
   <div data-expected-height="50" data-offset-y="550" style="flex: 1 0 0;"></div>
 </div>
@@ -106,7 +106,7 @@
 
 <div style="position:relative">
 <div class="flexbox vertical-lr">
-  <div data-expected-height="300" data-offset-y="0" class="horizontal-tb" style="flex: 1 0 0;-webkit-padding-start:200px;-webkit-padding-end:200px;-webkit-padding-before:100px;-webkit-padding-after:100px;"></div>
+  <div data-expected-height="300" data-offset-y="0" class="horizontal-tb" style="flex: 1 0 0;padding-inline-start:200px;padding-inline-end:200px;padding-block-start:100px;padding-block-end:100px;"></div>
   <div data-expected-height="200" data-offset-y="300" style="flex: 2 0 0;"></div>
   <div data-expected-height="100" data-offset-y="500" style="flex: 1 0 0;"></div>
 </div>
@@ -139,7 +139,7 @@
 
 <div style="position:relative">
 <div class="flexbox vertical-lr">
-  <div data-expected-height="150" data-offset-y="0" style="flex: 1 0 0;-webkit-margin-start:auto;-webkit-margin-end:auto;-webkit-margin-before:0;-webkit-margin-after:0;;"></div>
+  <div data-expected-height="150" data-offset-y="0" style="flex: 1 0 0;margin-inline-start:auto;margin-inline-end:auto;margin-block-start:0;margin-block-end:0;;"></div>
   <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0;"></div>
   <div data-expected-height="150" data-offset-y="450" style="flex: 1 0 0;"></div>
 </div>
@@ -147,7 +147,7 @@
 
 <div style="position:relative">
 <div class="flexbox vertical-lr">
-  <div data-expected-height="150" data-offset-y="0" class="horizontal-tb" style="flex: 1 0 0;-webkit-margin-start:auto;-webkit-margin-end:auto;-webkit-margin-before:0;-webkit-margin-after:0;;"></div>
+  <div data-expected-height="150" data-offset-y="0" class="horizontal-tb" style="flex: 1 0 0;margin-inline-start:auto;margin-inline-end:auto;margin-block-start:0;margin-block-end:0;;"></div>
   <div data-expected-height="300" data-offset-y="150" style="flex: 2 0 0;"></div>
   <div data-expected-height="150" data-offset-y="450" style="flex: 1 0 0;"></div>
 </div>
@@ -157,7 +157,7 @@
 <div class="flexbox vertical-lr rtl">
   <div data-expected-height="75" data-offset-y="525" style="flex: 1 0 0; margin: auto 0;"></div>
   <div data-expected-height="350" data-offset-y="175" class="horizontal-tb" style="flex: 2 0 0; padding: 100px 0;"></div>
-  <div data-expected-height="75" data-offset-y="0" style="flex: 1 0 0; -webkit-margin-start: 100px;"></div>
+  <div data-expected-height="75" data-offset-y="0" style="flex: 1 0 0; margin-inline-start: 100px;"></div>
 </div>
 </div>
 
diff --git a/third_party/WebKit/LayoutTests/editing/deleting/delete-blockquote-large-offsets.html b/third_party/WebKit/LayoutTests/editing/deleting/delete-blockquote-large-offsets.html
index 7ea982e..991b5cd 100644
--- a/third_party/WebKit/LayoutTests/editing/deleting/delete-blockquote-large-offsets.html
+++ b/third_party/WebKit/LayoutTests/editing/deleting/delete-blockquote-large-offsets.html
@@ -3,7 +3,7 @@
 <body>
 This tests setting caret inside a blockquote with a large offset and running execCommand('Delete').  WebKit should not crash and you should see PASS below:
 <div id="test" contentEditable="true">
-<blockquote type="cite" id="blockquote" style="font-size: 0px; -webkit-min-logical-height: 4px;"><br></blockquote>
+<blockquote type="cite" id="blockquote" style="font-size: 0px; min-block-size: 4px;"><br></blockquote>
 </div>
 <script>
 if (window.testRunner)
diff --git a/third_party/WebKit/LayoutTests/editing/selection/modify_move/move-by-word-visually-crash-test-5.html b/third_party/WebKit/LayoutTests/editing/selection/modify_move/move-by-word-visually-crash-test-5.html
index 5606a3d..e7a6276f 100644
--- a/third_party/WebKit/LayoutTests/editing/selection/modify_move/move-by-word-visually-crash-test-5.html
+++ b/third_party/WebKit/LayoutTests/editing/selection/modify_move/move-by-word-visually-crash-test-5.html
@@ -12,4 +12,4 @@
 <div dir=ltr title="test_for_crash"
 class="test_move_by_word"
 ><plaintext>class="test_move_by_word"
-style="-wap-marquee-style:mix; marker:sliderthumb-vertical; position:-0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001; border-top-style:rl-tb; -webkit-padding-after:absolute; position:destination-atop; border:ethiopic-halehame-ti-et; clip-rule:absolute; position:menulist-text; color-profile:-webkit-activelink; position:reset-size; margin-right:message-box; filter:absolute; position:no-punctuation; -webkit-border-end:no-punctuation; border-bottom-style:absolute; position:line-through; text-underline-mode:repeat-y; clip-rule:absolute; position:amharic-abegede; -webkit-text-fill-color:nowrap; -webkit-text-decorations-in-effect:absolute; position:sliderthumb-horizontal; border-top-right-radius:wave; border-top:absolute; position:up; border-right:not-allowed; enable-background:absolute; position:caption; max-height:ne-resize;
+style="-wap-marquee-style:mix; marker:sliderthumb-vertical; position:-0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001; border-top-style:rl-tb; padding-block-end:absolute; position:destination-atop; border:ethiopic-halehame-ti-et; clip-rule:absolute; position:menulist-text; color-profile:-webkit-activelink; position:reset-size; margin-right:message-box; filter:absolute; position:no-punctuation; border-inline-end:no-punctuation; border-bottom-style:absolute; position:line-through; text-underline-mode:repeat-y; clip-rule:absolute; position:amharic-abegede; -webkit-text-fill-color:nowrap; -webkit-text-decorations-in-effect:absolute; position:sliderthumb-horizontal; border-top-right-radius:wave; border-top:absolute; position:up; border-right:not-allowed; enable-background:absolute; position:caption; max-height:ne-resize;
diff --git a/third_party/WebKit/LayoutTests/editing/text-iterator/first_letter_find_string_crash.html b/third_party/WebKit/LayoutTests/editing/text-iterator/first_letter_find_string_crash.html
index 9e84c9270..17155e1 100644
--- a/third_party/WebKit/LayoutTests/editing/text-iterator/first_letter_find_string_crash.html
+++ b/third_party/WebKit/LayoutTests/editing/text-iterator/first_letter_find_string_crash.html
@@ -12,7 +12,7 @@
 
 * {
   -webkit-rtl-ordering: visual;
-  -webkit-margin-end: -1px
+  margin-inline-end: -1px
 }
 </style>
 <font dir="rtl">iC <q></q></font>
diff --git a/third_party/WebKit/LayoutTests/external/WPT_BASE_MANIFEST.json b/third_party/WebKit/LayoutTests/external/WPT_BASE_MANIFEST.json
index 5a010e31..3325142 100644
--- a/third_party/WebKit/LayoutTests/external/WPT_BASE_MANIFEST.json
+++ b/third_party/WebKit/LayoutTests/external/WPT_BASE_MANIFEST.json
@@ -6375,6 +6375,18 @@
      {}
     ]
    ],
+   "2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html": [
+    [
+     "/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html",
+     [
+      [
+       "/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
    "2dcontext/line-styles/canvas_linestyles_linecap_001.htm": [
     [
      "/2dcontext/line-styles/canvas_linestyles_linecap_001.htm",
@@ -47107,6 +47119,150 @@
      {}
     ]
    ],
+   "css/css-lists/content-property/marker-text-matches-armenian.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-armenian.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-armenian-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-circle.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-circle.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-circle-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-decimal.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-decimal.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-decimal-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-disc.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-disc.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-disc-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-georgian.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-georgian.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-georgian-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-greek.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-lower-greek.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-lower-greek-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-latin.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-lower-latin.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-lower-latin-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-roman.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-lower-roman.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-lower-roman-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-square.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-square.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-square-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-upper-latin.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-upper-latin.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-upper-latin-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-upper-roman.html": [
+    [
+     "/css/css-lists/content-property/marker-text-matches-upper-roman.html",
+     [
+      [
+       "/css/css-lists/content-property/marker-text-matches-upper-roman-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
    "css/css-lists/counter-increment-inside-display-contents.html": [
     [
      "/css/css-lists/counter-increment-inside-display-contents.html",
@@ -52279,6 +52435,18 @@
      {}
     ]
    ],
+   "css/css-pseudo/marker-inherit-line-height.html": [
+    [
+     "/css/css-pseudo/marker-inherit-line-height.html",
+     [
+      [
+       "/css/css-pseudo/marker-inherit-line-height-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
    "css/css-pseudo/marker-inherit-values.html": [
     [
      "/css/css-pseudo/marker-inherit-values.html",
@@ -57595,18 +57763,6 @@
      {}
     ]
    ],
-   "css/css-text/overflow-wrap/overflow-wrap-break-word-004.html": [
-    [
-     "/css/css-text/overflow-wrap/overflow-wrap-break-word-004.html",
-     [
-      [
-       "/css/css-text/overflow-wrap/reference/overflow-wrap-break-word-001-ref.html",
-       "=="
-      ]
-     ],
-     {}
-    ]
-   ],
    "css/css-text/overflow-wrap/overflow-wrap-break-word-fit-content-001.html": [
     [
      "/css/css-text/overflow-wrap/overflow-wrap-break-word-fit-content-001.html",
@@ -59587,30 +59743,6 @@
      {}
     ]
    ],
-   "css/css-text/word-break/word-break-break-all-010.html": [
-    [
-     "/css/css-text/word-break/word-break-break-all-010.html",
-     [
-      [
-       "/css/css-text/word-break/reference/word-break-break-all-010-ref.html",
-       "=="
-      ]
-     ],
-     {}
-    ]
-   ],
-   "css/css-text/word-break/word-break-break-all-011.html": [
-    [
-     "/css/css-text/word-break/word-break-break-all-011.html",
-     [
-      [
-       "/css/css-text/word-break/reference/word-break-break-all-010-ref.html",
-       "=="
-      ]
-     ],
-     {}
-    ]
-   ],
    "css/css-text/word-break/word-break-keep-all-000.html": [
     [
      "/css/css-text/word-break/word-break-keep-all-000.html",
@@ -95871,6 +96003,18 @@
      {}
     ]
    ],
+   "svg/linking/reftests/use-descendant-combinator-003.html": [
+    [
+     "/svg/linking/reftests/use-descendant-combinator-003.html",
+     [
+      [
+       "/svg/linking/reftests/use-descendant-combinator-ref.html",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
    "svg/painting/currentColor-override-pserver-fallback.svg": [
     [
      "/svg/painting/currentColor-override-pserver-fallback.svg",
@@ -96063,6 +96207,30 @@
      {}
     ]
    ],
+   "svg/render/reftests/blending-001.svg": [
+    [
+     "/svg/render/reftests/blending-001.svg",
+     [
+      [
+       "/svg/render/reftests/blending-001-ref.svg",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
+   "svg/render/reftests/blending-002.svg": [
+    [
+     "/svg/render/reftests/blending-002.svg",
+     [
+      [
+       "/svg/render/reftests/blending-002-ref.svg",
+       "=="
+      ]
+     ],
+     {}
+    ]
+   ],
    "svg/rendering/order/z-index.svg": [
     [
      "/svg/rendering/order/z-index.svg",
@@ -100904,6 +101072,11 @@
      {}
     ]
    ],
+   "2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html": [
+    [
+     {}
+    ]
+   ],
    "2dcontext/drawing-model/.gitkeep": [
     [
      {}
@@ -102559,6 +102732,11 @@
      {}
     ]
    ],
+   "accelerometer/idlharness.https.window-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "accname/description_from_content_of_describedby_element-manual-expected.txt": [
     [
      {}
@@ -103454,6 +103632,26 @@
      {}
     ]
    ],
+   "background-fetch/idlharness.any-expected.txt": [
+    [
+     {}
+    ]
+   ],
+   "background-fetch/idlharness.any.sharedworker-expected.txt": [
+    [
+     {}
+    ]
+   ],
+   "background-fetch/idlharness.any.worker-expected.txt": [
+    [
+     {}
+    ]
+   ],
+   "background-fetch/idlharness.https.any.serviceworker-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "background-fetch/interfaces-worker.https-expected.txt": [
     [
      {}
@@ -123129,6 +123327,11 @@
      {}
     ]
    ],
+   "css/css-images/idlharness-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "css/css-images/linear-gradient-ref.html": [
     [
      {}
@@ -123504,6 +123707,66 @@
      {}
     ]
    ],
+   "css/css-lists/content-property/marker-text-matches-armenian-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-circle-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-decimal-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-disc-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-georgian-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-greek-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-latin-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-lower-roman-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-square-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-upper-latin-ref.html": [
+    [
+     {}
+    ]
+   ],
+   "css/css-lists/content-property/marker-text-matches-upper-roman-ref.html": [
+    [
+     {}
+    ]
+   ],
    "css/css-lists/counter-7-ref.html": [
     [
      {}
@@ -123534,6 +123797,11 @@
      {}
     ]
    ],
+   "css/css-logical/animation-003.tentative-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "css/css-logical/cascading-001-ref.html": [
     [
      {}
@@ -125059,6 +125327,11 @@
      {}
     ]
    ],
+   "css/css-pseudo/marker-inherit-line-height-ref.html": [
+    [
+     {}
+    ]
+   ],
    "css/css-pseudo/marker-inherit-values-ref.html": [
     [
      {}
@@ -128444,11 +128717,6 @@
      {}
     ]
    ],
-   "css/css-text/word-break/reference/word-break-break-all-010-ref.html": [
-    [
-     {}
-    ]
-   ],
    "css/css-text/word-break/reference/word-break-break-all-ref-000.html": [
     [
      {}
@@ -158309,11 +158577,6 @@
      {}
     ]
    ],
-   "infrastructure/server/wpt-server-http.sub-expected.txt": [
-    [
-     {}
-    ]
-   ],
    "infrastructure/testdriver/send_keys-expected.txt": [
     [
      {}
@@ -158509,6 +158772,11 @@
      {}
     ]
    ],
+   "interfaces/css-counter-styles.idl": [
+    [
+     {}
+    ]
+   ],
    "interfaces/css-font-loading.idl": [
     [
      {}
@@ -158519,6 +158787,11 @@
      {}
     ]
    ],
+   "interfaces/css-images.idl": [
+    [
+     {}
+    ]
+   ],
    "interfaces/css-masking.idl": [
     [
      {}
@@ -158529,6 +158802,11 @@
      {}
     ]
    ],
+   "interfaces/css-regions.idl": [
+    [
+     {}
+    ]
+   ],
    "interfaces/css-transitions.idl": [
     [
      {}
@@ -158669,6 +158947,11 @@
      {}
     ]
    ],
+   "interfaces/media-source.idl": [
+    [
+     {}
+    ]
+   ],
    "interfaces/mediacapture-depth.idl": [
     [
      {}
@@ -159209,6 +159492,16 @@
      {}
     ]
    ],
+   "media-source/idlharness.any-expected.txt": [
+    [
+     {}
+    ]
+   ],
+   "media-source/idlharness.any.worker-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "media-source/import_tests.sh": [
     [
      {}
@@ -161029,11 +161322,6 @@
      {}
     ]
    ],
-   "payment-method-basic-card/idlharness.window-expected.txt": [
-    [
-     {}
-    ]
-   ],
    "payment-method-basic-card/payment-request-canmakepayment-method.https-expected.txt": [
     [
      {}
@@ -164624,6 +164912,11 @@
      {}
     ]
    ],
+   "resize-observer/idlharness.window-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "resize-observer/resources/iframe.html": [
     [
      {}
@@ -165009,6 +165302,11 @@
      {}
     ]
    ],
+   "selection/idlharness.window-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "selection/interfaces-expected.txt": [
     [
      {}
@@ -165899,6 +166197,16 @@
      {}
     ]
    ],
+   "server-timing/resources/parsing/84.js": [
+    [
+     {}
+    ]
+   ],
+   "server-timing/resources/parsing/84.js.sub.headers": [
+    [
+     {}
+    ]
+   ],
    "server-timing/resources/parsing/9.js": [
     [
      {}
@@ -167134,11 +167442,6 @@
      {}
     ]
    ],
-   "service-workers/service-worker/resources/interfaces-idls.js": [
-    [
-     {}
-    ]
-   ],
    "service-workers/service-worker/resources/interfaces-worker.sub.js": [
     [
      {}
@@ -169019,6 +169322,16 @@
      {}
     ]
    ],
+   "svg/render/reftests/blending-001-ref.svg": [
+    [
+     {}
+    ]
+   ],
+   "svg/render/reftests/blending-002-ref.svg": [
+    [
+     {}
+    ]
+   ],
    "svg/rendering/order/z-index-ref.svg": [
     [
      {}
@@ -170434,6 +170747,11 @@
      {}
     ]
    ],
+   "webaudio/idlharness.https.window-expected.txt": [
+    [
+     {}
+    ]
+   ],
    "webaudio/js/buffer-loader.js": [
     [
      {}
@@ -171024,7 +171342,7 @@
      {}
     ]
    ],
-   "webrtc/RTCPeerConnection-addTransceiver-expected.txt": [
+   "webrtc/RTCPeerConnection-addTransceiver.https-expected.txt": [
     [
      {}
     ]
@@ -183901,9 +184219,9 @@
      {}
     ]
    ],
-   "accelerometer/idlharness.https.html": [
+   "accelerometer/idlharness.https.window.js": [
     [
-     "/accelerometer/idlharness.https.html",
+     "/accelerometer/idlharness.https.window.html",
      {}
     ]
    ],
@@ -183979,13 +184297,21 @@
      {}
     ]
    ],
-   "background-fetch/interfaces.https.any.js": [
+   "background-fetch/idlharness.any.js": [
     [
-     "/background-fetch/interfaces.https.any.html",
+     "/background-fetch/idlharness.any.html",
      {}
     ],
     [
-     "/background-fetch/interfaces.https.any.worker.html",
+     "/background-fetch/idlharness.any.sharedworker.html",
+     {}
+    ],
+    [
+     "/background-fetch/idlharness.any.worker.html",
+     {}
+    ],
+    [
+     "/background-fetch/idlharness.https.any.serviceworker.html",
      {}
     ]
    ],
@@ -192021,6 +192347,12 @@
      {}
     ]
    ],
+   "css/css-images/idlharness.html": [
+    [
+     "/css/css-images/idlharness.html",
+     {}
+    ]
+   ],
    "css/css-images/parsing/gradient-position-invalid.html": [
     [
      "/css/css-images/parsing/gradient-position-invalid.html",
@@ -192135,9 +192467,9 @@
      {}
     ]
    ],
-   "css/css-logical/animation-003.tenative.html": [
+   "css/css-logical/animation-003.tentative.html": [
     [
-     "/css/css-logical/animation-003.tenative.html",
+     "/css/css-logical/animation-003.tentative.html",
      {}
     ]
    ],
@@ -216011,15 +216343,15 @@
      {}
     ]
    ],
-   "html/rendering/non-replaced-elements/tables/form-in-tables.html": [
+   "html/rendering/non-replaced-elements/tables/form-in-tables-xhtml.xhtml": [
     [
-     "/html/rendering/non-replaced-elements/tables/form-in-tables.html",
+     "/html/rendering/non-replaced-elements/tables/form-in-tables-xhtml.xhtml",
      {}
     ]
    ],
-   "html/rendering/non-replaced-elements/tables/form-in-tables.xhtml": [
+   "html/rendering/non-replaced-elements/tables/form-in-tables.html": [
     [
-     "/html/rendering/non-replaced-elements/tables/form-in-tables.xhtml",
+     "/html/rendering/non-replaced-elements/tables/form-in-tables.html",
      {}
     ]
    ],
@@ -218979,6 +219311,14 @@
      }
     ]
    ],
+   "html/semantics/embedded-content/the-img-element/decode/image-decode-with-quick-attach.html": [
+    [
+     "/html/semantics/embedded-content/the-img-element/decode/image-decode-with-quick-attach.html",
+     {
+      "timeout": "long"
+     }
+    ]
+   ],
    "html/semantics/embedded-content/the-img-element/decode/image-decode.html": [
     [
      "/html/semantics/embedded-content/the-img-element/decode/image-decode.html",
@@ -220253,18 +220593,42 @@
      {}
     ]
    ],
+   "html/semantics/forms/the-textarea-element/textarea-maxlength.html": [
+    [
+     "/html/semantics/forms/the-textarea-element/textarea-maxlength.html",
+     {}
+    ]
+   ],
+   "html/semantics/forms/the-textarea-element/textarea-minlength.html": [
+    [
+     "/html/semantics/forms/the-textarea-element/textarea-minlength.html",
+     {}
+    ]
+   ],
    "html/semantics/forms/the-textarea-element/textarea-setcustomvalidity.html": [
     [
      "/html/semantics/forms/the-textarea-element/textarea-setcustomvalidity.html",
      {}
     ]
    ],
+   "html/semantics/forms/the-textarea-element/textarea-textLength.html": [
+    [
+     "/html/semantics/forms/the-textarea-element/textarea-textLength.html",
+     {}
+    ]
+   ],
    "html/semantics/forms/the-textarea-element/textarea-type.html": [
     [
      "/html/semantics/forms/the-textarea-element/textarea-type.html",
      {}
     ]
    ],
+   "html/semantics/forms/the-textarea-element/textarea-validity-clone.html": [
+    [
+     "/html/semantics/forms/the-textarea-element/textarea-validity-clone.html",
+     {}
+    ]
+   ],
    "html/semantics/forms/the-textarea-element/value-defaultValue-textContent-xhtml.xhtml": [
     [
      "/html/semantics/forms/the-textarea-element/value-defaultValue-textContent-xhtml.xhtml",
@@ -225895,9 +226259,13 @@
      {}
     ]
    ],
-   "media-source/interfaces.html": [
+   "media-source/idlharness.any.js": [
     [
-     "/media-source/interfaces.html",
+     "/media-source/idlharness.any.html",
+     {}
+    ],
+    [
+     "/media-source/idlharness.any.worker.html",
      {}
     ]
    ],
@@ -247621,9 +247989,9 @@
      {}
     ]
    ],
-   "resize-observer/idlharness.html": [
+   "resize-observer/idlharness.window.js": [
     [
-     "/resize-observer/idlharness.html",
+     "/resize-observer/idlharness.window.html",
      {}
     ]
    ],
@@ -248205,9 +248573,9 @@
      {}
     ]
    ],
-   "selection/interfaces.html": [
+   "selection/idlharness.window.js": [
     [
-     "/selection/interfaces.html",
+     "/selection/idlharness.window.html",
      {}
     ]
    ],
@@ -253789,9 +254157,9 @@
      {}
     ]
    ],
-   "web-share/idlharness.https.html": [
+   "web-share/idlharness.https.window.js": [
     [
-     "/web-share/idlharness.https.html",
+     "/web-share/idlharness.https.window.html",
      {}
     ]
    ],
@@ -253825,9 +254193,9 @@
      {}
     ]
    ],
-   "webaudio/idlharness.https.html": [
+   "webaudio/idlharness.https.window.js": [
     [
-     "/webaudio/idlharness.https.html",
+     "/webaudio/idlharness.https.window.html",
      {
       "timeout": "long"
      }
@@ -255701,9 +256069,9 @@
      {}
     ]
    ],
-   "webrtc/RTCPeerConnection-addTransceiver.html": [
+   "webrtc/RTCPeerConnection-addTransceiver.https.html": [
     [
-     "/webrtc/RTCPeerConnection-addTransceiver.html",
+     "/webrtc/RTCPeerConnection-addTransceiver.https.html",
      {}
     ]
    ],
@@ -258275,9 +258643,9 @@
      }
     ]
    ],
-   "webstorage/idlharness.html": [
+   "webstorage/idlharness.window.js": [
     [
-     "/webstorage/idlharness.html",
+     "/webstorage/idlharness.window.html",
      {}
     ]
    ],
@@ -268664,6 +269032,14 @@
    "3b15af010f2ce13316fed6fcab9d85e05484b60d",
    "testharness"
   ],
+  "2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html": [
+   "ec6a6d1111aae9ce051cd1a2503a5b01149ceca6",
+   "reftest"
+  ],
+  "2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html": [
+   "f09c2922fc630872519fc37de47f232ecc8cc677",
+   "support"
+  ],
   "2dcontext/drawing-images-to-the-canvas/drawimage_html_image.html": [
    "ec86f8f5c84628cd5f3b8673de8dde34dc372fd9",
    "testharness"
@@ -274600,8 +274976,12 @@
    "fccd52aeb8ce8476e31cd5a881c3284fa5c16556",
    "support"
   ],
-  "accelerometer/idlharness.https.html": [
-   "93a78732bf63dcdd30ca1004f7ab850d5e7faaae",
+  "accelerometer/idlharness.https.window-expected.txt": [
+   "b613fec755d4ffdd89b73e672ad6ad85f93a1a07",
+   "support"
+  ],
+  "accelerometer/idlharness.https.window.js": [
+   "80f5c68be13e93686b4851871cc115bad160b633",
    "testharness"
   ],
   "accname/description_from_content_of_describedby_element-manual-expected.txt": [
@@ -275380,6 +275760,26 @@
    "6177ea08e069fd1aca85fd3d772a53022b07f519",
    "support"
   ],
+  "background-fetch/idlharness.any-expected.txt": [
+   "6d0c8ed4939faa52b94fede0968b379edca7daa5",
+   "support"
+  ],
+  "background-fetch/idlharness.any.js": [
+   "5c0eebc307b9cdb534003593248b03004352dacd",
+   "testharness"
+  ],
+  "background-fetch/idlharness.any.sharedworker-expected.txt": [
+   "d3818aef8b559e61186ed89c5343601e8f8164f3",
+   "support"
+  ],
+  "background-fetch/idlharness.any.worker-expected.txt": [
+   "d3818aef8b559e61186ed89c5343601e8f8164f3",
+   "support"
+  ],
+  "background-fetch/idlharness.https.any.serviceworker-expected.txt": [
+   "bc0333837b7e504edcc444ef0e41fcf6d97d6baf",
+   "support"
+  ],
   "background-fetch/interfaces-worker.https-expected.txt": [
    "88211a0d44012d229af146366440ced5d12b0988",
    "support"
@@ -275392,10 +275792,6 @@
    "8a5cf2119205440349c364ccdeaa65ce3cf42791",
    "support"
   ],
-  "background-fetch/interfaces.https.any.js": [
-   "34a07e3b0425cb5a1b940c9581573e2b3322724f",
-   "testharness"
-  ],
   "background-fetch/interfaces.https.any.worker-expected.txt": [
    "66d01d6b1e71e7381f1a1b87b56372ffe850b27f",
    "support"
@@ -277205,7 +277601,7 @@
    "testharness"
   ],
   "content-security-policy/embedded-enforcement/idlharness.window.js": [
-   "e009d639fbb4ede1085c365038fb79d1e0625143",
+   "a64a6a47127c3a770d9190df2d0a3ad570e32427",
    "testharness"
   ],
   "content-security-policy/embedded-enforcement/iframe-csp-attribute.html": [
@@ -313964,6 +314360,14 @@
    "4b49a6bc4cfef048f3a4ac1d62feeb604e0d28a5",
    "reftest"
   ],
+  "css/css-images/idlharness-expected.txt": [
+   "deb34377105c08e4511c5a4c8b90c58972679287",
+   "support"
+  ],
+  "css/css-images/idlharness.html": [
+   "43a0b8582221d2cd3838fcf6ba93bc7198cbdbf1",
+   "testharness"
+  ],
   "css/css-images/image-fit-001.xht": [
    "868d9469f57e6013a024efa5e04ef2455d97296e",
    "visual"
@@ -314688,6 +315092,102 @@
    "3ddf25363a61baa64374747cdfadeb9329e87f81",
    "support"
   ],
+  "css/css-lists/content-property/marker-text-matches-armenian-ref.html": [
+   "36a124835188ab9172d59dcf8615d050f3b4b37d",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-armenian.html": [
+   "08e07d745f07a15806d532ab2b859a528ea58f0b",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-circle-ref.html": [
+   "5cf7f2cb8c066d220dfd441bb50aeec2f514bf25",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-circle.html": [
+   "3191e867c0d1a02840977838fd13e25e12cdb491",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html": [
+   "428d093039a3e489e0b5765b0dfe1a3587d1bf92",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html": [
+   "648d651fce939501916074632c5704764115fcd5",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-decimal-ref.html": [
+   "f7c2fcd3655546e3d8e6d5fdd82523db81c69927",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-decimal.html": [
+   "c00f56ab58e7fd3ce8a2fc281543ddcde05d26fb",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-disc-ref.html": [
+   "6eee62533f2d54b12a5c9b39e93c3224cb2ffafb",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-disc.html": [
+   "876d93ca33eac5044084c7a5d0e5e445f3770a98",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-georgian-ref.html": [
+   "60f3e664d51a2a71df464f470cb644904f21f612",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-georgian.html": [
+   "192b49e24f250a1b403ab4b4f815c2d916a17abd",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-greek-ref.html": [
+   "4f2732719698d51272bb5cccf891e64a016cef8d",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-greek.html": [
+   "967b17032b21f3a041fd69b00a7b10a78d6743e6",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-latin-ref.html": [
+   "0b9981e525a3bcfe86a1b837d32c70af5355f7c0",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-latin.html": [
+   "258697c2a8b72849da7e99db6fcb1e18bed313b1",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-roman-ref.html": [
+   "d0c2e9b1b84254e32956f9f26461f7d8751e17b8",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-lower-roman.html": [
+   "8507450cf81d498202277ad20f47d9047b82803c",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-square-ref.html": [
+   "1b69ce33b9dd6e674407f78937e47b3aa4d012af",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-square.html": [
+   "198deb47052925b046d8b9f37ddb2b2109d29c25",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-upper-latin-ref.html": [
+   "f2b5bb160a67b42add362f0bdd3d492f1992b3ff",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-upper-latin.html": [
+   "5640698b5296d42d403591796cb35984d4f44e3a",
+   "reftest"
+  ],
+  "css/css-lists/content-property/marker-text-matches-upper-roman-ref.html": [
+   "188a77cab595081eac21b18005a911cf2c3c53ec",
+   "support"
+  ],
+  "css/css-lists/content-property/marker-text-matches-upper-roman.html": [
+   "5b66ea59ac497741a1501c3c6d91764fb8d99e0d",
+   "reftest"
+  ],
   "css/css-lists/counter-7-ref.html": [
    "6391e214ad5262afaab7cd6caaf57e7f2506fb4d",
    "support"
@@ -314748,7 +315248,11 @@
    "ff44f2aadd8f274135c40ffc70a94b71df022718",
    "support"
   ],
-  "css/css-logical/animation-003.tenative.html": [
+  "css/css-logical/animation-003.tentative-expected.txt": [
+   "ff44f2aadd8f274135c40ffc70a94b71df022718",
+   "support"
+  ],
+  "css/css-logical/animation-003.tentative.html": [
    "bdb7e952eb7fecf402f64129a00b511d89470195",
    "testharness"
   ],
@@ -317940,6 +318444,14 @@
    "7d4739f2252956461b38d8b8566a84ead3c1d8b1",
    "reftest"
   ],
+  "css/css-pseudo/marker-inherit-line-height-ref.html": [
+   "e0b76bc582df84e630c911cd45a77bd714ae3727",
+   "support"
+  ],
+  "css/css-pseudo/marker-inherit-line-height.html": [
+   "d6da862bc43db77db9e22a1fd1090a26a497e8cc",
+   "reftest"
+  ],
   "css/css-pseudo/marker-inherit-values-ref.html": [
    "92bdc9d8f482c34ad389f27c957d4024a7e05b43",
    "support"
@@ -323280,10 +323792,6 @@
    "63bbe4f7fb67f33217876af58c1de4d032c369f4",
    "reftest"
   ],
-  "css/css-text/overflow-wrap/overflow-wrap-break-word-004.html": [
-   "ca2b98f9db6ac13da149380dab3311301d741045",
-   "reftest"
-  ],
   "css/css-text/overflow-wrap/overflow-wrap-break-word-fit-content-001.html": [
    "9f88a667825f8cb725dc348e17081e1a25b3f4de",
    "reftest"
@@ -324820,10 +325328,6 @@
    "5ad54c7b9197746f66fa30628b6059a369f5bd36",
    "support"
   ],
-  "css/css-text/word-break/reference/word-break-break-all-010-ref.html": [
-   "0b16a0bdb25ddd647ad96dd82e3430274667ee87",
-   "support"
-  ],
   "css/css-text/word-break/reference/word-break-break-all-ref-000.html": [
    "765afbeeede3d3dd8ad33b88927d6eb8e5435463",
    "support"
@@ -325000,14 +325504,6 @@
    "fd5f0cf9f57152d3c6df7c4b59c0fe70733bf9f4",
    "manual"
   ],
-  "css/css-text/word-break/word-break-break-all-010.html": [
-   "f9eedfe8e72bcf6693f8049bfdf2e1efdf3084ef",
-   "reftest"
-  ],
-  "css/css-text/word-break/word-break-break-all-011.html": [
-   "047e950374d902e426287900a2d03c63726420a2",
-   "reftest"
-  ],
   "css/css-text/word-break/word-break-keep-all-000.html": [
    "219e6b0435541a834ebbee114b0dd40e15956f75",
    "reftest"
@@ -366836,12 +367332,12 @@
    "1732dc6dd595354e21c7fb6006c32b844ceedf18",
    "reftest"
   ],
-  "html/rendering/non-replaced-elements/tables/form-in-tables.html": [
-   "8319c136723513a019d48f66c663ce3673777bcd",
+  "html/rendering/non-replaced-elements/tables/form-in-tables-xhtml.xhtml": [
+   "610932708cb18b8022ca64b8c9f3f59d8cf1ae56",
    "testharness"
   ],
-  "html/rendering/non-replaced-elements/tables/form-in-tables.xhtml": [
-   "610932708cb18b8022ca64b8c9f3f59d8cf1ae56",
+  "html/rendering/non-replaced-elements/tables/form-in-tables.html": [
+   "8319c136723513a019d48f66c663ce3673777bcd",
    "testharness"
   ],
   "html/rendering/non-replaced-elements/tables/hidden-attr-expected.txt": [
@@ -370668,6 +371164,10 @@
    "9405efb65176096957438cbdcc59109b488d80e6",
    "testharness"
   ],
+  "html/semantics/embedded-content/the-img-element/decode/image-decode-with-quick-attach.html": [
+   "cd7e495a854d848712b46cddec40e3ebffe9d3ba",
+   "testharness"
+  ],
   "html/semantics/embedded-content/the-img-element/decode/image-decode.html": [
    "5368b62bf6c950e8d57b16b36148e5695ce16fd8",
    "testharness"
@@ -371992,6 +372492,14 @@
    "b61235681689807b5d46b8aaca9ae6c7a18039f7",
    "testharness"
   ],
+  "html/semantics/forms/the-textarea-element/textarea-maxlength.html": [
+   "fb2796fe7e542bd9551c18c0176a4f822ee347cd",
+   "testharness"
+  ],
+  "html/semantics/forms/the-textarea-element/textarea-minlength.html": [
+   "fcccb00d0db7222af1fb03e7481ccf31e51ec924",
+   "testharness"
+  ],
   "html/semantics/forms/the-textarea-element/textarea-newline-bidi-ref.html": [
    "26eb3e615f8b0e15cf02b7ee18d39fd71b04da70",
    "support"
@@ -372012,10 +372520,18 @@
    "5f5932aeba8a0a7da66ee006a6e604780210d57b",
    "testharness"
   ],
+  "html/semantics/forms/the-textarea-element/textarea-textLength.html": [
+   "0f9b8cdaace12d4f4e7fbd57a90fcba7e826ced5",
+   "testharness"
+  ],
   "html/semantics/forms/the-textarea-element/textarea-type.html": [
    "81a270bc3c9304f8b2e7dd526519f4eab7d94f45",
    "testharness"
   ],
+  "html/semantics/forms/the-textarea-element/textarea-validity-clone.html": [
+   "d0334cee779937e5d800e478013e5c8ffeaa6a1c",
+   "testharness"
+  ],
   "html/semantics/forms/the-textarea-element/value-defaultValue-textContent-expected.txt": [
    "4f2f22b676dea822e14a2073fa9ea727488fdc95",
    "support"
@@ -377936,10 +378452,6 @@
    "d1220af2e8cc4fab720497311064666393eea7c8",
    "testharness"
   ],
-  "infrastructure/server/wpt-server-http.sub-expected.txt": [
-   "f43e81e2a389b59723ff9787014c6d96a030e13b",
-   "support"
-  ],
   "infrastructure/server/wpt-server-http.sub.html": [
    "2a400478de23a6aecf31bdc08b187784c36be629",
    "testharness"
@@ -378089,7 +378601,7 @@
    "support"
   ],
   "interfaces/background-fetch.idl": [
-   "272d5ef66a2df3a6c3fefaf9688802ed93a9ad5f",
+   "4f744971f4eb6fc90bfb5fa4cc7514230e5e57dd",
    "support"
   ],
   "interfaces/battery-status.idl": [
@@ -378140,6 +378652,10 @@
    "0019e54201874e5d2a2b10c887e44b8c42199c32",
    "support"
   ],
+  "interfaces/css-counter-styles.idl": [
+   "437dd5859385f5a17f9b9b2509b8f50eeb42a717",
+   "support"
+  ],
   "interfaces/css-font-loading.idl": [
    "9f2f252c5b63c159d9680de46a932bfa4335bf11",
    "support"
@@ -378148,6 +378664,10 @@
    "ff2d83e9468c743993c9b4a1ecf3fab09684dc16",
    "support"
   ],
+  "interfaces/css-images.idl": [
+   "29ff2de2d9dac38494a4e2b7cfc6dfc2947f16c1",
+   "support"
+  ],
   "interfaces/css-masking.idl": [
    "5f4ed3d8922e30ab3ddb714d185c6e6f794e5a29",
    "support"
@@ -378156,6 +378676,10 @@
    "9939fb7f08cab0f167e6e0762eac6ad94b2dfd9f",
    "support"
   ],
+  "interfaces/css-regions.idl": [
+   "23d01fa1c91539c443b2227df6a89e377b65a393",
+   "support"
+  ],
   "interfaces/css-transitions.idl": [
    "54dd701233a71187c8b675ebbf7336f852a7cace",
    "support"
@@ -378268,6 +378792,10 @@
    "dad6123e39feca39ec620de51307e56823cc5408",
    "support"
   ],
+  "interfaces/media-source.idl": [
+   "a3c8e49db54b906ae99e2aa2cc385c0e4d949a80",
+   "support"
+  ],
   "interfaces/mediacapture-depth.idl": [
    "2ac0dcf83f5afb227f24c9063d738dd22e941fe9",
    "support"
@@ -378373,7 +378901,7 @@
    "support"
   ],
   "interfaces/resource-timing.idl": [
-   "75fcdf5c6ac811254e1a025cd58d958f27ab5b5b",
+   "d500e42e860b94614c0efeeb0b46d2ad7e8d79f9",
    "support"
   ],
   "interfaces/screen-orientation.idl": [
@@ -378389,7 +378917,7 @@
    "support"
   ],
   "interfaces/selection-api.idl": [
-   "10e7cdeb0e3c5e3871c81e30e89e72412d3f572f",
+   "622e064c826b3d25618e0bd6041e4297f9939d78",
    "support"
   ],
   "interfaces/server-timing.idl": [
@@ -378397,7 +378925,7 @@
    "support"
   ],
   "interfaces/service-workers.idl": [
-   "979a3b43c2db10f5a93213f46f1abf90082da22b",
+   "d219b474e3981be9cdeed598046e8002b4b48ddf",
    "support"
   ],
   "interfaces/shape-detection-api.idl": [
@@ -378409,7 +378937,7 @@
    "support"
   ],
   "interfaces/touch-events.idl": [
-   "00811ff421c7ecafb8070d1a3d17310e1228c725",
+   "4e456ecce04a5f8fb19a7be5ae0d9e5bd69a1ec1",
    "support"
   ],
   "interfaces/touchevents.idl": [
@@ -378976,6 +379504,18 @@
    "fefa39b93290d8d97b18f606c5166d7792193251",
    "support"
   ],
+  "media-source/idlharness.any-expected.txt": [
+   "5d955cd235b47e9a85ec1992410b576835e88040",
+   "support"
+  ],
+  "media-source/idlharness.any.js": [
+   "d9b97ca6d0883ef11491723a5db5b9e92e32a380",
+   "testharness"
+  ],
+  "media-source/idlharness.any.worker-expected.txt": [
+   "8abee7dbcb3581e009040a8b4cab306a0f5e7f4c",
+   "support"
+  ],
   "media-source/import_tests.sh": [
    "943d26f264d59f2a7c01db30f5d91e72dafa9935",
    "support"
@@ -378984,10 +379524,6 @@
    "f7122a377e6ceb667a42f978dc549457d8c301e1",
    "support"
   ],
-  "media-source/interfaces.html": [
-   "f836da38e12662cd56590d5a0534ad555dbf249e",
-   "testharness"
-  ],
   "media-source/manifest.txt": [
    "d1e89d132c19294d2c7bb677668c03f3309b475e",
    "support"
@@ -388640,10 +389176,6 @@
    "01e6ebc11174b8de17212aaf004890685bd90fae",
    "manual"
   ],
-  "payment-method-basic-card/idlharness.window-expected.txt": [
-   "d6c9c468cebdcd639c1d1d17beba93b3808751b1",
-   "support"
-  ],
   "payment-method-basic-card/idlharness.window.js": [
    "0219c2651ea2bd20d1c694f4f450646a38b8513f",
    "testharness"
@@ -389101,7 +389633,7 @@
    "support"
   ],
   "picture-in-picture/disable-picture-in-picture.html": [
-   "d97bf03a6e2f991c173248f9e28bd713da19a6f4",
+   "f80cdccd1c6edcf6a859a1ae2315a9f65188d812",
    "testharness"
   ],
   "picture-in-picture/enter-picture-in-picture.html": [
@@ -389113,11 +389645,11 @@
    "testharness"
   ],
   "picture-in-picture/idlharness.window-expected.txt": [
-   "a048d5c50c8e843efdfb66389f95f16b3e1bca6a",
+   "a9cf0a080146f82378696b5732b1d56da3b10368",
    "support"
   ],
   "picture-in-picture/idlharness.window.js": [
-   "a2a5c08651d42ef14d4ffd1ac0fa0c56a7a2eb37",
+   "8861876d089790497afb99496c118ac24f43eadd",
    "testharness"
   ],
   "picture-in-picture/leave-picture-in-picture.html": [
@@ -389133,7 +389665,7 @@
    "testharness"
   ],
   "picture-in-picture/request-picture-in-picture-twice.html": [
-   "289c2decddd418722b3d5345b31706b4fad59ed3",
+   "c45cc223ec4336916d46c3d4db8bc4decbbd3332",
    "testharness"
   ],
   "picture-in-picture/request-picture-in-picture.html": [
@@ -389141,7 +389673,7 @@
    "testharness"
   ],
   "picture-in-picture/resources/picture-in-picture-helpers.js": [
-   "1d4f14127753148c7dfc4a79d2beec34410e416b",
+   "b41a3c97a777a69ab28eea3b33f464379303da3d",
    "support"
   ],
   "picture-in-picture/shadow-dom.html": [
@@ -389849,7 +390381,7 @@
    "testharness"
   ],
   "presentation-api/controlling-ua/idlharness.https.html": [
-   "75dca68e8398772129feacc28bfe574e9c8d402b",
+   "b369c2a6001fa89daffbee9dc588388c74c6badc",
    "testharness"
   ],
   "presentation-api/controlling-ua/reconnectToPresentation_notfound_error.https.html": [
@@ -398392,8 +398924,12 @@
    "369f73cd032f618992d1746184927d0e78a920e8",
    "support"
   ],
-  "resize-observer/idlharness.html": [
-   "2b1960834f7164e351cdbba119694830055dcf17",
+  "resize-observer/idlharness.window-expected.txt": [
+   "50f625b9289202a2da470abb4b42c2e3a119a6a3",
+   "support"
+  ],
+  "resize-observer/idlharness.window.js": [
+   "678952851933ad07531c0bebe6b9dc7be52dcdc3",
    "testharness"
   ],
   "resize-observer/notify.html": [
@@ -398433,7 +398969,7 @@
    "testharness"
   ],
   "resource-timing/idlharness.any.js": [
-   "a1bb5ee0f67683b42035fa683e9a984b0cf4d5dc",
+   "2065b03393c21199e49553ec99902aaddec74ce0",
    "testharness"
   ],
   "resource-timing/iframe-setdomain.sub.html": [
@@ -399056,14 +399592,18 @@
    "4b5b638ae5cdaab02fbe9cca255a119134238a7d",
    "testharness"
   ],
+  "selection/idlharness.window-expected.txt": [
+   "1bdfe84569e3607508422c62bc7b03ecfc2df38a",
+   "support"
+  ],
+  "selection/idlharness.window.js": [
+   "2a89076946fd12a1629ef962c2e459accba952df",
+   "testharness"
+  ],
   "selection/interfaces-expected.txt": [
    "8a53b2e46fc24e712365a3e1484acdca85752864",
    "support"
   ],
-  "selection/interfaces.html": [
-   "78e92642e2562da8040c791e4c3a7ad1de53e600",
-   "testharness"
-  ],
   "selection/isCollapsed.html": [
    "d1984a9359d880dfb81197e7ec31b2456833809d",
    "testharness"
@@ -399820,6 +400360,14 @@
    "ba24e1d89104fb25d5100dfa0b38388ca19ee9c1",
    "support"
   ],
+  "server-timing/resources/parsing/84.js": [
+   "aa4b60a397c4704689621874403329c705598c9c",
+   "support"
+  ],
+  "server-timing/resources/parsing/84.js.sub.headers": [
+   "94fd8cac84f6fdc22d8b46786ac690f788e657f4",
+   "support"
+  ],
   "server-timing/resources/parsing/9.js": [
    "5660bb21eb42f322a58f80c6c001d14c6e0fc0c9",
    "support"
@@ -399829,19 +400377,19 @@
    "support"
   ],
   "server-timing/server_timing_header-parsing-expected.txt": [
-   "87ae6268c574b3305a79fc1376d7a62aa7587f2d",
+   "3fb854f9b672fcac8995d77ba7d23e996c8f3485",
    "support"
   ],
   "server-timing/server_timing_header-parsing.html": [
-   "10f756bbf749b7ad8f7c6eb4efe752ee79c44b4a",
+   "8df39d7b393cb0ae6dac1997b2d8755210a97ea5",
    "testharness"
   ],
   "server-timing/server_timing_header-parsing.https-expected.txt": [
-   "ef70b0cf8dbcb0090957b0bba170ee610b11a39d",
+   "531ef8b2e35dc7b7984fb6ce4a84eead76fb3a12",
    "support"
   ],
   "server-timing/server_timing_header-parsing.https.html": [
-   "10f756bbf749b7ad8f7c6eb4efe752ee79c44b4a",
+   "8df39d7b393cb0ae6dac1997b2d8755210a97ea5",
    "testharness"
   ],
   "server-timing/service_worker_idl.https.html": [
@@ -400685,7 +401233,7 @@
    "testharness"
   ],
   "service-workers/service-worker/interfaces-window.https.html": [
-   "71c0e09861863b9f36571137d2398fa8af516601",
+   "18b96c6b13d9b561e2fdb1eca32872d6596c76b0",
    "testharness"
   ],
   "service-workers/service-worker/invalid-blobtype.https.html": [
@@ -401580,12 +402128,8 @@
    "0f2d441a80fb4641af35ad851e86690b7c02d1f7",
    "support"
   ],
-  "service-workers/service-worker/resources/interfaces-idls.js": [
-   "2d3ae3bdd1be17a2871aa3af2009daaacf43d3db",
-   "support"
-  ],
   "service-workers/service-worker/resources/interfaces-worker.sub.js": [
-   "694568194185de15fb912b3e4b4655ea3f65420d",
+   "cd067da4cc6683916037aaf13d71f6309f04460b",
    "support"
   ],
   "service-workers/service-worker/resources/invalid-blobtype-iframe.https.html": [
@@ -404892,6 +405436,10 @@
    "3d51ca0fc007d52147e7ea03493cac7ee1bb7903",
    "reftest"
   ],
+  "svg/linking/reftests/use-descendant-combinator-003.html": [
+   "d9155d3b92ecf0735f82ed9a0f2a8fd3fc380d55",
+   "reftest"
+  ],
   "svg/linking/reftests/use-descendant-combinator-ref.html": [
    "fb8aec792684b97151d2964b85d1e70829e141ad",
    "support"
@@ -405048,6 +405596,22 @@
    "03751dba3c2dfc7b4d478879dc94878b0dffcc99",
    "reftest"
   ],
+  "svg/render/reftests/blending-001-ref.svg": [
+   "2eab909f1ad53777522f75b968e73879236573f9",
+   "support"
+  ],
+  "svg/render/reftests/blending-001.svg": [
+   "5503dc3d96e7fa0c5884f6692469010fe28e4527",
+   "reftest"
+  ],
+  "svg/render/reftests/blending-002-ref.svg": [
+   "cf08e0892c32439a738b1eef93683702b6139990",
+   "support"
+  ],
+  "svg/render/reftests/blending-002.svg": [
+   "f77800e686cee3007fb7c40574c61b22d65217ef",
+   "reftest"
+  ],
   "svg/rendering/order/z-index-ref.svg": [
    "955f465ece3132da0bdd1e3b832e97db84f22b55",
    "support"
@@ -405449,7 +406013,7 @@
    "support"
   ],
   "touch-events/idlharness.window.js": [
-   "c5bcdd44381f6f49f0e653c48d05cc14c93f424a",
+   "8a4eee7d4002836a205a1cd0aceb3a8008a1a929",
    "testharness"
   ],
   "touch-events/multi-touch-interactions.js": [
@@ -407608,8 +408172,8 @@
    "07eb3c26df43537e0735d741c8bdc8c041b91175",
    "support"
   ],
-  "web-share/idlharness.https.html": [
-   "17e370aefd324ffac6a6f4b0211fbaecd1781ad4",
+  "web-share/idlharness.https.window.js": [
+   "cd9160bc9790e6e734a73a1f0656f70f8583e246",
    "testharness"
   ],
   "web-share/resources/manual-helper.js": [
@@ -407705,15 +408269,19 @@
    "support"
   ],
   "webaudio/historical.html": [
-   "93068df297042344669093ce899f0230c87ebf54",
+   "c6e3c7d6751731c708edfb0f4e32df8a6a3b80b0",
    "testharness"
   ],
   "webaudio/idlharness.https-expected.txt": [
    "451ba6e7843b5bc4e64983f0ac2aac11467f91e4",
    "support"
   ],
-  "webaudio/idlharness.https.html": [
-   "0403de985b3346240ba1d4b465a8e0838f3860bc",
+  "webaudio/idlharness.https.window-expected.txt": [
+   "77fb40fa0aeedc7f4d8e90d1d51f847895128fbf",
+   "support"
+  ],
+  "webaudio/idlharness.https.window.js": [
+   "cadc212b4ecb16e95581e34886f49f0357c593bc",
    "testharness"
   ],
   "webaudio/js/buffer-loader.js": [
@@ -409432,11 +410000,11 @@
    "74096ec62288029bf7aea81008e63e43a5f40549",
    "testharness"
   ],
-  "webrtc/RTCPeerConnection-addTransceiver-expected.txt": [
+  "webrtc/RTCPeerConnection-addTransceiver.https-expected.txt": [
    "f081cfd4da18191f04c211a70735b7142907e61f",
    "support"
   ],
-  "webrtc/RTCPeerConnection-addTransceiver.html": [
+  "webrtc/RTCPeerConnection-addTransceiver.https.html": [
    "3d1531048c244d0f3391d6fe30e7c8bc7e13ef35",
    "testharness"
   ],
@@ -409537,7 +410105,7 @@
    "testharness"
   ],
   "webrtc/RTCPeerConnection-helper.js": [
-   "e92d33cfb3fdacc6a2f172ecfe3444a1ced5680e",
+   "4c2699a8df315a475f1037a3ac8dcbb313f2757b",
    "support"
   ],
   "webrtc/RTCPeerConnection-iceConnectionState-expected.txt": [
@@ -409829,11 +410397,11 @@
    "testharness"
   ],
   "webrtc/RTCRtpSender-replaceTrack.https-expected.txt": [
-   "a6b5ff94414685af0bb1f94f914c47270f26f7b5",
+   "0addd08af3b15ccf95020bdcd609732d37535a50",
    "support"
   ],
   "webrtc/RTCRtpSender-replaceTrack.https.html": [
-   "ea6ff719d0939e22fd5af2c72af009a1c9602c93",
+   "c73569cdd7272bdda237661599f6c1f62b4c1f0c",
    "testharness"
   ],
   "webrtc/RTCRtpSender-setParameters-expected.txt": [
@@ -411152,8 +411720,8 @@
    "6ba7d14a894e782495d818e43f4b0901607c207e",
    "support"
   ],
-  "webstorage/idlharness.html": [
-   "9bbb03ed7a846411183963fba16241a107652c0f",
+  "webstorage/idlharness.window.js": [
+   "4aeab6abb761637eeb0d03e6f95dcc5c8a9e00c1",
    "testharness"
   ],
   "webstorage/missing_arguments.html": [
diff --git a/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html b/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html
new file mode 100644
index 0000000..83cf535
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self.html
@@ -0,0 +1,17 @@
+<link rel="match" href="drawimage_canvas_self_ref.html">
+<canvas id="dest" height="100" width="100"></canvas>
+<script>
+var canvasWidth = canvasHeight = 100;
+var destWidth = canvasWidth / 4;
+var destHeight = canvasHeight / 4;
+var destCanvas = document.getElementById('dest');
+var destCtx = destCanvas.getContext('2d');
+
+destCtx.fillStyle = 'red';
+destCtx.fillRect(0, 0,  canvasWidth, canvasHeight);
+destCtx.fillStyle = 'green';
+destCtx.fillRect(0, 0, canvasWidth / 2, canvasHeight / 2);
+destCtx.drawImage(destCanvas,
+                  0, 0, destWidth, destHeight,
+                  canvasWidth / 2, canvasHeight / 2, destWidth, destHeight);
+</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html b/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html
new file mode 100644
index 0000000..9f297ca
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/2dcontext/drawing-images-to-the-canvas/drawimage_canvas_self_ref.html
@@ -0,0 +1,11 @@
+<canvas id="dest" height="100" width="100"></canvas>
+<script>
+var canvasWidth = canvasHeight = 100;
+var destCanvas = document.getElementById('dest');
+var destCtx = destCanvas.getContext('2d');
+destCtx.fillStyle = 'red';
+destCtx.fillRect(0, 0,  canvasWidth, canvasHeight);
+destCtx.fillStyle = 'green';
+destCtx.fillRect(0, 0, canvasWidth / 2, canvasHeight / 2);
+destCtx.fillRect(canvasWidth / 2, canvasHeight / 2, canvasWidth / 4, canvasHeight / 4);
+</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.html b/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.html
deleted file mode 100644
index 1f32be9..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.html
+++ /dev/null
@@ -1,39 +0,0 @@
-<!DOCTYPE html>
-<meta charset="utf-8">
-<title>Accelerometer Sensor IDL tests</title>
-<link rel="author" title="Intel" href="http://www.intel.com">
-<link rel="help" href="https://w3c.github.io/accelerometer/">
-<link rel="help" href="https://w3c.github.io/sensors/">
-<script src="/resources/testharness.js"></script>
-<script src="/resources/testharnessreport.js"></script>
-<script src="/resources/WebIDLParser.js"></script>
-<script src="/resources/idlharness.js"></script>
-<script>
-"use strict";
-
-function doTest([dom, generic_sensor, accelerometer]) {
-  const idl_array = new IdlArray();
-  idl_array.add_untested_idls(dom);
-  idl_array.add_untested_idls('interface EventHandler {};');
-  idl_array.add_idls(generic_sensor, { only: ['Sensor', 'SensorOptions'] });
-  idl_array.add_idls(accelerometer);
-  idl_array.add_objects({
-    Accelerometer: ['new Accelerometer();'],
-    LinearAccelerationSensor: ['new LinearAccelerationSensor();'],
-    GravitySensor: ['new GravitySensor();']
-  });
-  idl_array.test();
-}
-
-function fetchText(url) {
-  return fetch(url).then((response) => response.text());
-}
-
-promise_test(() => {
-  return Promise.all([
-    "/interfaces/dom.idl",
-    "/interfaces/sensors.idl",
-    "/interfaces/accelerometer.idl",
-  ].map(fetchText)).then(doTest);
-}, "Test IDL implementation of Accelerometer Sensor");
-</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window-expected.txt
new file mode 100644
index 0000000..cacfccf
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window-expected.txt
@@ -0,0 +1,43 @@
+This is a testharness.js-based test.
+PASS Test IDL implementation of Accelerometer Sensor
+PASS Accelerometer interface: existence and properties of interface object
+PASS Accelerometer interface object length
+PASS Accelerometer interface object name
+PASS Accelerometer interface: existence and properties of interface prototype object
+PASS Accelerometer interface: existence and properties of interface prototype object's "constructor" property
+PASS Accelerometer interface: existence and properties of interface prototype object's @@unscopables property
+PASS Accelerometer interface: attribute x
+PASS Unscopable handled correctly for x property on Accelerometer
+PASS Accelerometer interface: attribute y
+PASS Unscopable handled correctly for y property on Accelerometer
+PASS Accelerometer interface: attribute z
+PASS Unscopable handled correctly for z property on Accelerometer
+PASS Accelerometer must be primary interface of new Accelerometer();
+PASS Stringification of new Accelerometer();
+PASS Accelerometer interface: new Accelerometer(); must inherit property "x" with the proper type
+PASS Accelerometer interface: new Accelerometer(); must inherit property "y" with the proper type
+PASS Accelerometer interface: new Accelerometer(); must inherit property "z" with the proper type
+PASS LinearAccelerationSensor interface: existence and properties of interface object
+PASS LinearAccelerationSensor interface object length
+PASS LinearAccelerationSensor interface object name
+PASS LinearAccelerationSensor interface: existence and properties of interface prototype object
+PASS LinearAccelerationSensor interface: existence and properties of interface prototype object's "constructor" property
+PASS LinearAccelerationSensor interface: existence and properties of interface prototype object's @@unscopables property
+PASS LinearAccelerationSensor must be primary interface of new LinearAccelerationSensor();
+PASS Stringification of new LinearAccelerationSensor();
+PASS Accelerometer interface: new LinearAccelerationSensor(); must inherit property "x" with the proper type
+PASS Accelerometer interface: new LinearAccelerationSensor(); must inherit property "y" with the proper type
+PASS Accelerometer interface: new LinearAccelerationSensor(); must inherit property "z" with the proper type
+FAIL GravitySensor interface: existence and properties of interface object assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor interface object length assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor interface object name assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor interface: existence and properties of interface prototype object assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "GravitySensor" expected property "GravitySensor" missing
+FAIL GravitySensor must be primary interface of new GravitySensor(); assert_equals: Unexpected exception when evaluating object expected null but got object "ReferenceError: GravitySensor is not defined"
+FAIL Stringification of new GravitySensor(); assert_equals: Unexpected exception when evaluating object expected null but got object "ReferenceError: GravitySensor is not defined"
+FAIL Accelerometer interface: new GravitySensor(); must inherit property "x" with the proper type assert_equals: Unexpected exception when evaluating object expected null but got object "ReferenceError: GravitySensor is not defined"
+FAIL Accelerometer interface: new GravitySensor(); must inherit property "y" with the proper type assert_equals: Unexpected exception when evaluating object expected null but got object "ReferenceError: GravitySensor is not defined"
+FAIL Accelerometer interface: new GravitySensor(); must inherit property "z" with the proper type assert_equals: Unexpected exception when evaluating object expected null but got object "ReferenceError: GravitySensor is not defined"
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window.js b/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window.js
new file mode 100644
index 0000000..f6cbcd96
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/accelerometer/idlharness.https.window.js
@@ -0,0 +1,19 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+// https://w3c.github.io/accelerometer/
+
+"use strict";
+
+idl_test(
+  ['accelerometer'],
+  ['generic-sensor', 'dom'],
+  idl_array => {
+    idl_array.add_objects({
+      Accelerometer: ['new Accelerometer();'],
+      LinearAccelerationSensor: ['new LinearAccelerationSensor();'],
+      GravitySensor: ['new GravitySensor();']
+    });
+  },
+  'Test IDL implementation of Accelerometer Sensor'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any-expected.txt
new file mode 100644
index 0000000..6faed0f
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any-expected.txt
@@ -0,0 +1,80 @@
+This is a testharness.js-based test.
+Found 76 tests; 58 PASS, 18 FAIL, 0 TIMEOUT, 0 NOTRUN.
+PASS background-fetch interfaces
+PASS Partial interface ServiceWorkerGlobalScope: original interface defined
+PASS Partial interface ServiceWorkerRegistration: original interface defined
+PASS BackgroundFetchManager interface: existence and properties of interface object
+PASS BackgroundFetchManager interface object length
+PASS BackgroundFetchManager interface object name
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchManager interface: operation fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions)
+PASS Unscopable handled correctly for fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation get(DOMString)
+PASS Unscopable handled correctly for get(DOMString) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation getIds()
+PASS Unscopable handled correctly for getIds() on BackgroundFetchManager
+PASS BackgroundFetchRegistration interface: existence and properties of interface object
+PASS BackgroundFetchRegistration interface object length
+PASS BackgroundFetchRegistration interface object name
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchRegistration interface: attribute id
+PASS Unscopable handled correctly for id property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploadTotal
+PASS Unscopable handled correctly for uploadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploaded
+PASS Unscopable handled correctly for uploaded property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloadTotal
+PASS Unscopable handled correctly for downloadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloaded
+PASS Unscopable handled correctly for downloaded property on BackgroundFetchRegistration
+FAIL BackgroundFetchRegistration interface: attribute activeFetches assert_true: The prototype object must have a property "activeFetches" expected true got false
+PASS Unscopable handled correctly for activeFetches property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute onprogress
+PASS Unscopable handled correctly for onprogress property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: operation abort()
+PASS Unscopable handled correctly for abort() on BackgroundFetchRegistration
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: operation match(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for match(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation matchAll(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for matchAll(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation values() assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for values() on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: attribute responseReady assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+PASS Unscopable handled correctly for responseReady property on BackgroundFetchActiveFetch
+PASS BackgroundFetchFetch interface: existence and properties of interface object
+PASS BackgroundFetchFetch interface object length
+PASS BackgroundFetchFetch interface object name
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchFetch interface: attribute request
+PASS Unscopable handled correctly for request property on BackgroundFetchFetch
+PASS BackgroundFetchEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface object
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface object
+PASS BackgroundFetchClickEvent interface: existence and properties of interface object
+FAIL ServiceWorkerRegistration interface: attribute backgroundFetch assert_own_property: self does not have own property "ServiceWorkerRegistration" expected property "ServiceWorkerRegistration" missing
+PASS Unscopable handled correctly for backgroundFetch property on ServiceWorkerRegistration
+PASS ServiceWorkerGlobalScope interface: existence and properties of interface object
+PASS ExtendableEvent interface: existence and properties of interface object
+PASS WorkerGlobalScope interface: existence and properties of interface object
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.js b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.js
new file mode 100644
index 0000000..ad20477
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.js
@@ -0,0 +1,25 @@
+// META: global=window,worker
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+'use strict';
+
+// https://wicg.github.io/background-fetch/
+
+idl_test(
+  ['background-fetch'],
+  ['service-workers', 'dedicated-workers', 'dom'],
+  idl_array => {
+    const isServiceWorker = location.pathname.includes('.serviceworker.');
+    if (isServiceWorker) {
+      idl_array.add_objects({
+        ServiceWorkerGlobalScope: ['self'],
+        ServiceWorkerRegistration: ['registration'],
+        BackgroundFetchManager: ['registration.backgroundFetch'],
+        BackgroundFetchEvent: ['new BackgroundFetchEvent("type")'],
+        BackgroundFetchUpdateEvent: ['new BackgroundFetchUpdateEvent("type")'],
+      });
+    }
+  },
+  'background-fetch interfaces'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.sharedworker-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.sharedworker-expected.txt
new file mode 100644
index 0000000..5f3ed505
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.sharedworker-expected.txt
@@ -0,0 +1,79 @@
+This is a testharness.js-based test.
+Found 75 tests; 57 PASS, 18 FAIL, 0 TIMEOUT, 0 NOTRUN.
+PASS background-fetch interfaces
+PASS Partial interface ServiceWorkerGlobalScope: original interface defined
+PASS Partial interface ServiceWorkerRegistration: original interface defined
+PASS BackgroundFetchManager interface: existence and properties of interface object
+PASS BackgroundFetchManager interface object length
+PASS BackgroundFetchManager interface object name
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchManager interface: operation fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions)
+PASS Unscopable handled correctly for fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation get(DOMString)
+PASS Unscopable handled correctly for get(DOMString) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation getIds()
+PASS Unscopable handled correctly for getIds() on BackgroundFetchManager
+PASS BackgroundFetchRegistration interface: existence and properties of interface object
+PASS BackgroundFetchRegistration interface object length
+PASS BackgroundFetchRegistration interface object name
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchRegistration interface: attribute id
+PASS Unscopable handled correctly for id property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploadTotal
+PASS Unscopable handled correctly for uploadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploaded
+PASS Unscopable handled correctly for uploaded property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloadTotal
+PASS Unscopable handled correctly for downloadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloaded
+PASS Unscopable handled correctly for downloaded property on BackgroundFetchRegistration
+FAIL BackgroundFetchRegistration interface: attribute activeFetches assert_true: The prototype object must have a property "activeFetches" expected true got false
+PASS Unscopable handled correctly for activeFetches property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute onprogress
+PASS Unscopable handled correctly for onprogress property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: operation abort()
+PASS Unscopable handled correctly for abort() on BackgroundFetchRegistration
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: operation match(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for match(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation matchAll(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for matchAll(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation values() assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for values() on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: attribute responseReady assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+PASS Unscopable handled correctly for responseReady property on BackgroundFetchActiveFetch
+PASS BackgroundFetchFetch interface: existence and properties of interface object
+PASS BackgroundFetchFetch interface object length
+PASS BackgroundFetchFetch interface object name
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchFetch interface: attribute request
+PASS Unscopable handled correctly for request property on BackgroundFetchFetch
+PASS BackgroundFetchEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface object
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface object
+PASS BackgroundFetchClickEvent interface: existence and properties of interface object
+FAIL ServiceWorkerRegistration interface: attribute backgroundFetch assert_own_property: self does not have own property "ServiceWorkerRegistration" expected property "ServiceWorkerRegistration" missing
+PASS Unscopable handled correctly for backgroundFetch property on ServiceWorkerRegistration
+PASS ServiceWorkerGlobalScope interface: existence and properties of interface object
+PASS ExtendableEvent interface: existence and properties of interface object
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.worker-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.worker-expected.txt
new file mode 100644
index 0000000..5f3ed505
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.any.worker-expected.txt
@@ -0,0 +1,79 @@
+This is a testharness.js-based test.
+Found 75 tests; 57 PASS, 18 FAIL, 0 TIMEOUT, 0 NOTRUN.
+PASS background-fetch interfaces
+PASS Partial interface ServiceWorkerGlobalScope: original interface defined
+PASS Partial interface ServiceWorkerRegistration: original interface defined
+PASS BackgroundFetchManager interface: existence and properties of interface object
+PASS BackgroundFetchManager interface object length
+PASS BackgroundFetchManager interface object name
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchManager interface: operation fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions)
+PASS Unscopable handled correctly for fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation get(DOMString)
+PASS Unscopable handled correctly for get(DOMString) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation getIds()
+PASS Unscopable handled correctly for getIds() on BackgroundFetchManager
+PASS BackgroundFetchRegistration interface: existence and properties of interface object
+PASS BackgroundFetchRegistration interface object length
+PASS BackgroundFetchRegistration interface object name
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchRegistration interface: attribute id
+PASS Unscopable handled correctly for id property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploadTotal
+PASS Unscopable handled correctly for uploadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploaded
+PASS Unscopable handled correctly for uploaded property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloadTotal
+PASS Unscopable handled correctly for downloadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloaded
+PASS Unscopable handled correctly for downloaded property on BackgroundFetchRegistration
+FAIL BackgroundFetchRegistration interface: attribute activeFetches assert_true: The prototype object must have a property "activeFetches" expected true got false
+PASS Unscopable handled correctly for activeFetches property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute onprogress
+PASS Unscopable handled correctly for onprogress property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: operation abort()
+PASS Unscopable handled correctly for abort() on BackgroundFetchRegistration
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: operation match(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for match(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation matchAll(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for matchAll(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation values() assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for values() on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: attribute responseReady assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+PASS Unscopable handled correctly for responseReady property on BackgroundFetchActiveFetch
+PASS BackgroundFetchFetch interface: existence and properties of interface object
+PASS BackgroundFetchFetch interface object length
+PASS BackgroundFetchFetch interface object name
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchFetch interface: attribute request
+PASS Unscopable handled correctly for request property on BackgroundFetchFetch
+PASS BackgroundFetchEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface object
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface object
+PASS BackgroundFetchClickEvent interface: existence and properties of interface object
+FAIL ServiceWorkerRegistration interface: attribute backgroundFetch assert_own_property: self does not have own property "ServiceWorkerRegistration" expected property "ServiceWorkerRegistration" missing
+PASS Unscopable handled correctly for backgroundFetch property on ServiceWorkerRegistration
+PASS ServiceWorkerGlobalScope interface: existence and properties of interface object
+PASS ExtendableEvent interface: existence and properties of interface object
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.https.any.serviceworker-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.https.any.serviceworker-expected.txt
new file mode 100644
index 0000000..d897833
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/idlharness.https.any.serviceworker-expected.txt
@@ -0,0 +1,130 @@
+This is a testharness.js-based test.
+PASS background-fetch interfaces
+PASS Partial interface ServiceWorkerGlobalScope: original interface defined
+PASS Partial interface ServiceWorkerRegistration: original interface defined
+PASS BackgroundFetchManager interface: existence and properties of interface object
+PASS BackgroundFetchManager interface object length
+PASS BackgroundFetchManager interface object name
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchManager interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchManager interface: operation fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions)
+PASS Unscopable handled correctly for fetch(DOMString, [object Object],[object Object], BackgroundFetchOptions) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation get(DOMString)
+PASS Unscopable handled correctly for get(DOMString) on BackgroundFetchManager
+PASS BackgroundFetchManager interface: operation getIds()
+PASS Unscopable handled correctly for getIds() on BackgroundFetchManager
+PASS BackgroundFetchRegistration interface: existence and properties of interface object
+PASS BackgroundFetchRegistration interface object length
+PASS BackgroundFetchRegistration interface object name
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchRegistration interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchRegistration interface: attribute id
+PASS Unscopable handled correctly for id property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploadTotal
+PASS Unscopable handled correctly for uploadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute uploaded
+PASS Unscopable handled correctly for uploaded property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloadTotal
+PASS Unscopable handled correctly for downloadTotal property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute downloaded
+PASS Unscopable handled correctly for downloaded property on BackgroundFetchRegistration
+FAIL BackgroundFetchRegistration interface: attribute activeFetches assert_true: The prototype object must have a property "activeFetches" expected true got false
+PASS Unscopable handled correctly for activeFetches property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: attribute onprogress
+PASS Unscopable handled correctly for onprogress property on BackgroundFetchRegistration
+PASS BackgroundFetchRegistration interface: operation abort()
+PASS Unscopable handled correctly for abort() on BackgroundFetchRegistration
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+FAIL BackgroundFetchActiveFetches interface: operation match(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for match(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation matchAll(RequestInfo, CacheQueryOptions) assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for matchAll(RequestInfo, CacheQueryOptions) on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetches interface: operation values() assert_own_property: self does not have own property "BackgroundFetchActiveFetches" expected property "BackgroundFetchActiveFetches" missing
+PASS Unscopable handled correctly for values() on BackgroundFetchActiveFetches
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object length assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface object name assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+FAIL BackgroundFetchActiveFetch interface: attribute responseReady assert_own_property: self does not have own property "BackgroundFetchActiveFetch" expected property "BackgroundFetchActiveFetch" missing
+PASS Unscopable handled correctly for responseReady property on BackgroundFetchActiveFetch
+PASS BackgroundFetchFetch interface: existence and properties of interface object
+PASS BackgroundFetchFetch interface object length
+PASS BackgroundFetchFetch interface object name
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchFetch interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchFetch interface: attribute request
+PASS Unscopable handled correctly for request property on BackgroundFetchFetch
+PASS BackgroundFetchEvent interface: existence and properties of interface object
+PASS BackgroundFetchEvent interface object length
+PASS BackgroundFetchEvent interface object name
+PASS BackgroundFetchEvent interface: existence and properties of interface prototype object
+PASS BackgroundFetchEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchEvent interface: attribute id
+PASS Unscopable handled correctly for id property on BackgroundFetchEvent
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface object
+PASS BackgroundFetchSettledEvent interface object length
+PASS BackgroundFetchSettledEvent interface object name
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface prototype object
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchSettledEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchSettledEvent interface: attribute fetches
+PASS Unscopable handled correctly for fetches property on BackgroundFetchSettledEvent
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface object
+PASS BackgroundFetchSettledFetches interface object length
+PASS BackgroundFetchSettledFetches interface object name
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface prototype object
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchSettledFetches interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchSettledFetches interface: operation match(RequestInfo, CacheQueryOptions)
+PASS Unscopable handled correctly for match(RequestInfo, CacheQueryOptions) on BackgroundFetchSettledFetches
+FAIL BackgroundFetchSettledFetches interface: operation matchAll(RequestInfo, CacheQueryOptions) assert_own_property: interface prototype object missing non-static operation expected property "matchAll" missing
+PASS Unscopable handled correctly for matchAll(RequestInfo, CacheQueryOptions) on BackgroundFetchSettledFetches
+PASS BackgroundFetchSettledFetches interface: operation values()
+PASS Unscopable handled correctly for values() on BackgroundFetchSettledFetches
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface object
+FAIL BackgroundFetchSettledFetch interface object length assert_equals: wrong value for BackgroundFetchSettledFetch.length expected 0 but got 2
+PASS BackgroundFetchSettledFetch interface object name
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface prototype object
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchSettledFetch interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchSettledFetch interface: attribute response
+PASS Unscopable handled correctly for response property on BackgroundFetchSettledFetch
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface object
+PASS BackgroundFetchUpdateEvent interface object length
+PASS BackgroundFetchUpdateEvent interface object name
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface prototype object
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchUpdateEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchUpdateEvent interface: operation updateUI(DOMString)
+PASS Unscopable handled correctly for updateUI(DOMString) on BackgroundFetchUpdateEvent
+PASS BackgroundFetchClickEvent interface: existence and properties of interface object
+PASS BackgroundFetchClickEvent interface object length
+PASS BackgroundFetchClickEvent interface object name
+PASS BackgroundFetchClickEvent interface: existence and properties of interface prototype object
+PASS BackgroundFetchClickEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS BackgroundFetchClickEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS BackgroundFetchClickEvent interface: attribute state
+PASS Unscopable handled correctly for state property on BackgroundFetchClickEvent
+PASS ServiceWorkerRegistration interface: attribute backgroundFetch
+PASS Unscopable handled correctly for backgroundFetch property on ServiceWorkerRegistration
+PASS ServiceWorkerGlobalScope interface: attribute onbackgroundfetched
+PASS Unscopable handled correctly for onbackgroundfetched property on ServiceWorkerGlobalScope
+PASS ServiceWorkerGlobalScope interface: attribute onbackgroundfetchfail
+PASS Unscopable handled correctly for onbackgroundfetchfail property on ServiceWorkerGlobalScope
+PASS ServiceWorkerGlobalScope interface: attribute onbackgroundfetchabort
+PASS Unscopable handled correctly for onbackgroundfetchabort property on ServiceWorkerGlobalScope
+PASS ServiceWorkerGlobalScope interface: attribute onbackgroundfetchclick
+PASS Unscopable handled correctly for onbackgroundfetchclick property on ServiceWorkerGlobalScope
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/interfaces.https.any.js b/third_party/WebKit/LayoutTests/external/wpt/background-fetch/interfaces.https.any.js
deleted file mode 100644
index b95b856..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/background-fetch/interfaces.https.any.js
+++ /dev/null
@@ -1,24 +0,0 @@
-// META: script=/resources/WebIDLParser.js
-// META: script=/resources/idlharness.js
-
-'use strict';
-
-// https://wicg.github.io/background-fetch/
-
-promise_test(async () => {
-  const srcs = [
-    'background-fetch',
-    'dedicated-workers',
-    'service-workers',
-    'dom'
-  ];
-  const [idls, worker, serviceWorker, dom] = await Promise.all(
-      srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
-
-  var idlArray = new IdlArray();
-  idlArray.add_idls(idls);
-  idlArray.add_dependency_idls(serviceWorker);
-  idlArray.add_dependency_idls(worker);
-  idlArray.add_dependency_idls(dom);
-  idlArray.test();
-}, 'background-fetch interfaces');
diff --git a/third_party/WebKit/LayoutTests/external/wpt/content-security-policy/embedded-enforcement/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/content-security-policy/embedded-enforcement/idlharness.window.js
index 38fa663..6ac306c 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/content-security-policy/embedded-enforcement/idlharness.window.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/content-security-policy/embedded-enforcement/idlharness.window.js
@@ -5,17 +5,13 @@
 
 'use strict';
 
-promise_test(async () => {
-  const idl = await fetch('/interfaces/csp-embedded-enforcement.idl').then(r => r.text());
-  const html = await fetch('/interfaces/html.idl').then(r => r.text());
-  const dom = await fetch('/interfaces/dom.idl').then(r => r.text());
-
-  const idl_array = new IdlArray();
-  idl_array.add_idls(idl);
-  idl_array.add_dependency_idls(html);
-  idl_array.add_dependency_idls(dom);
-  idl_array.add_objects({
-    HTMLIFrameElement: ['document.createElement("iframe")'],
-  });
-  idl_array.test();
-}, 'csp-embedded-enforcement IDL');
+idl_test(
+  ['csp-embedded-enforcement'],
+  ['html', 'dom'],
+  idl_array => {
+    idl_array.add_objects({
+      HTMLIFrameElement: ['document.createElement("iframe")'],
+    });
+  },
+  'csp-embedded-enforcement IDL'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-flexbox/position-absolute-001.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-flexbox/position-absolute-001.html
index 3c3d593..f601740 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/css/css-flexbox/position-absolute-001.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-flexbox/position-absolute-001.html
@@ -692,7 +692,7 @@
                 flexbox.className = 'flexbox ' + flexboxClassName;
                 flexbox.setAttribute('style', mainAxis + ': 80px;' + crossAxis + ': 20px');
 
-                var baselineMargin = (flexDirection.indexOf('row') != -1) ? '-webkit-margin-before: 5px' : '-webkit-margin-start: 5px';
+                var baselineMargin = (flexDirection.indexOf('row') != -1) ? 'margin-block-start: 5px' : 'margin-inline-start: 5px';
 
                 var testExpectations = expectations[writingMode][flexDirection][direction][justifyContent];
                 addChild(flexbox, mainAxis, crossAxis, 40, 10, testExpectations['child1']);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness-expected.txt
new file mode 100644
index 0000000..594163a
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness-expected.txt
@@ -0,0 +1,5 @@
+This is a testharness.js-based test.
+PASS Test IDL implementation of css-masking
+FAIL Partial namespace CSS: original namespace defined assert_true: Original CSS definition should have type namespace expected true got false
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness.html
new file mode 100644
index 0000000..ec9c358
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-images/idlharness.html
@@ -0,0 +1,19 @@
+<!doctype html>
+<title>css-images IDL tests</title>
+<link rel="help" href="https://drafts.csswg.org/css-images-4/">
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src="/resources/WebIDLParser.js"></script>
+<script src="/resources/idlharness.js"></script>
+<script>
+  'use strict';
+
+  idl_test(
+    ['css-images'],
+    ['cssom'],
+    idl_array => {
+      // No objects,
+    },
+    'Test IDL implementation of css-masking'
+  );
+</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian-ref.html
new file mode 100644
index 0000000..f21dfff
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian-ref.html
@@ -0,0 +1,10 @@
+<!doctype html>
+<meta charset=utf8>
+<title>Reference: Inside list marker and normal text render identically: armenian</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>Ա. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian.html
new file mode 100644
index 0000000..fd0df63
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-armenian.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: armenian</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-armenian-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: armenian inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle-ref.html
new file mode 100644
index 0000000..60a342f1
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle-ref.html
@@ -0,0 +1,12 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: circle</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+p::before {
+    content: counter(test, circle);
+}
+</style>
+<p> Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle.html
new file mode 100644
index 0000000..138f73c
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-circle.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: circle</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-circle-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: circle inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html
new file mode 100644
index 0000000..997801d5
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: decimal-leading-zero</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>01. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html
new file mode 100644
index 0000000..5044d4e6
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-leading-zero.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: decimal-leading-zero</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-decimal-leading-zero-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: decimal-leading-zero inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-ref.html
new file mode 100644
index 0000000..8f97cc36
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: decimal</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>1. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal.html
new file mode 100644
index 0000000..8c991bc
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-decimal.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: decimal</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-decimal-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: decimal inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc-ref.html
new file mode 100644
index 0000000..e34a900
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc-ref.html
@@ -0,0 +1,12 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: disc</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+p::before {
+    content: counter(test, disc);
+}
+</style>
+<p> Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc.html
new file mode 100644
index 0000000..e2456cb
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-disc.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: disc</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-disc-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: disc inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian-ref.html
new file mode 100644
index 0000000..c400aca
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian-ref.html
@@ -0,0 +1,10 @@
+<!doctype html>
+<meta charset=utf8>
+<title>Reference: Inside list marker and normal text render identically: georgian</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>ა. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian.html
new file mode 100644
index 0000000..ef7615f
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-georgian.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: georgian</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-georgian-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: georgian inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek-ref.html
new file mode 100644
index 0000000..f28db82
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek-ref.html
@@ -0,0 +1,10 @@
+<!doctype html>
+<meta charset=utf8>
+<title>Reference: Inside list marker and normal text render identically: lower-greek</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>α. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek.html
new file mode 100644
index 0000000..96b5d811
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-greek.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: lower-greek</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-lower-greek-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: lower-greek inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin-ref.html
new file mode 100644
index 0000000..b86fe8ec
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: lower-latin</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>a. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin.html
new file mode 100644
index 0000000..0f6f242
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-latin.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: lower-latin</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-lower-latin-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: lower-latin inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman-ref.html
new file mode 100644
index 0000000..92dfa3d
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: lower-roman</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>i. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman.html
new file mode 100644
index 0000000..db7301b
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-lower-roman.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: lower-roman</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-lower-roman-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: lower-roman inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square-ref.html
new file mode 100644
index 0000000..99b64a54
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square-ref.html
@@ -0,0 +1,12 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: square</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+p::before {
+    content: counter(test, square);
+}
+</style>
+<p> Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square.html
new file mode 100644
index 0000000..9095017
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-square.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: square</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-square-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: square inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin-ref.html
new file mode 100644
index 0000000..6cbb712
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: upper-latin</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>A. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin.html
new file mode 100644
index 0000000..903ef8f5
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-latin.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: upper-latin</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-upper-latin-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: upper-latin inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman-ref.html
new file mode 100644
index 0000000..6fca49f2
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman-ref.html
@@ -0,0 +1,9 @@
+<!doctype html>
+<title>Reference: Inside list marker and normal text render identically: upper-roman</title>
+<style>
+* {
+    margin: 0;
+    padding: 0;
+}
+</style>
+<p>I. Filler Text
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman.html
new file mode 100644
index 0000000..69f34d74
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-lists/content-property/marker-text-matches-upper-roman.html
@@ -0,0 +1,18 @@
+<!doctype html>
+<title>Inside list marker and normal text render identically: upper-roman</title>
+<link rel=help href="https://drafts.csswg.org/css-lists-3/#content-property">
+<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=787382">
+<link rel=match href="marker-text-matches-upper-roman-ref.html">
+<style>
+* {
+    padding: 0;
+    margin: 0;
+}
+ol {
+    list-style: upper-roman inside;
+}
+</style>
+
+<ol>
+  <li>Filler Text</li>
+</ol>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tentative-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tentative-expected.txt
new file mode 100644
index 0000000..24c710be
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tentative-expected.txt
@@ -0,0 +1,4 @@
+This is a testharness.js-based test.
+FAIL Logical properties are represented as physical properties in keyframes assert_own_property: expected property "height" missing
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tenative.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tentative.html
similarity index 100%
rename from third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tenative.html
rename to third_party/WebKit/LayoutTests/external/wpt/css/css-logical/animation-003.tentative.html
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height-ref.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height-ref.html
new file mode 100644
index 0000000..fe57a281f
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height-ref.html
@@ -0,0 +1,16 @@
+<!doctype html>
+<title>Reference: Check that :marker inherits line-height</title>
+<style>
+div {
+  font: 128px/1 Ahem;
+  border: 2px solid blue;
+  background: orange;
+  color: black;
+}
+</style>
+
+<p>There should be two black rectangles below with no space between them and
+  the border above/below.</p>
+<div>
+  1. X
+</div>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height.html b/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height.html
new file mode 100644
index 0000000..145aedcb
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/css/css-pseudo/marker-inherit-line-height.html
@@ -0,0 +1,29 @@
+<!doctype html>
+<title>Check that :marker inherits line-height</title>
+<link rel=help href="https://drafts.csswg.org/css-pseudo-4/#tree-abiding">
+<link rel=help href="https://drafts.csswg.org/css-pseudo-4/#marker-pseudo">
+<link rel=help href="https://drafts.csswg.org/css-display-3/#list-items">
+<link rel=help href="https://bugzilla.mozilla.org/show_bug.cgi?id=1415759">
+<link rel=match href=marker-inherit-line-height-ref.html>
+<style>
+div {
+  font: 128px/1 Ahem;
+  border: 2px solid blue;
+  background: orange;
+  color: black;
+}
+ol {
+  margin: 0;
+  padding: 0;
+  list-style-position: inside;
+  list-style-type: decimal;
+}
+</style>
+
+<p>There should be two black rectangles below with no space between them and
+  the border above/below.</p>
+<div>
+  <ol>
+    <li>X
+  </ol>
+</div>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-maxlength.html b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-maxlength.html
new file mode 100644
index 0000000..ff4e8f1
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-maxlength.html
@@ -0,0 +1,51 @@
+<!DOCTYPE html>
+<html>
+
+<head>
+  <title>textarea maxlength</title>
+  <link rel="author" title="tigercosmos" href="mailto:phy.tiger@gmail.com">
+  <link rel=help href="https://html.spec.whatwg.org/multipage/#attr-textarea-maxlength">
+  <script src="/resources/testharness.js"></script>
+  <script src="/resources/testharnessreport.js"></script>
+</head>
+
+<body>
+
+  <textarea id="none"></textarea>
+  <textarea id="negative" maxlength="-5"></textarea>
+  <textarea id="non-numeric" maxlength="not-a-number"></textarea>
+  <textarea id="assign-negative"></textarea>
+  <textarea id="assign-non-numeric"></textarea>
+
+  <script>
+    test(
+      function () {
+        assert_equals(document.getElementById("none").maxLength, -1);
+      }, "Unset maxlength is -1");
+
+    test(
+      function () {
+        assert_equals(document.getElementById("negative").maxLength, -1);
+      }, "Negative maxlength is always -1");
+
+    test(
+      function () {
+        assert_equals(document.getElementById("non-numeric").maxLength, -1);
+      }, "Non-numeric maxlength is -1");
+
+    test(
+      function () {
+        assert_throws("INDEX_SIZE_ERR", function () {
+          document.getElementById("assign-negative").maxLength = -5;
+        });
+      }, "Assigning negative integer throws IndexSizeError");
+
+    test(
+      function () {
+        document.getElementById("assign-non-numeric").maxLength = "not-a-number";
+        assert_equals(document.getElementById("assign-non-numeric").maxLength, 0);
+      }, "Assigning non-numeric to maxlength sets maxlength to 0");
+  </script>
+</body>
+
+</html>
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-minlength.html b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-minlength.html
new file mode 100644
index 0000000..9a15a12
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-minlength.html
@@ -0,0 +1,51 @@
+<!DOCTYPE html>
+<html>
+
+<head>
+  <title>textarea minlength</title>
+  <link rel="author" title="tigercosmos" href="mailto:phy.tiger@gmail.com">
+  <link rel=help href="https://html.spec.whatwg.org/multipage/#attr-textarea-minlength">
+  <script src="/resources/testharness.js"></script>
+  <script src="/resources/testharnessreport.js"></script>
+</head>
+
+<body>
+
+  <textarea id="none"></textarea>
+  <textarea id="negative" minlength=-5></textarea>
+  <textarea id="non-numeric" minlength="not-a-number"></textarea>
+  <textarea id="assign-negative"></textarea>
+  <textarea id="assign-non-numeric"></textarea>
+
+  <script>
+    test(
+      function () {
+        assert_equals(document.getElementById("none").minLength, -1);
+      }, "Unset minlength is -1");
+
+    test(
+      function () {
+        assert_equals(document.getElementById("negative").minLength, -1);
+      }, "Negative minlength is always -1");
+
+    test(
+      function () {
+        assert_equals(document.getElementById("non-numeric").minLength, -1);
+      }, "Non-numeric minlength is -1");
+
+    test(
+      function () {
+        assert_throws("INDEX_SIZE_ERR", function () {
+          document.getElementById("assign-negative").minLength = -5;
+        });
+      }, "Assigning negative integer throws IndexSizeError");
+
+    test(
+      function () {
+        document.getElementById("assign-non-numeric").minLength = "not-a-number";
+        assert_equals(document.getElementById("assign-non-numeric").minLength, 0);
+      }, "Assigning non-numeric to minlength sets minlength to 0");
+  </script>
+</body>
+
+</html>
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-textLength.html b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-textLength.html
new file mode 100644
index 0000000..d249278
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-textLength.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<title>The textLengh IDL attribute</title>
+<meta content="charset=utf-16">
+<link rel="author" title="tigercosmos" href="mailto:phy.tiger@gmail.com">
+<link rel="help" href="https://html.spec.whatwg.org/multipage/#dom-textarea-textlength">
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<textarea id="textarea"></textarea>
+<script>
+    var textarea = document.getElementById("textarea");
+
+    test(function () {
+        textarea.value= "Hello, World!";
+        assert_equals(textarea.textLength, 13);
+
+        textarea.value = "\u4f60\u597d\uff0c\u4e16\u754c\uff01"; //你好,世界!
+        assert_equals(textarea.textLength, 6);
+    }, "Textarea's 'testLength' should work for utf-16.");
+</script>
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-validity-clone.html b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-validity-clone.html
new file mode 100644
index 0000000..23d90e7
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/html/semantics/forms/the-textarea-element/textarea-validity-clone.html
@@ -0,0 +1,27 @@
+<!doctype html>
+<meta charset="utf-8">
+<title>HTML Test: &lt;textarea&gt; validity state is correct after a clone</title>
+<link rel="author" title="Emilio Cobos Álvarez" href="mailto:emilio@crisal.io">
+<link rel="help" href="https://html.spec.whatwg.org/multipage/#the-textarea-element">
+<link rel="help" href="https://bugzil.la/1472169">
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script>
+test(function() {
+  let form = document.createElement("form");
+  let textarea = document.createElement("textarea");
+  textarea.required = true;
+
+  textarea.appendChild(document.createTextNode("A"));
+  form.appendChild(textarea);
+
+  assert_true(textarea.validity.valid);
+
+  let formClone = form.cloneNode(true);
+  assert_equals(
+    formClone.querySelector('textarea').validity.valid,
+    textarea.validity.valid,
+    "Validity state should be preserved after a clone"
+  );
+}, "<textarea> validity state should be preserved after a clone");
+</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/infrastructure/server/wpt-server-http.sub-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/infrastructure/server/wpt-server-http.sub-expected.txt
deleted file mode 100644
index 776d0af5..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/infrastructure/server/wpt-server-http.sub-expected.txt
+++ /dev/null
@@ -1,25 +0,0 @@
-This is a testharness.js-based test.
-PASS HTTP protocol, no subdomain, port #1
-PASS HTTP protocol, no subdomain, port #2
-PASS HTTP protocol, www subdomain #1, port #1
-PASS HTTP protocol, www subdomain #1, port #2
-PASS HTTP protocol, www subdomain #2, port #1
-PASS HTTP protocol, www subdomain #2, port #2
-PASS HTTP protocol, www subdomain #3, port #1
-PASS HTTP protocol, www subdomain #3, port #2
-PASS HTTP protocol, punycode subdomain #1, port #1
-PASS HTTP protocol, punycode subdomain #1, port #2
-PASS HTTP protocol, punycode subdomain #2, port #1
-PASS HTTP protocol, punycode subdomain #2, port #2
-FAIL HTTP protocol, non-existent domain, port #1 assert_false: expected false got true
-FAIL HTTP protocol, non-existent domain, port #2 assert_false: expected false got true
-PASS HTTPS protocol, no subdomain
-PASS HTTPS protocol, www subdomain #1
-PASS HTTPS protocol, www subdomain #2
-PASS HTTPS protocol, www subdomain #3
-PASS HTTPS protocol, punycode subdomain #1
-PASS HTTPS protocol, punycode subdomain #2
-PASS HTTPS protocol, non-existent domain, port #1
-PASS HTTPS protocol, non-existent domain, port #2
-Harness: the test ran to completion.
-
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/background-fetch.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/background-fetch.idl
index 68509fc1a..fc3ed8d 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/interfaces/background-fetch.idl
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/background-fetch.idl
@@ -18,7 +18,7 @@
 interface BackgroundFetchManager {
   Promise<BackgroundFetchRegistration> fetch(DOMString id, (RequestInfo or sequence<RequestInfo>) requests, optional BackgroundFetchOptions options);
   Promise<BackgroundFetchRegistration?> get(DOMString id);
-  Promise<FrozenArray<DOMString>> getIds();
+  Promise<sequence<DOMString>> getIds();
   // TODO: in future this should become an async iterator for BackgroundFetchRegistration objects
 };
 
@@ -51,14 +51,10 @@
 };
 
 [Exposed=(Window,Worker)]
-interface BackgroundFetchFetch {
-  readonly attribute Request request;
-};
-
-[Exposed=(Window,Worker)]
 interface BackgroundFetchActiveFetches {
-  Promise<BackgroundFetchActiveFetch> match(RequestInfo request);
-  Promise<FrozenArray<BackgroundFetchActiveFetch>> values();
+  Promise<BackgroundFetchActiveFetch> match(RequestInfo request, optional CacheQueryOptions options);
+  Promise<sequence<BackgroundFetchActiveFetch>> matchAll(RequestInfo request, optional CacheQueryOptions options);
+  Promise<sequence<BackgroundFetchActiveFetch>> values();
 };
 
 [Exposed=(Window,Worker)]
@@ -67,6 +63,11 @@
   // In future this will include a fetch observer
 };
 
+[Exposed=(Window,Worker)]
+interface BackgroundFetchFetch {
+  readonly attribute Request request;
+};
+
 [Constructor(DOMString type, BackgroundFetchEventInit init), Exposed=ServiceWorker]
 interface BackgroundFetchEvent : ExtendableEvent {
   readonly attribute DOMString id;
@@ -87,8 +88,9 @@
 
 [Exposed=ServiceWorker]
 interface BackgroundFetchSettledFetches {
-  Promise<BackgroundFetchSettledFetch> match(RequestInfo request);
-  Promise<FrozenArray<BackgroundFetchSettledFetch>> values();
+  Promise<BackgroundFetchSettledFetch> match(RequestInfo request, optional CacheQueryOptions options);
+  Promise<sequence<BackgroundFetchSettledFetch>> matchAll(RequestInfo request, optional CacheQueryOptions options);
+  Promise<sequence<BackgroundFetchSettledFetch>> values();
 };
 
 [Exposed=ServiceWorker]
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-counter-styles.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-counter-styles.idl
new file mode 100644
index 0000000..b1c3663
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-counter-styles.idl
@@ -0,0 +1,23 @@
+// GENERATED CONTENT - DO NOT EDIT
+// Content of this file was automatically extracted from the
+// "CSS Counter Styles Level 3" spec.
+// See: https://drafts.csswg.org/css-counter-styles/
+
+partial interface CSSRule {
+    const unsigned short COUNTER_STYLE_RULE = 11;
+};
+
+[Exposed=Window]
+interface CSSCounterStyleRule : CSSRule {
+  attribute CSSOMString name;
+  attribute CSSOMString system;
+  attribute CSSOMString symbols;
+  attribute CSSOMString additiveSymbols;
+  attribute CSSOMString negative;
+  attribute CSSOMString prefix;
+  attribute CSSOMString suffix;
+  attribute CSSOMString range;
+  attribute CSSOMString pad;
+  attribute CSSOMString speakAs;
+  attribute CSSOMString fallback;
+};
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-images.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-images.idl
new file mode 100644
index 0000000..0dbac1c
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-images.idl
@@ -0,0 +1,8 @@
+// GENERATED CONTENT - DO NOT EDIT
+// Content of this file was automatically extracted from the
+// "CSS Images Module Level 4" spec.
+// See: https://drafts.csswg.org/css-images-4/
+
+partial namespace CSS {
+  // [SameObject] readonly attribute Map elementSources;
+};
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-regions.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-regions.idl
new file mode 100644
index 0000000..546c21ad
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/css-regions.idl
@@ -0,0 +1,35 @@
+// GENERATED CONTENT - DO NOT EDIT
+// Content of this file was automatically extracted from the
+// "CSS Regions Module Level 1" spec.
+// See: https://drafts.csswg.org/css-regions/
+
+partial interface Document {
+  readonly attribute NamedFlowMap namedFlows;
+};
+
+[Exposed=Window,
+ MapClass=(CSSOMString, NamedFlow)] interface NamedFlowMap {
+  NamedFlow? get(CSSOMString flowName);
+  boolean has(CSSOMString flowName);
+  NamedFlowMap set(CSSOMString flowName, NamedFlow flowValue);
+  boolean delete(CSSOMString flowName);
+};
+
+[Exposed=Window]
+interface NamedFlow : EventTarget {
+  readonly attribute CSSOMString name;
+  readonly attribute boolean overset;
+  sequence<Region> getRegions();
+  readonly attribute short firstEmptyRegionIndex;
+  sequence<Node> getContent();
+  sequence<Region> getRegionsByContent(Node node);
+};
+
+[Exposed=Window,
+ NoInterfaceObject]
+interface Region {
+  readonly attribute CSSOMString regionOverset;
+  sequence<Range>? getRegionFlowRanges();
+};
+
+Element implements Region;
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/media-source.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/media-source.idl
new file mode 100644
index 0000000..6476794
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/media-source.idl
@@ -0,0 +1,72 @@
+// GENERATED CONTENT - DO NOT EDIT
+// Content of this file was automatically extracted from the
+// "Media Source Extensions™" spec.
+// See: https://w3c.github.io/media-source/
+
+enum ReadyState {
+    "closed",
+    "open",
+    "ended"
+};
+enum EndOfStreamError {
+    "network",
+    "decode"
+};
+[Constructor]
+interface MediaSource : EventTarget {
+    readonly attribute SourceBufferList    sourceBuffers;
+    readonly attribute SourceBufferList    activeSourceBuffers;
+    readonly attribute ReadyState          readyState;
+             attribute unrestricted double duration;
+             attribute EventHandler        onsourceopen;
+             attribute EventHandler        onsourceended;
+             attribute EventHandler        onsourceclose;
+    SourceBuffer addSourceBuffer(DOMString type);
+    void         removeSourceBuffer(SourceBuffer sourceBuffer);
+    void         endOfStream(optional EndOfStreamError error);
+    void         setLiveSeekableRange(double start, double end);
+    void         clearLiveSeekableRange();
+    static boolean isTypeSupported(DOMString type);
+};
+enum AppendMode {
+    "segments",
+    "sequence"
+};
+interface SourceBuffer : EventTarget {
+             attribute AppendMode          mode;
+    readonly attribute boolean             updating;
+    readonly attribute TimeRanges          buffered;
+             attribute double              timestampOffset;
+    readonly attribute AudioTrackList      audioTracks;
+    readonly attribute VideoTrackList      videoTracks;
+    readonly attribute TextTrackList       textTracks;
+             attribute double              appendWindowStart;
+             attribute unrestricted double appendWindowEnd;
+             attribute EventHandler        onupdatestart;
+             attribute EventHandler        onupdate;
+             attribute EventHandler        onupdateend;
+             attribute EventHandler        onerror;
+             attribute EventHandler        onabort;
+    void appendBuffer(BufferSource data);
+    void abort();
+    void remove(double start, unrestricted double end);
+};
+interface SourceBufferList : EventTarget {
+    readonly attribute unsigned long length;
+             attribute EventHandler  onaddsourcebuffer;
+             attribute EventHandler  onremovesourcebuffer;
+    getter SourceBuffer (unsigned long index);
+};
+[Exposed=Window]
+partial interface URL {
+    static DOMString createObjectURL(MediaSource mediaSource);
+};
+partial interface AudioTrack {
+    readonly attribute SourceBuffer? sourceBuffer;
+};
+partial interface VideoTrack {
+    readonly attribute SourceBuffer? sourceBuffer;
+};
+partial interface TextTrack {
+    readonly attribute SourceBuffer? sourceBuffer;
+};
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/resource-timing.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/resource-timing.idl
index 3f041e1..2668f91 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/interfaces/resource-timing.idl
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/resource-timing.idl
@@ -5,27 +5,28 @@
 
 [Exposed=(Window,Worker)]
 interface PerformanceResourceTiming : PerformanceEntry {
-    readonly attribute DOMString           initiatorType;
-    readonly attribute DOMString           nextHopProtocol;
-    readonly attribute DOMHighResTimeStamp workerStart;
-    readonly attribute DOMHighResTimeStamp redirectStart;
-    readonly attribute DOMHighResTimeStamp redirectEnd;
-    readonly attribute DOMHighResTimeStamp fetchStart;
-    readonly attribute DOMHighResTimeStamp domainLookupStart;
-    readonly attribute DOMHighResTimeStamp domainLookupEnd;
-    readonly attribute DOMHighResTimeStamp connectStart;
-    readonly attribute DOMHighResTimeStamp connectEnd;
-    readonly attribute DOMHighResTimeStamp secureConnectionStart;
-    readonly attribute DOMHighResTimeStamp requestStart;
-    readonly attribute DOMHighResTimeStamp responseStart;
-    readonly attribute DOMHighResTimeStamp responseEnd;
-    readonly attribute unsigned long long  transferSize;
-    readonly attribute unsigned long long  encodedBodySize;
-    readonly attribute unsigned long long  decodedBodySize;
+    readonly        attribute DOMString initiatorType;
+    readonly        attribute DOMString nextHopProtocol;
+    readonly        attribute DOMHighResTimeStamp workerStart;
+    readonly        attribute DOMHighResTimeStamp redirectStart;
+    readonly        attribute DOMHighResTimeStamp redirectEnd;
+    readonly        attribute DOMHighResTimeStamp fetchStart;
+    readonly        attribute DOMHighResTimeStamp domainLookupStart;
+    readonly        attribute DOMHighResTimeStamp domainLookupEnd;
+    readonly        attribute DOMHighResTimeStamp connectStart;
+    readonly        attribute DOMHighResTimeStamp connectEnd;
+    readonly        attribute DOMHighResTimeStamp secureConnectionStart;
+    readonly        attribute DOMHighResTimeStamp requestStart;
+    readonly        attribute DOMHighResTimeStamp responseStart;
+    readonly        attribute DOMHighResTimeStamp responseEnd;
+    readonly        attribute unsigned long long transferSize;
+    readonly        attribute unsigned long long encodedBodySize;
+    readonly        attribute unsigned long long decodedBodySize;
     [Default] object toJSON();
 };
+
 partial interface Performance {
-    void clearResourceTimings();
-    void setResourceTimingBufferSize(unsigned long maxSize);
-    attribute EventHandler onresourcetimingbufferfull;
+  void clearResourceTimings();
+  void setResourceTimingBufferSize(unsigned long maxSize);
+              attribute EventHandler onresourcetimingbufferfull;
 };
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/selection-api.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/selection-api.idl
index b2d2f86..6e814ea 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/interfaces/selection-api.idl
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/selection-api.idl
@@ -4,42 +4,40 @@
 // See: https://w3c.github.io/selection-api/
 
 interface Selection {
-    readonly attribute Node?         anchorNode;
-    readonly attribute unsigned long anchorOffset;
-    readonly attribute Node?         focusNode;
-    readonly attribute unsigned long focusOffset;
-    readonly attribute boolean       isCollapsed;
-    readonly attribute unsigned long rangeCount;
-    readonly attribute DOMString     type;
-    Range                 getRangeAt(unsigned long index);
-    void                  addRange(Range range);
-    void                  removeRange(Range range);
-    void                  removeAllRanges();
-    void                  empty();
-    void                  collapse(Node? node,
-                                   optional unsigned long offset = 0);
-    void                  setPosition(Node? node,
-                                      optional unsigned long offset = 0);
-    void                  collapseToStart();
-    void                  collapseToEnd();
-    void                  extend(Node node, optional unsigned long offset = 0);
-    void                  setBaseAndExtent(Node anchorNode,
-                                           unsigned long anchorOffset,
-                                           Node focusNode,
-                                           unsigned long focusOffset);
-    void                  selectAllChildren(Node node);
-    [CEReactions] void                  deleteFromDocument();
-    boolean               containsNode(Node node,
-                                       optional boolean allowPartialContainment = false);
-    stringifier DOMString ();
+  readonly attribute Node? anchorNode;
+  readonly attribute unsigned long anchorOffset;
+  readonly attribute Node? focusNode;
+  readonly attribute unsigned long focusOffset;
+  readonly attribute boolean isCollapsed;
+  readonly attribute unsigned long rangeCount;
+  readonly attribute DOMString type;
+  Range getRangeAt(unsigned long index);
+  void addRange(Range range);
+  void removeRange(Range range);
+  void removeAllRanges();
+  void empty();
+  void collapse(Node? node, optional unsigned long offset = 0);
+  void setPosition(Node? node, optional unsigned long offset = 0);
+  void collapseToStart();
+  void collapseToEnd();
+  void extend(Node node, optional unsigned long offset = 0);
+  void setBaseAndExtent(Node anchorNode, unsigned long anchorOffset, Node focusNode, unsigned long focusOffset);
+  void selectAllChildren(Node node);
+  [CEReactions]
+  void deleteFromDocument();
+  boolean containsNode(Node node, optional boolean allowPartialContainment = false);
+  stringifier DOMString ();
 };
+
 partial interface Document {
-    Selection? getSelection();
+  Selection? getSelection();
 };
+
 partial interface Window {
-    Selection? getSelection();
+  Selection? getSelection();
 };
+
 partial interface GlobalEventHandlers {
-    attribute EventHandler onselectstart;
-    attribute EventHandler onselectionchange;
+  attribute EventHandler onselectstart;
+  attribute EventHandler onselectionchange;
 };
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/service-workers.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/service-workers.idl
index edf2985bd..38d22c6 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/interfaces/service-workers.idl
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/service-workers.idl
@@ -12,7 +12,7 @@
   // event
   attribute EventHandler onstatechange;
 };
-ServiceWorker implements AbstractWorker;
+ServiceWorker includes AbstractWorker;
 
 enum ServiceWorkerState {
   "installing",
diff --git a/third_party/WebKit/LayoutTests/external/wpt/interfaces/touch-events.idl b/third_party/WebKit/LayoutTests/external/wpt/interfaces/touch-events.idl
index 5c5a3ef..097b0e9 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/interfaces/touch-events.idl
+++ b/third_party/WebKit/LayoutTests/external/wpt/interfaces/touch-events.idl
@@ -9,78 +9,74 @@
 };
 
 dictionary TouchInit {
-    required long        identifier;
+    required long identifier;
     required EventTarget target;
-             double      clientX = 0;
-             double      clientY = 0;
-             double      screenX = 0;
-             double      screenY = 0;
-             double      pageX = 0;
-             double      pageY = 0;
-             float       radiusX = 0;
-             float       radiusY = 0;
-             float       rotationAngle = 0;
-             float       force = 0;
-             double      altitudeAngle = 0;
-             double      azimuthAngle = 0;
-             TouchType   touchType = "direct";
+             double clientX = 0;
+             double clientY = 0;
+             double screenX = 0;
+             double screenY = 0;
+             double pageX = 0;
+             double pageY = 0;
+             float radiusX = 0;
+             float radiusY = 0;
+             float rotationAngle = 0;
+             float force = 0;
+             double altitudeAngle = 0;
+             double azimuthAngle = 0;
+             TouchType touchType = "direct";
 };
 
-[Constructor(TouchInit touchInitDict),
- Exposed=Window]
+[Constructor(TouchInit touchInitDict), Exposed=Window]
 interface Touch {
-    readonly attribute long        identifier;
-    readonly attribute EventTarget target;
-    readonly attribute double      screenX;
-    readonly attribute double      screenY;
-    readonly attribute double      clientX;
-    readonly attribute double      clientY;
-    readonly attribute double      pageX;
-    readonly attribute double      pageY;
-    readonly attribute float       radiusX;
-    readonly attribute float       radiusY;
-    readonly attribute float       rotationAngle;
-    readonly attribute float       force;
-    readonly attribute float       altitudeAngle;
-    readonly attribute float       azimuthAngle;
-    readonly attribute TouchType   touchType;
+    readonly        attribute long identifier;
+    readonly        attribute EventTarget target;
+    readonly        attribute double screenX;
+    readonly        attribute double screenY;
+    readonly        attribute double clientX;
+    readonly        attribute double clientY;
+    readonly        attribute double pageX;
+    readonly        attribute double pageY;
+    readonly        attribute float radiusX;
+    readonly        attribute float radiusY;
+    readonly        attribute float rotationAngle;
+    readonly        attribute float force;
+    readonly        attribute float altitudeAngle;
+    readonly        attribute float azimuthAngle;
+    readonly        attribute TouchType touchType;
 };
+
 interface TouchList {
-    readonly attribute unsigned long length;
+    readonly        attribute unsigned long length;
     getter Touch? item(unsigned long index);
 };
+
 dictionary TouchEventInit : EventModifierInit {
-    sequence<Touch> touches = [];
-    sequence<Touch> targetTouches = [];
-    sequence<Touch> changedTouches = [];
+             sequence<Touch> touches = [];
+             sequence<Touch> targetTouches = [];
+             sequence<Touch> changedTouches = [];
 };
 
-[Constructor(DOMString type, optional TouchEventInit eventInitDict),
- Exposed=Window]
+[Constructor(DOMString type, optional TouchEventInit eventInitDict), Exposed=Window]
 interface TouchEvent : UIEvent {
-    readonly attribute TouchList touches;
-    readonly attribute TouchList targetTouches;
-    readonly attribute TouchList changedTouches;
-    readonly attribute boolean   altKey;
-    readonly attribute boolean   metaKey;
-    readonly attribute boolean   ctrlKey;
-    readonly attribute boolean   shiftKey;
+    readonly        attribute TouchList touches;
+    readonly        attribute TouchList targetTouches;
+    readonly        attribute TouchList changedTouches;
+    readonly        attribute boolean altKey;
+    readonly        attribute boolean metaKey;
+    readonly        attribute boolean ctrlKey;
+    readonly        attribute boolean shiftKey;
 };
+
 partial interface GlobalEventHandlers {
-    attribute EventHandler ontouchstart;
-    attribute EventHandler ontouchend;
-    attribute EventHandler ontouchmove;
-    attribute EventHandler ontouchcancel;
+                    attribute EventHandler ontouchstart;
+                    attribute EventHandler ontouchend;
+                    attribute EventHandler ontouchmove;
+                    attribute EventHandler ontouchcancel;
 };
+
 partial interface Document {
-    // Deprecated in this specification
-    Touch     createTouch(WindowProxy view,
-                          EventTarget target,
-                          long identifier,
-                          double pageX,
-                          double pageY,
-                          double screenX,
-                          double screenY);
-    // Deprecated in this specification
-    TouchList createTouchList(Touch... touches);
+  // Deprecated in this specification
+  Touch createTouch(WindowProxy view, EventTarget target, long identifier, double pageX, double pageY, double screenX, double screenY);
+  // Deprecated in this specification
+  TouchList createTouchList(Touch... touches);
 };
diff --git a/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any-expected.txt
new file mode 100644
index 0000000..4d85278
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any-expected.txt
@@ -0,0 +1,148 @@
+This is a testharness.js-based test.
+Found 144 tests; 141 PASS, 3 FAIL, 0 TIMEOUT, 0 NOTRUN.
+PASS media-source interfaces
+PASS Partial interface URL: original interface defined
+PASS Partial interface URL: valid exposure set
+PASS Partial interface AudioTrack: original interface defined
+PASS Partial interface VideoTrack: original interface defined
+PASS Partial interface TextTrack: original interface defined
+PASS MediaSource interface: existence and properties of interface object
+PASS MediaSource interface object length
+PASS MediaSource interface object name
+PASS MediaSource interface: existence and properties of interface prototype object
+PASS MediaSource interface: existence and properties of interface prototype object's "constructor" property
+PASS MediaSource interface: existence and properties of interface prototype object's @@unscopables property
+PASS MediaSource interface: attribute sourceBuffers
+PASS Unscopable handled correctly for sourceBuffers property on MediaSource
+PASS MediaSource interface: attribute activeSourceBuffers
+PASS Unscopable handled correctly for activeSourceBuffers property on MediaSource
+PASS MediaSource interface: attribute readyState
+PASS Unscopable handled correctly for readyState property on MediaSource
+PASS MediaSource interface: attribute duration
+PASS Unscopable handled correctly for duration property on MediaSource
+PASS MediaSource interface: attribute onsourceopen
+PASS Unscopable handled correctly for onsourceopen property on MediaSource
+PASS MediaSource interface: attribute onsourceended
+PASS Unscopable handled correctly for onsourceended property on MediaSource
+PASS MediaSource interface: attribute onsourceclose
+PASS Unscopable handled correctly for onsourceclose property on MediaSource
+PASS MediaSource interface: operation addSourceBuffer(DOMString)
+PASS Unscopable handled correctly for addSourceBuffer(DOMString) on MediaSource
+PASS MediaSource interface: operation removeSourceBuffer(SourceBuffer)
+PASS Unscopable handled correctly for removeSourceBuffer(SourceBuffer) on MediaSource
+PASS MediaSource interface: operation endOfStream(EndOfStreamError)
+PASS Unscopable handled correctly for endOfStream(EndOfStreamError) on MediaSource
+PASS MediaSource interface: operation setLiveSeekableRange(double, double)
+PASS Unscopable handled correctly for setLiveSeekableRange(double, double) on MediaSource
+PASS MediaSource interface: operation clearLiveSeekableRange()
+PASS Unscopable handled correctly for clearLiveSeekableRange() on MediaSource
+PASS MediaSource interface: operation isTypeSupported(DOMString)
+PASS Unscopable handled correctly for isTypeSupported(DOMString) on MediaSource
+PASS MediaSource must be primary interface of mediaSource
+PASS Stringification of mediaSource
+PASS MediaSource interface: mediaSource must inherit property "sourceBuffers" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "activeSourceBuffers" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "readyState" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "duration" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "onsourceopen" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "onsourceended" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "onsourceclose" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "addSourceBuffer(DOMString)" with the proper type
+PASS MediaSource interface: calling addSourceBuffer(DOMString) on mediaSource with too few arguments must throw TypeError
+PASS MediaSource interface: mediaSource must inherit property "removeSourceBuffer(SourceBuffer)" with the proper type
+PASS MediaSource interface: calling removeSourceBuffer(SourceBuffer) on mediaSource with too few arguments must throw TypeError
+PASS MediaSource interface: mediaSource must inherit property "endOfStream(EndOfStreamError)" with the proper type
+PASS MediaSource interface: calling endOfStream(EndOfStreamError) on mediaSource with too few arguments must throw TypeError
+PASS MediaSource interface: mediaSource must inherit property "setLiveSeekableRange(double, double)" with the proper type
+PASS MediaSource interface: calling setLiveSeekableRange(double, double) on mediaSource with too few arguments must throw TypeError
+PASS MediaSource interface: mediaSource must inherit property "clearLiveSeekableRange()" with the proper type
+PASS MediaSource interface: mediaSource must inherit property "isTypeSupported(DOMString)" with the proper type
+PASS MediaSource interface: calling isTypeSupported(DOMString) on mediaSource with too few arguments must throw TypeError
+PASS SourceBuffer interface: existence and properties of interface object
+PASS SourceBuffer interface object length
+PASS SourceBuffer interface object name
+PASS SourceBuffer interface: existence and properties of interface prototype object
+PASS SourceBuffer interface: existence and properties of interface prototype object's "constructor" property
+PASS SourceBuffer interface: existence and properties of interface prototype object's @@unscopables property
+PASS SourceBuffer interface: attribute mode
+PASS Unscopable handled correctly for mode property on SourceBuffer
+PASS SourceBuffer interface: attribute updating
+PASS Unscopable handled correctly for updating property on SourceBuffer
+PASS SourceBuffer interface: attribute buffered
+PASS Unscopable handled correctly for buffered property on SourceBuffer
+PASS SourceBuffer interface: attribute timestampOffset
+PASS Unscopable handled correctly for timestampOffset property on SourceBuffer
+PASS SourceBuffer interface: attribute audioTracks
+PASS Unscopable handled correctly for audioTracks property on SourceBuffer
+PASS SourceBuffer interface: attribute videoTracks
+PASS Unscopable handled correctly for videoTracks property on SourceBuffer
+FAIL SourceBuffer interface: attribute textTracks assert_true: The prototype object must have a property "textTracks" expected true got false
+PASS Unscopable handled correctly for textTracks property on SourceBuffer
+PASS SourceBuffer interface: attribute appendWindowStart
+PASS Unscopable handled correctly for appendWindowStart property on SourceBuffer
+PASS SourceBuffer interface: attribute appendWindowEnd
+PASS Unscopable handled correctly for appendWindowEnd property on SourceBuffer
+PASS SourceBuffer interface: attribute onupdatestart
+PASS Unscopable handled correctly for onupdatestart property on SourceBuffer
+PASS SourceBuffer interface: attribute onupdate
+PASS Unscopable handled correctly for onupdate property on SourceBuffer
+PASS SourceBuffer interface: attribute onupdateend
+PASS Unscopable handled correctly for onupdateend property on SourceBuffer
+PASS SourceBuffer interface: attribute onerror
+PASS Unscopable handled correctly for onerror property on SourceBuffer
+PASS SourceBuffer interface: attribute onabort
+PASS Unscopable handled correctly for onabort property on SourceBuffer
+PASS SourceBuffer interface: operation appendBuffer(BufferSource)
+PASS Unscopable handled correctly for appendBuffer(BufferSource) on SourceBuffer
+PASS SourceBuffer interface: operation abort()
+PASS Unscopable handled correctly for abort() on SourceBuffer
+PASS SourceBuffer interface: operation remove(double, unrestricted double)
+PASS Unscopable handled correctly for remove(double, unrestricted double) on SourceBuffer
+PASS SourceBuffer must be primary interface of sourceBuffer
+PASS Stringification of sourceBuffer
+PASS SourceBuffer interface: sourceBuffer must inherit property "mode" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "updating" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "buffered" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "timestampOffset" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "audioTracks" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "videoTracks" with the proper type
+FAIL SourceBuffer interface: sourceBuffer must inherit property "textTracks" with the proper type assert_inherits: property "textTracks" not found in prototype chain
+PASS SourceBuffer interface: sourceBuffer must inherit property "appendWindowStart" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "appendWindowEnd" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "onupdatestart" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "onupdate" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "onupdateend" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "onerror" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "onabort" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "appendBuffer(BufferSource)" with the proper type
+PASS SourceBuffer interface: calling appendBuffer(BufferSource) on sourceBuffer with too few arguments must throw TypeError
+PASS SourceBuffer interface: sourceBuffer must inherit property "abort()" with the proper type
+PASS SourceBuffer interface: sourceBuffer must inherit property "remove(double, unrestricted double)" with the proper type
+PASS SourceBuffer interface: calling remove(double, unrestricted double) on sourceBuffer with too few arguments must throw TypeError
+PASS SourceBufferList interface: existence and properties of interface object
+PASS SourceBufferList interface object length
+PASS SourceBufferList interface object name
+PASS SourceBufferList interface: existence and properties of interface prototype object
+PASS SourceBufferList interface: existence and properties of interface prototype object's "constructor" property
+PASS SourceBufferList interface: existence and properties of interface prototype object's @@unscopables property
+PASS SourceBufferList interface: attribute length
+PASS Unscopable handled correctly for length property on SourceBufferList
+PASS SourceBufferList interface: attribute onaddsourcebuffer
+PASS Unscopable handled correctly for onaddsourcebuffer property on SourceBufferList
+PASS SourceBufferList interface: attribute onremovesourcebuffer
+PASS Unscopable handled correctly for onremovesourcebuffer property on SourceBufferList
+PASS SourceBufferList must be primary interface of mediaSource.sourceBuffers
+PASS Stringification of mediaSource.sourceBuffers
+PASS SourceBufferList interface: mediaSource.sourceBuffers must inherit property "length" with the proper type
+PASS SourceBufferList interface: mediaSource.sourceBuffers must inherit property "onaddsourcebuffer" with the proper type
+PASS SourceBufferList interface: mediaSource.sourceBuffers must inherit property "onremovesourcebuffer" with the proper type
+PASS AudioTrack interface: attribute sourceBuffer
+PASS Unscopable handled correctly for sourceBuffer property on AudioTrack
+PASS VideoTrack interface: attribute sourceBuffer
+PASS Unscopable handled correctly for sourceBuffer property on VideoTrack
+FAIL TextTrack interface: attribute sourceBuffer assert_true: The prototype object must have a property "sourceBuffer" expected true got false
+PASS Unscopable handled correctly for sourceBuffer property on TextTrack
+PASS URL interface: operation createObjectURL(MediaSource)
+PASS Unscopable handled correctly for createObjectURL(MediaSource) on URL
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.js b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.js
new file mode 100644
index 0000000..7992b11
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.js
@@ -0,0 +1,59 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+// https://w3c.github.io/media-source/
+
+'use strict';
+
+var mediaSource;
+var sourceBuffer;
+var video = document.createElement("video");
+
+promise_test(async t => {
+  const srcs = ['media-source', 'dom', 'html', 'url'];
+  const [idl, dom, html, url] = await Promise.all(
+      srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
+
+  var idlArray = new IdlArray();
+  idlArray.add_idls(idl);
+  idlArray.add_dependency_idls(dom);
+  idlArray.add_dependency_idls(html);
+  idlArray.add_dependency_idls(url);
+
+  const testIdls = new Promise(resolve => {
+    try {
+      mediaSource = new MediaSource();
+      video.src = URL.createObjectURL(mediaSource);
+      mediaSource.addEventListener("sourceopen", function () {
+        var defaultType ='video/webm;codecs="vp8,vorbis"';
+        if (MediaSource.isTypeSupported(defaultType)) {
+          sourceBuffer = mediaSource.addSourceBuffer(defaultType);
+        } else {
+          sourceBuffer = mediaSource.addSourceBuffer('video/mp4');
+        }
+        sourceBuffer.addEventListener("updateend", function (e) {
+          mediaSource.endOfStream();
+          resolve();
+        });
+        sourceBuffer.appendBuffer(new ArrayBuffer());
+      });
+    } catch (e) {
+      // Will be surfaced in idlharness.js's test_object below.
+    }
+  })
+
+  idlArray.add_objects({
+    MediaSource: ['mediaSource'],
+    SourceBuffer: ['sourceBuffer'],
+    SourceBufferList: ['mediaSource.sourceBuffers']
+  });
+
+  const timeout = new Promise((_, reject) => t.step_timeout(reject, 3000));
+  return Promise
+      .race([testIdls, timeout])
+      .then(() => { idlArray.test(); })
+      .catch(() => {
+        idlArray.test();
+        return Promise.reject('Failed to create media-source objects')
+      });
+}, 'media-source interfaces');
diff --git a/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.worker-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.worker-expected.txt
new file mode 100644
index 0000000..b86bc74
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/media-source/idlharness.any.worker-expected.txt
@@ -0,0 +1,4 @@
+This is a testharness.js-based test.
+FAIL idlharness Uncaught ReferenceError: document is not defined
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/media-source/interfaces.html b/third_party/WebKit/LayoutTests/external/wpt/media-source/interfaces.html
deleted file mode 100644
index 13e3148..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/media-source/interfaces.html
+++ /dev/null
@@ -1,144 +0,0 @@
-<!doctype html>
-<meta charset=utf-8>
-<title>Media Source Extensions IDL tests</title>
-<div id=log></div>
-<script src=/resources/testharness.js></script>
-<script src=/resources/testharnessreport.js></script>
-<script src=/resources/WebIDLParser.js></script>
-<script src=/resources/idlharness.js></script>
-<script type=text/plain class=untested>
-interface EventTarget {
-  void addEventListener(DOMString type, EventListener? callback, optional boolean capture /* = false */);
-  void removeEventListener(DOMString type, EventListener? callback, optional boolean capture /* = false */);
-  boolean dispatchEvent(Event event);
-};
-interface EventHandler {};
-interface URL {};
-interface HTMLVideoElement {};
-interface AudioTrack {};
-interface AudioTrackList {};
-interface VideoTrack {};
-interface VideoTrackList {};
-interface TextTrack {};
-interface TextTrackList {};
-interface TimeRanges {};
-typedef double DOMHighResTimeStamp;
-</script>
-<script type=text/plain>
-[Constructor]
-interface MediaSource : EventTarget {
-    readonly attribute SourceBufferList    sourceBuffers;
-    readonly attribute SourceBufferList    activeSourceBuffers;
-    readonly attribute ReadyState          readyState;
-             attribute unrestricted double duration;
-             attribute EventHandler        onsourceopen;
-             attribute EventHandler        onsourceended;
-             attribute EventHandler        onsourceclose;
-    SourceBuffer addSourceBuffer(DOMString type);
-    void         removeSourceBuffer(SourceBuffer sourceBuffer);
-    void         endOfStream(optional EndOfStreamError error);
-    void         setLiveSeekableRange(double start, double end);
-    void         clearLiveSeekableRange();
-    static boolean isTypeSupported(DOMString type);
-};
-
-interface SourceBuffer : EventTarget {
-             attribute AppendMode          mode;
-    readonly attribute boolean             updating;
-    readonly attribute TimeRanges          buffered;
-             attribute double              timestampOffset;
-    readonly attribute AudioTrackList      audioTracks;
-    readonly attribute VideoTrackList      videoTracks;
-    readonly attribute TextTrackList       textTracks;
-             attribute double              appendWindowStart;
-             attribute unrestricted double appendWindowEnd;
-             attribute EventHandler        onupdatestart;
-             attribute EventHandler        onupdate;
-             attribute EventHandler        onupdateend;
-             attribute EventHandler        onerror;
-             attribute EventHandler        onabort;
-    void appendBuffer(BufferSource data);
-    void abort();
-    void changeType(DOMString type);
-    void remove(double start, unrestricted double end);
-};
-
-interface SourceBufferList : EventTarget {
-    readonly attribute unsigned long length;
-             attribute EventHandler  onaddsourcebuffer;
-             attribute EventHandler  onremovesourcebuffer;
-    getter SourceBuffer (unsigned long index);
-};
-
-[Exposed=Window,DedicatedWorker,SharedWorker]
-partial interface URL {
-    static DOMString createObjectURL(MediaSource mediaSource);
-};
-
-partial interface AudioTrack {
-    readonly attribute SourceBuffer? sourceBuffer;
-};
-
-partial interface VideoTrack {
-    readonly attribute SourceBuffer? sourceBuffer;
-};
-
-partial interface TextTrack {
-    readonly attribute SourceBuffer? sourceBuffer;
-};
-
-enum EndOfStreamError {
-    "network",
-    "decode"
-};
-
-enum AppendMode {
-    "segments",
-    "sequence"
-};
-
-enum ReadyState {
-    "closed",
-    "open",
-    "ended"
-};
-</script>
-<script>
-"use strict";
-setup({ explicit_done: true });
-var mediaSource;
-var sourceBuffer;
-var video = document.createElement("video");
-var idlCheck = function() {
-  var idlArray = new IdlArray();
-  [].forEach.call(document.querySelectorAll("script[type=text\\/plain]"), function(node) {
-    if (node.className == "untested") {
-      idlArray.add_untested_idls(node.textContent);
-    } else {
-      idlArray.add_idls(node.textContent);
-    }
-  });
-  idlArray.add_objects({
-    MediaSource: ['mediaSource'],
-    SourceBuffer: ['sourceBuffer'],
-    SourceBufferList: ['mediaSource.sourceBuffers']
-  });
-  idlArray.test();
-  done();
-}
-mediaSource = new MediaSource();
-video.src = URL.createObjectURL(mediaSource);
-mediaSource.addEventListener("sourceopen", function () {
-  var defaultType ='video/webm;codecs="vp8,vorbis"';
-  if (MediaSource.isTypeSupported(defaultType)) {
-    sourceBuffer = mediaSource.addSourceBuffer(defaultType);
-  } else {
-    sourceBuffer = mediaSource.addSourceBuffer('video/mp4');
-  }
-  sourceBuffer.addEventListener("updateend", function (e) {
-    mediaSource.endOfStream();
-    idlCheck();
-  });
-  sourceBuffer.appendBuffer(new ArrayBuffer());
-});
-</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/payment-method-basic-card/idlharness.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/payment-method-basic-card/idlharness.window-expected.txt
deleted file mode 100644
index d29913a..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/payment-method-basic-card/idlharness.window-expected.txt
+++ /dev/null
@@ -1,4 +0,0 @@
-This is a testharness.js-based test.
-FAIL payment-method-basic-card interfaces. promise_test: Unhandled rejection with value: object "TypeError: idl_setup_func is not a function"
-Harness: the test ran to completion.
-
diff --git a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/disable-picture-in-picture.html b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/disable-picture-in-picture.html
index 1a59b7a3..fdd8e97 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/disable-picture-in-picture.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/disable-picture-in-picture.html
@@ -33,7 +33,7 @@
 
 promise_test(async t => {
   const video = await loadVideo();
-  return callWithTrustedClick(async () => {
+  return test_driver.bless('request picture in picture', async () => {
     const promise = video.requestPictureInPicture();
     video.disablePictureInPicture = true;
     await promise_rejects(t, 'InvalidStateError', promise);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window-expected.txt
index 5f0cb84..ed8f389 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window-expected.txt
+++ b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window-expected.txt
@@ -1,6 +1,5 @@
 This is a testharness.js-based test.
-PASS idlharness
-PASS picture-in-picture interfaces.
+FAIL picture-in-picture interfaces. promise_test: Unhandled rejection with value: object "NotSupportedError: Picture-in-Picture is not available."
 PASS Partial interface HTMLVideoElement: original interface defined
 PASS Partial interface Document: original interface defined
 PASS Partial interface DocumentOrShadowRoot: original interface defined
diff --git a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window.js
index 6ada0fe..ac4c0bf 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/idlharness.window.js
@@ -8,24 +8,19 @@
 
 // https://wicg.github.io/picture-in-picture/
 
-promise_test(async () => {
-  try {
+idl_test(
+  ['picture-in-picture'],
+  ['html', 'dom'],
+  async idl_array => {
+    idl_array.add_objects({
+      Document: ['document'],
+      DocumentOrShadowRoot: ['document'],
+      HTMLVideoElement: ['video'],
+      PictureInPictureWindow: ['pipw'],
+    });
+
     self.video = await loadVideo();
     self.pipw = await requestPictureInPictureWithTrustedClick(video);
-  } catch (e) {
-    // Will be surfaced when video/pipw are undefined below.
-  }
-
-  idl_test(
-    ['picture-in-picture'],
-    ['html', 'dom'],
-    idl_array => {
-      idl_array.add_objects({
-        Document: ['document'],
-        DocumentOrShadowRoot: ['document'],
-        HTMLVideoElement: ['video'],
-        PictureInPictureWindow: ['pipw'],
-      });
-    },
-    'picture-in-picture interfaces.');
-})
+  },
+  'picture-in-picture interfaces.'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/request-picture-in-picture-twice.html b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/request-picture-in-picture-twice.html
index 7f1f81f8..556cf83 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/request-picture-in-picture-twice.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/request-picture-in-picture-twice.html
@@ -10,13 +10,12 @@
 promise_test(async t => {
   const video1 = await loadVideo();
   const video2 = await loadVideo();
-  return callWithTrustedClick(() => {
-    const first = video1.requestPictureInPicture();
-    const second = video2.requestPictureInPicture();
-    return Promise.all([
-      first,
-      promise_rejects(t, 'NotAllowedError', second)
-    ]);
-  });
+  return test_driver.bless(
+    'request picture in picture',
+    async () => {
+      await video1.requestPictureInPicture();
+      promise_rejects(t, 'NotAllowedError', video2.requestPictureInPicture());
+    }
+  );
 }, 'request Picture-in-Picture consumes user gesture');
 </script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/resources/picture-in-picture-helpers.js b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/resources/picture-in-picture-helpers.js
index 4514edf..86b9f29 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/resources/picture-in-picture-helpers.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/picture-in-picture/resources/picture-in-picture-helpers.js
@@ -4,38 +4,19 @@
   }
 }
 
-function callWithTrustedClick(callback) {
-  return new Promise((resolve, reject) => {
-    let button = document.createElement('button');
-    button.textContent = 'click to continue test';
-    button.style.display = 'block';
-    button.style.fontSize = '20px';
-    button.style.padding = '10px';
-    button.onclick = () => {
-      document.body.removeChild(button);
-      resolve(callback());
-    };
-    document.body.appendChild(button);
-    test_driver.click(button).catch(_ => reject('Click failed'));
-  });
-}
-
 function loadVideo(activeDocument, sourceUrl) {
   return new Promise((resolve, reject) => {
     const document = activeDocument || window.document;
     const video = document.createElement('video');
     video.src = sourceUrl || '/media/movie_5.ogv';
-    video.onloadedmetadata = () => {
-      resolve(video);
-    };
-    video.onerror = error => {
-      reject(error);
-    };
+    video.onloadedmetadata = () => { resolve(video); };
+    video.onerror = error => { reject(error); };
   });
 }
 
 // Calls requestPictureInPicture() in a context that's 'allowed to request PiP'.
 function requestPictureInPictureWithTrustedClick(videoElement) {
-  return callWithTrustedClick(
-      () => videoElement.requestPictureInPicture());
+  return test_driver.bless(
+    'request picture in picture',
+    () => videoElement.requestPictureInPicture());
 }
diff --git a/third_party/WebKit/LayoutTests/external/wpt/presentation-api/controlling-ua/idlharness.https.html b/third_party/WebKit/LayoutTests/external/wpt/presentation-api/controlling-ua/idlharness.https.html
index 04d1741..52aa8c5 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/presentation-api/controlling-ua/idlharness.https.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/presentation-api/controlling-ua/idlharness.https.html
@@ -26,16 +26,24 @@
     idl_array.add_dependency_idls(dom);
     idl_array.add_dependency_idls(html);
 
-    window.presentation_request = new PresentationRequest("/presentation-api/receiving-ua/idlharness.html");
-    window.presentation_request_urls = new PresentationRequest([
-      "/presentation-api/receiving-ua/idlharness.html",
-      "https://www.example.com/presentation.html"
-    ]);
-    navigator.presentation.defaultRequest = presentation_request;
+    try {
+      window.presentation_request = new PresentationRequest("/presentation-api/receiving-ua/idlharness.html");
+      window.presentation_request_urls = new PresentationRequest([
+        "/presentation-api/receiving-ua/idlharness.html",
+        "https://www.example.com/presentation.html"
+      ]);
+      navigator.presentation.defaultRequest = window.presentation_request;
+    } catch (e) {
+      // Will be surfaced in idlharness.js's test_object below.
+    }
 
     idl_array.add_objects({
-        Presentation: ['navigator.presentation'],
-        PresentationRequest: ['navigator.presentation.defaultRequest', 'presentation_request', 'presentation_request_urls']
+      Presentation: ['navigator.presentation'],
+      PresentationRequest: [
+        'navigator.presentation.defaultRequest',
+        'presentation_request',
+        'presentation_request_urls'
+      ],
     });
     idl_array.test();
   }, "Test IDL implementation of Presentation API");
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.html b/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.html
deleted file mode 100644
index 51ab4fd..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.html
+++ /dev/null
@@ -1,20 +0,0 @@
-<!doctype html>
-<title>ResizeObserver IDL tests</title>
-<link rel="help" href="https://wicg.github.io/ResizeObserver/">
-<script src="/resources/testharness.js"></script>
-<script src="/resources/testharnessreport.js"></script>
-<script src="/resources/WebIDLParser.js"></script>
-<script src="/resources/idlharness.js"></script>
-<script>
-  "use strict";
-
-  promise_test(async () => {
-    const idl_array = new IdlArray();
-    const idl = await fetch("/interfaces/ResizeObserver.idl").then(r => r.text());
-    idl_array.add_idls(idl);
-    idl_array.add_objects({
-      ResizeObserver: ["new ResizeObserver(entries => {})"],
-    });
-    idl_array.test();
-  }, "Test IDL implementation of ResizeObserver");
-</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window-expected.txt
new file mode 100644
index 0000000..f07e46a
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window-expected.txt
@@ -0,0 +1,52 @@
+This is a testharness.js-based test.
+PASS Test IDL implementation of ResizeObserver
+PASS ResizeObserverEntry creator
+PASS ResizeObserver interface: existence and properties of interface object
+PASS ResizeObserver interface object length
+PASS ResizeObserver interface object name
+PASS ResizeObserver interface: existence and properties of interface prototype object
+PASS ResizeObserver interface: existence and properties of interface prototype object's "constructor" property
+PASS ResizeObserver interface: existence and properties of interface prototype object's @@unscopables property
+PASS ResizeObserver interface: operation observe(Element)
+PASS Unscopable handled correctly for observe(Element) on ResizeObserver
+PASS ResizeObserver interface: operation unobserve(Element)
+PASS Unscopable handled correctly for unobserve(Element) on ResizeObserver
+PASS ResizeObserver interface: operation disconnect()
+PASS Unscopable handled correctly for disconnect() on ResizeObserver
+PASS ResizeObserver must be primary interface of observer
+PASS Stringification of observer
+PASS ResizeObserver interface: observer must inherit property "observe(Element)" with the proper type
+PASS ResizeObserver interface: calling observe(Element) on observer with too few arguments must throw TypeError
+PASS ResizeObserver interface: observer must inherit property "unobserve(Element)" with the proper type
+PASS ResizeObserver interface: calling unobserve(Element) on observer with too few arguments must throw TypeError
+PASS ResizeObserver interface: observer must inherit property "disconnect()" with the proper type
+PASS ResizeObserverEntry interface: existence and properties of interface object
+FAIL ResizeObserverEntry interface object length assert_equals: wrong value for ResizeObserverEntry.length expected 1 but got 0
+PASS ResizeObserverEntry interface object name
+PASS ResizeObserverEntry interface: existence and properties of interface prototype object
+PASS ResizeObserverEntry interface: existence and properties of interface prototype object's "constructor" property
+PASS ResizeObserverEntry interface: existence and properties of interface prototype object's @@unscopables property
+PASS ResizeObserverEntry interface: attribute target
+PASS Unscopable handled correctly for target property on ResizeObserverEntry
+PASS ResizeObserverEntry interface: attribute contentRect
+PASS Unscopable handled correctly for contentRect property on ResizeObserverEntry
+PASS ResizeObserverEntry must be primary interface of entry
+PASS Stringification of entry
+PASS ResizeObserverEntry interface: entry must inherit property "target" with the proper type
+PASS ResizeObserverEntry interface: entry must inherit property "contentRect" with the proper type
+FAIL ResizeObservation interface: existence and properties of interface object assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface object length assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface object name assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface: existence and properties of interface prototype object assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+FAIL ResizeObservation interface: attribute target assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+PASS Unscopable handled correctly for target property on ResizeObservation
+FAIL ResizeObservation interface: attribute broadcastWidth assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+PASS Unscopable handled correctly for broadcastWidth property on ResizeObservation
+FAIL ResizeObservation interface: attribute broadcastHeight assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+PASS Unscopable handled correctly for broadcastHeight property on ResizeObservation
+FAIL ResizeObservation interface: operation isActive() assert_own_property: self does not have own property "ResizeObservation" expected property "ResizeObservation" missing
+PASS Unscopable handled correctly for isActive() on ResizeObservation
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window.js
new file mode 100644
index 0000000..240f5ec
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/resize-observer/idlharness.window.js
@@ -0,0 +1,38 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+// META: script=resources/resizeTestHelper.js
+
+'use strict';
+
+// https://wicg.github.io/ResizeObserver/
+
+idl_test(
+  ['ResizeObserver'],
+  ['dom', 'geometry'],
+  async idl_array => {
+    idl_array.add_objects({
+      ResizeObserver: ['observer'],
+      ResizeObserverEntry: ['entry'],
+    });
+
+    const div = document.createElement('div');
+    document.body.appendChild(div);
+    let helper = new ResizeTestHelper(
+      "ResizeObserverEntry creator",
+      [
+        {
+          setup: observer => {
+            self.observer = observer;
+            observer.observe(div);
+            div.style.width = "5px";
+          },
+          notify: entries => {
+            self.entry = entries[0];
+            assert_equals(entries[0].contentRect.width, 5, "target width");
+          }
+        }
+      ]);
+    await helper.start();
+  },
+  'Test IDL implementation of ResizeObserver'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resource-timing/idlharness.any.js b/third_party/WebKit/LayoutTests/external/wpt/resource-timing/idlharness.any.js
index d522aae..6dff5e85 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/resource-timing/idlharness.any.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/resource-timing/idlharness.any.js
@@ -5,24 +5,20 @@
 
 // https://w3c.github.io/resource-timing/
 
-promise_test(async () => {
-  const [idl, perf, hrtime, dom, html] = await Promise.all([
-    '/interfaces/resource-timing.idl',
-    '/interfaces/performance-timeline.idl',
-    '/interfaces/hr-time.idl',
-    '/interfaces/dom.idl',
-    '/interfaces/html.idl',
-  ].map(url => fetch(url).then(r => r.text())));
+idl_test(
+  ['resource-timing'],
+  ['performance-timeline', 'hr-time', 'dom', 'html'],
+  idl_array => {
+    try {
+      self.resource = performance.getEntriesByType('resource')[0];
+    } catch (e) {
+      // Will be surfaced when resource is undefined below.
+    }
 
-  const idl_array = new IdlArray();
-  idl_array.add_idls(idl);
-  idl_array.add_dependency_idls(perf);
-  idl_array.add_dependency_idls(hrtime);
-  idl_array.add_dependency_idls(dom);
-  idl_array.add_dependency_idls(html);
-  idl_array.add_objects({
-    Performance: ['performance'],
-    PerformanceResourceTiming: ["performance.getEntriesByType('resource')[0]"]
-  });
-  idl_array.test();
-}, 'Test server-timing IDL implementation');
+    idl_array.add_objects({
+      Performance: ['performance'],
+      PerformanceResourceTiming: ['resource']
+    });
+  },
+  'Test server-timing IDL implementation'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resources/chromium/webxr-test.js b/third_party/WebKit/LayoutTests/external/wpt/resources/chromium/webxr-test.js
index 74be1f0..224befd 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/resources/chromium/webxr-test.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/resources/chromium/webxr-test.js
@@ -62,11 +62,14 @@
 }
 
 // Implements both VRDisplayHost and VRMagicWindowProvider. Maintains a mock for
-// VRPresentationProvider.
+// XRPresentationProvider.
 class MockDevice {
   constructor(fakeDeviceInit, service) {
     this.displayClient_ = new device.mojom.VRDisplayClientPtr();
-    this.presentation_provider_ = new MockVRPresentationProvider();
+    this.presentation_provider_ = new MockXRPresentationProvider();
+
+    this.pose_ = null;
+    this.next_frame_id_ = 0;
 
     this.service_ = service;
 
@@ -101,7 +104,7 @@
     if (poseMatrix == null) {
       this.presentation_provider_.pose_ = null;
     } else {
-      this.presentation_provider_.setPoseFromMatrix(poseMatrix);
+      this.setPoseFromMatrix(poseMatrix);
     }
 
     if (views) {
@@ -122,6 +125,49 @@
     }
   }
 
+
+  setPoseFromMatrix(poseMatrix) {
+    this.pose_ = {
+      orientation: null,
+      position: null,
+      angularVelocity: null,
+      linearVelocity: null,
+      angularAcceleration: null,
+      linearAcceleration: null,
+      inputState: null,
+      poseIndex: 0
+    };
+
+    let pose = this.poseFromMatrix(poseMatrix);
+    for (let field in pose) {
+      if (this.pose_.hasOwnProperty(field)) {
+        this.pose_[field] = pose[field];
+      }
+    }
+  }
+
+  poseFromMatrix(m) {
+    let orientation = [];
+
+    let m00 = m[0];
+    let m11 = m[5];
+    let m22 = m[10];
+    // The max( 0, ... ) is just a safeguard against rounding error.
+    orientation[3] = Math.sqrt(Math.max(0, 1 + m00 + m11 + m22)) / 2;
+    orientation[0] = Math.sqrt(Math.max(0, 1 + m00 - m11 - m22)) / 2;
+    orientation[1] = Math.sqrt(Math.max(0, 1 - m00 + m11 - m22)) / 2;
+    orientation[2] = Math.sqrt(Math.max(0, 1 - m00 - m11 + m22)) / 2;
+
+    let position = [];
+    position[0] = m[12];
+    position[1] = m[13];
+    position[2] = m[14];
+
+    return {
+      orientation, position
+    }
+  }
+
   getNonImmersiveDisplayInfo() {
     let displayInfo = this.getImmersiveDisplayInfo();
 
@@ -202,158 +248,7 @@
 
   // Mojo function implementations.
 
-  // VRMagicWindowProvider implementation.
-
-  getFrameData() {
-    // Convert current document time to monotonic time.
-    let now = window.performance.now() / 1000.0;
-    let diff = now - internals.monotonicTimeToZeroBasedDocumentTime(now);
-    now += diff;
-    now *= 1000000;
-
-    return Promise.resolve({
-      frameData: {
-        pose: this.presentation_provider_.pose_,
-        bufferHolder: null,
-        bufferSize: {},
-        timeDelta: [],
-        projectionMatrix: [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]
-      }
-    });
-  }
-
-  updateSessionGeometry(frame_size, display_rotation) {
-    // This function must exist to ensure that calls to it do not crash, but we
-    // do not have any use for this data at present.
-  }
-
-  // VRDisplayHost implementation.
-
-  requestSession(sessionOptions, was_activation) {
-    return this.supportsSession(sessionOptions).then((result) => {
-      // The JavaScript bindings convert c_style_names to camelCase names.
-      let options = new device.mojom.VRDisplayFrameTransportOptions();
-      options.transportMethod =
-          device.mojom.VRDisplayFrameTransportMethod.SUBMIT_AS_MAILBOX_HOLDER;
-      options.waitForTransferNotification = true;
-      options.waitForRenderNotification = true;
-
-      let connection;
-      if (result.supportsSession) {
-        connection = {
-          clientRequest: this.presentation_provider_.getClientRequest(),
-          provider: this.presentation_provider_.bindProvider(sessionOptions),
-          transportOptions: options
-        };
-
-        let magicWindowPtr = new device.mojom.VRMagicWindowProviderPtr();
-        let magicWindowRequest = mojo.makeRequest(magicWindowPtr);
-        let magicWindowBinding = new mojo.Binding(
-            device.mojom.VRMagicWindowProvider, this, magicWindowRequest);
-
-        return Promise.resolve({
-          session:
-              {connection: connection, magicWindowProvider: magicWindowPtr}
-        });
-      } else {
-        return Promise.resolve({session: null});
-      }
-    });
-  }
-
-  supportsSession(options) {
-    return Promise.resolve({
-      supportsSession:
-          !options.exclusive || this.displayInfo_.capabilities.canPresent
-    });
-  };
-}
-
-class MockVRPresentationProvider {
-  constructor() {
-    this.binding_ = new mojo.Binding(device.mojom.VRPresentationProvider, this);
-    this.pose_ = null;
-    this.next_frame_id_ = 0;
-    this.submit_frame_count_ = 0;
-    this.missing_frame_count_ = 0;
-  }
-
-  bindProvider(request) {
-    let providerPtr = new device.mojom.VRPresentationProviderPtr();
-    let providerRequest = mojo.makeRequest(providerPtr);
-
-    this.binding_.close();
-
-    this.binding_ = new mojo.Binding(
-        device.mojom.VRPresentationProvider, this, providerRequest);
-
-    return providerPtr;
-  }
-
-  getClientRequest() {
-    this.submitFrameClient_ = new device.mojom.VRSubmitFrameClientPtr();
-    return mojo.makeRequest(this.submitFrameClient_);
-  }
-
-  setPoseFromMatrix(poseMatrix) {
-    this.pose_ = {
-      orientation: null,
-      position: null,
-      angularVelocity: null,
-      linearVelocity: null,
-      angularAcceleration: null,
-      linearAcceleration: null,
-      inputState: null,
-      poseIndex: 0
-    };
-
-    let pose = this.poseFromMatrix(poseMatrix);
-    for (let field in pose) {
-      if (this.pose_.hasOwnProperty(field)) {
-        this.pose_[field] = pose[field];
-      }
-    }
-  }
-
-  poseFromMatrix(m) {
-    let orientation = [];
-
-    let m00 = m[0];
-    let m11 = m[5];
-    let m22 = m[10];
-    // The max( 0, ... ) is just a safeguard against rounding error.
-    orientation[3] = Math.sqrt(Math.max(0, 1 + m00 + m11 + m22)) / 2;
-    orientation[0] = Math.sqrt(Math.max(0, 1 + m00 - m11 - m22)) / 2;
-    orientation[1] = Math.sqrt(Math.max(0, 1 - m00 + m11 - m22)) / 2;
-    orientation[2] = Math.sqrt(Math.max(0, 1 - m00 - m11 + m22)) / 2;
-
-    let position = [];
-    position[0] = m[12];
-    position[1] = m[13];
-    position[2] = m[14];
-
-    return {
-      orientation, position
-    }
-  }
-
-  // VRPresentationProvider mojo implementation
-  submitFrameMissing(frameId, mailboxHolder, timeWaited) {
-    this.missing_frame_count_++;
-  }
-
-  submitFrame(frameId, mailboxHolder, timeWaited) {
-    this.submit_frame_count_++;
-
-    // Trigger the submit completion callbacks here. WARNING: The
-    // Javascript-based mojo mocks are *not* re-entrant. It's OK to
-    // wait for these notifications on the next frame, but waiting
-    // within the current frame would never finish since the incoming
-    // calls would be queued until the current execution context finishes.
-    this.submitFrameClient_.onSubmitFrameTransferred(true);
-    this.submitFrameClient_.onSubmitFrameRendered();
-  }
-
+  // XRFrameDataProvider implementation.
   getFrameData() {
     if (this.pose_) {
       this.pose_.poseIndex++;
@@ -378,6 +273,105 @@
       }
     });
   }
+
+  updateSessionGeometry(frame_size, display_rotation) {
+    // This function must exist to ensure that calls to it do not crash, but we
+    // do not have any use for this data at present.
+  }
+
+  // VRDisplayHost implementation.
+
+  requestSession(sessionOptions, was_activation) {
+    return this.supportsSession(sessionOptions).then((result) => {
+      // The JavaScript bindings convert c_style_names to camelCase names.
+      let options = new device.mojom.XRPresentationTransportOptions();
+      options.transportMethod =
+          device.mojom.XRPresentationTransportMethod.SUBMIT_AS_MAILBOX_HOLDER;
+      options.waitForTransferNotification = true;
+      options.waitForRenderNotification = true;
+
+      let submit_frame_sink;
+      if (result.supportsSession) {
+        submit_frame_sink = {
+          clientRequest: this.presentation_provider_.getClientRequest(),
+          provider: this.presentation_provider_.bindProvider(sessionOptions),
+          transportOptions: options
+        };
+
+        let dataProviderPtr = new device.mojom.XRFrameDataProviderPtr();
+        let dataProviderRequest = mojo.makeRequest(dataProviderPtr);
+        let dataProviderBinding = new mojo.Binding(
+            device.mojom.XRFrameDataProvider, this, dataProviderRequest);
+
+        let enviromentProviderPtr =
+            new device.mojom.XREnviromentIntegrationProviderPtr();
+        let enviromentProviderRequest = mojo.makeRequest(enviromentProviderPtr);
+        let enviromentProviderBinding = new mojo.Binding(
+            device.mojom.XREnviromentIntegrationProvider, this,
+            enviromentProviderRequest);
+
+        return Promise.resolve({
+          session: {
+            submitFrameSink: submit_frame_sink,
+            dataProvider: dataProviderPtr,
+            enviromentProvider: enviromentProviderPtr
+          }
+        });
+      } else {
+        return Promise.resolve({session: null});
+      }
+    });
+  }
+
+  supportsSession(options) {
+    return Promise.resolve({
+      supportsSession:
+          !options.exclusive || this.displayInfo_.capabilities.canPresent
+    });
+  };
+}
+
+class MockXRPresentationProvider {
+  constructor() {
+    this.binding_ = new mojo.Binding(device.mojom.XRPresentationProvider, this);
+
+    this.submit_frame_count_ = 0;
+    this.missing_frame_count_ = 0;
+  }
+
+  bindProvider(request) {
+    let providerPtr = new device.mojom.XRPresentationProviderPtr();
+    let providerRequest = mojo.makeRequest(providerPtr);
+
+    this.binding_.close();
+
+    this.binding_ = new mojo.Binding(
+        device.mojom.XRPresentationProvider, this, providerRequest);
+
+    return providerPtr;
+  }
+
+  getClientRequest() {
+    this.submitFrameClient_ = new device.mojom.XRPresentationClientPtr();
+    return mojo.makeRequest(this.submitFrameClient_);
+  }
+
+  // XRPresentationProvider mojo implementation
+  submitFrameMissing(frameId, mailboxHolder, timeWaited) {
+    this.missing_frame_count_++;
+  }
+
+  submitFrame(frameId, mailboxHolder, timeWaited) {
+    this.submit_frame_count_++;
+
+    // Trigger the submit completion callbacks here. WARNING: The
+    // Javascript-based mojo mocks are *not* re-entrant. It's OK to
+    // wait for these notifications on the next frame, but waiting
+    // within the current frame would never finish since the incoming
+    // calls would be queued until the current execution context finishes.
+    this.submitFrameClient_.onSubmitFrameTransferred(true);
+    this.submitFrameClient_.onSubmitFrameRendered();
+  }
 }
 
 let XRTest = new ChromeXRTest();
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/external/wpt/resources/idlharness.js b/third_party/WebKit/LayoutTests/external/wpt/resources/idlharness.js
index bfb2582..6f89a71 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/resources/idlharness.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/resources/idlharness.js
@@ -870,6 +870,16 @@
 
             test(function () {
                 assert_true(originalExists, `Original ${parsed_idl.type} should be defined`);
+
+                var expected = IdlInterface;
+                switch (parsed_idl.type) {
+                    case 'interface': expected = IdlInterface; break;
+                    case 'dictionary': expected = IdlDictionary; break;
+                    case 'namespace': expected = IdlNamespace; break;
+                }
+                assert_true(
+                    expected.prototype.isPrototypeOf(this.members[parsed_idl.name]),
+                    `Original ${parsed_idl.name} definition should have type ${parsed_idl.type}`);
             }.bind(this), `Partial ${parsed_idl.type} ${partialTestName}: original ${parsed_idl.type} defined`);
         }
         if (!originalExists) {
@@ -3187,7 +3197,9 @@
                 }
             })
             .then(function() {
-                return idl_setup_func(idl_array, t);
+                if (idl_setup_func) {
+                    return idl_setup_func(idl_array, t);
+                }
             })
             .then(function() { idl_array.test(); })
             .catch(function (reason) {
diff --git a/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window-expected.txt
new file mode 100644
index 0000000..949a6b7
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window-expected.txt
@@ -0,0 +1,103 @@
+This is a testharness.js-based test.
+Found 99 tests; 93 PASS, 6 FAIL, 0 TIMEOUT, 0 NOTRUN.
+PASS selection-api interfaces
+PASS Partial interface Document: original interface defined
+PASS Partial interface Window: original interface defined
+PASS Partial interface GlobalEventHandlers: original interface defined
+PASS Selection interface: existence and properties of interface object
+PASS Selection interface object length
+PASS Selection interface object name
+PASS Selection interface: existence and properties of interface prototype object
+PASS Selection interface: existence and properties of interface prototype object's "constructor" property
+PASS Selection interface: existence and properties of interface prototype object's @@unscopables property
+PASS Selection interface: attribute anchorNode
+PASS Unscopable handled correctly for anchorNode property on Selection
+PASS Selection interface: attribute anchorOffset
+PASS Unscopable handled correctly for anchorOffset property on Selection
+PASS Selection interface: attribute focusNode
+PASS Unscopable handled correctly for focusNode property on Selection
+PASS Selection interface: attribute focusOffset
+PASS Unscopable handled correctly for focusOffset property on Selection
+PASS Selection interface: attribute isCollapsed
+PASS Unscopable handled correctly for isCollapsed property on Selection
+PASS Selection interface: attribute rangeCount
+PASS Unscopable handled correctly for rangeCount property on Selection
+PASS Selection interface: attribute type
+PASS Unscopable handled correctly for type property on Selection
+PASS Selection interface: operation getRangeAt(unsigned long)
+PASS Unscopable handled correctly for getRangeAt(unsigned long) on Selection
+PASS Selection interface: operation addRange(Range)
+PASS Unscopable handled correctly for addRange(Range) on Selection
+PASS Selection interface: operation removeRange(Range)
+PASS Unscopable handled correctly for removeRange(Range) on Selection
+PASS Selection interface: operation removeAllRanges()
+PASS Unscopable handled correctly for removeAllRanges() on Selection
+PASS Selection interface: operation empty()
+PASS Unscopable handled correctly for empty() on Selection
+PASS Selection interface: operation collapse(Node, unsigned long)
+PASS Unscopable handled correctly for collapse(Node, unsigned long) on Selection
+PASS Selection interface: operation setPosition(Node, unsigned long)
+PASS Unscopable handled correctly for setPosition(Node, unsigned long) on Selection
+PASS Selection interface: operation collapseToStart()
+PASS Unscopable handled correctly for collapseToStart() on Selection
+PASS Selection interface: operation collapseToEnd()
+PASS Unscopable handled correctly for collapseToEnd() on Selection
+PASS Selection interface: operation extend(Node, unsigned long)
+PASS Unscopable handled correctly for extend(Node, unsigned long) on Selection
+PASS Selection interface: operation setBaseAndExtent(Node, unsigned long, Node, unsigned long)
+PASS Unscopable handled correctly for setBaseAndExtent(Node, unsigned long, Node, unsigned long) on Selection
+PASS Selection interface: operation selectAllChildren(Node)
+PASS Unscopable handled correctly for selectAllChildren(Node) on Selection
+PASS Selection interface: operation deleteFromDocument()
+PASS Unscopable handled correctly for deleteFromDocument() on Selection
+PASS Selection interface: operation containsNode(Node, boolean)
+PASS Unscopable handled correctly for containsNode(Node, boolean) on Selection
+PASS Selection interface: stringifier
+PASS Selection must be primary interface of getSelection()
+PASS Stringification of getSelection()
+PASS Selection interface: getSelection() must inherit property "anchorNode" with the proper type
+PASS Selection interface: getSelection() must inherit property "anchorOffset" with the proper type
+PASS Selection interface: getSelection() must inherit property "focusNode" with the proper type
+PASS Selection interface: getSelection() must inherit property "focusOffset" with the proper type
+PASS Selection interface: getSelection() must inherit property "isCollapsed" with the proper type
+PASS Selection interface: getSelection() must inherit property "rangeCount" with the proper type
+PASS Selection interface: getSelection() must inherit property "type" with the proper type
+PASS Selection interface: getSelection() must inherit property "getRangeAt(unsigned long)" with the proper type
+PASS Selection interface: calling getRangeAt(unsigned long) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "addRange(Range)" with the proper type
+PASS Selection interface: calling addRange(Range) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "removeRange(Range)" with the proper type
+PASS Selection interface: calling removeRange(Range) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "removeAllRanges()" with the proper type
+PASS Selection interface: getSelection() must inherit property "empty()" with the proper type
+PASS Selection interface: getSelection() must inherit property "collapse(Node, unsigned long)" with the proper type
+PASS Selection interface: calling collapse(Node, unsigned long) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "setPosition(Node, unsigned long)" with the proper type
+PASS Selection interface: calling setPosition(Node, unsigned long) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "collapseToStart()" with the proper type
+PASS Selection interface: getSelection() must inherit property "collapseToEnd()" with the proper type
+PASS Selection interface: getSelection() must inherit property "extend(Node, unsigned long)" with the proper type
+PASS Selection interface: calling extend(Node, unsigned long) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "setBaseAndExtent(Node, unsigned long, Node, unsigned long)" with the proper type
+PASS Selection interface: calling setBaseAndExtent(Node, unsigned long, Node, unsigned long) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "selectAllChildren(Node)" with the proper type
+PASS Selection interface: calling selectAllChildren(Node) on getSelection() with too few arguments must throw TypeError
+PASS Selection interface: getSelection() must inherit property "deleteFromDocument()" with the proper type
+PASS Selection interface: getSelection() must inherit property "containsNode(Node, boolean)" with the proper type
+PASS Selection interface: calling containsNode(Node, boolean) on getSelection() with too few arguments must throw TypeError
+PASS Document interface: operation getSelection()
+PASS Unscopable handled correctly for getSelection() on Document
+PASS Document interface: document must inherit property "getSelection()" with the proper type
+PASS Window interface: operation getSelection()
+PASS Unscopable handled correctly for getSelection() on Window
+FAIL Window interface: attribute onselectstart assert_own_property: The global object must have a property "onselectstart" expected property "onselectstart" missing
+PASS Unscopable handled correctly for onselectstart property on Window
+FAIL Window interface: attribute onselectionchange assert_own_property: The global object must have a property "onselectionchange" expected property "onselectionchange" missing
+PASS Unscopable handled correctly for onselectionchange property on Window
+PASS Window interface: window must inherit property "getSelection()" with the proper type
+FAIL Window interface: window must inherit property "onselectstart" with the proper type assert_own_property: expected property "onselectstart" missing
+FAIL Window interface: window must inherit property "onselectionchange" with the proper type assert_own_property: expected property "onselectionchange" missing
+FAIL GlobalEventHandlers interface: self must inherit property "onselectstart" with the proper type assert_inherits: property "onselectstart" not found in prototype chain
+FAIL GlobalEventHandlers interface: self must inherit property "onselectionchange" with the proper type assert_inherits: property "onselectionchange" not found in prototype chain
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window.js
new file mode 100644
index 0000000..d7b851d
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/selection/idlharness.window.js
@@ -0,0 +1,20 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+'use strict';
+
+// https://w3c.github.io/selection-api/
+
+idl_test(
+  ['selection-api'],
+  ['dom', 'html'],
+  idlArray => {
+    idlArray.add_objects({
+      Window: ['window'],
+      Document: ['document'],
+      Selection: ['getSelection()'],
+      GlobalEventHandlers: ['self'],
+    });
+  },
+  'selection-api interfaces'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/selection/interfaces.html b/third_party/WebKit/LayoutTests/external/wpt/selection/interfaces.html
deleted file mode 100644
index 6b72f28..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/selection/interfaces.html
+++ /dev/null
@@ -1,33 +0,0 @@
-<!doctype html>
-<title>Selection interface tests</title>
-<div id=log></div>
-<script src=/resources/testharness.js></script>
-<script src=/resources/testharnessreport.js></script>
-<script src=/resources/WebIDLParser.js></script>
-<script src=/resources/idlharness.js></script>
-<script>
-"use strict";
-
-function doTest([selection, dom, cssom, touchevents, uievents, html]) {
-  var idlArray = new IdlArray();
-  idlArray.add_untested_idls('interface SVGElement {};');
-  idlArray.add_untested_idls(dom + cssom + touchevents + uievents + html);
-  idlArray.add_idls(selection);
-  idlArray.add_objects({Selection: ['getSelection()']});
-  idlArray.test();
-}
-
-function fetchData(url) {
-  return fetch(url).then((response) => response.text());
-}
-
-promise_test(function() {
-  return Promise.all([fetchData("/interfaces/selection-api.idl"),
-                      fetchData("/interfaces/dom.idl"),
-                      fetchData("/interfaces/cssom.idl"),
-                      fetchData("/interfaces/touchevents.idl"),
-                      fetchData("/interfaces/uievents.idl"),
-                      fetchData("/interfaces/html.idl")])
-                .then(doTest);
-}, "Test driver");
-</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/interfaces-window.https.html b/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/interfaces-window.https.html
index 8786b29..d3e85f2 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/interfaces-window.https.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/interfaces-window.https.html
@@ -4,18 +4,16 @@
 <script src="/resources/testharnessreport.js"></script>
 <script src="/resources/WebIDLParser.js"></script>
 <script src="/resources/idlharness.js"></script>
-<script src="resources/interfaces-idls.js"></script>
 <script src="resources/test-helpers.sub.js"></script>
 <script>
 'use strict';
 
 promise_test(async (t) => {
-  const srcs = ['dom', 'service-workers'];
-  const [dom, serviceWorkerIdl] = await Promise.all(
+  const srcs = ['dom', 'html', 'service-workers', 'dedicated-workers'];
+  const [dom, html, serviceWorkerIdl, dedicated] = await Promise.all(
     srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
 
   var idlArray = new IdlArray();
-  idlArray.add_untested_idls(idls.untested);
   idlArray.add_idls(serviceWorkerIdl, { only: [
     'ServiceWorkerGlobalScope',
     'Client',
@@ -30,7 +28,9 @@
     'Cache',
     'CacheStorage',
   ]});
+  idlArray.add_dependency_idls(dedicated);
   idlArray.add_dependency_idls(dom);
+  idlArray.add_dependency_idls(html);
   idlArray.add_objects({
     ServiceWorkerContainer: ['navigator.serviceWorker']
   });
diff --git a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-idls.js b/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-idls.js
deleted file mode 100644
index 3db45c6..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-idls.js
+++ /dev/null
@@ -1,14 +0,0 @@
-var idls = {};
-idls.untested = `
-[Exposed=Worker]
-interface WorkerGlobalScope {};
-
-[TreatNonObjectAsNull]
-callback EventHandlerNonNull = any (Event event);
-typedef EventHandlerNonNull? EventHandler;
-
-[NoInterfaceObject, Exposed=(Window,Worker)]
-interface AbstractWorker {
-  attribute EventHandler onerror;
-};
-`;
diff --git a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-worker.sub.js b/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-worker.sub.js
index 606ec07..10fe14d5 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-worker.sub.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/service-workers/service-worker/resources/interfaces-worker.sub.js
@@ -1,17 +1,15 @@
 'use strict';
 
-importScripts('interfaces-idls.js');
 importScripts('worker-testharness.js');
 importScripts('/resources/WebIDLParser.js');
 importScripts('/resources/idlharness.js');
 
 promise_test(async (t) => {
-  const srcs = ['dom', 'service-workers'];
-  const [dom, serviceWorkerIdl] = await Promise.all(
+  const srcs = ['dom', 'html', 'service-workers', 'dedicated-workers'];
+  const [dom, html, serviceWorkerIdl, dedicated] = await Promise.all(
     srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
 
   var idlArray = new IdlArray();
-  idlArray.add_untested_idls(idls.untested);
   idlArray.add_idls(serviceWorkerIdl, { only: [
     'ServiceWorkerGlobalScope',
     'Client',
@@ -26,7 +24,9 @@
     'Cache',
     'CacheStorage',
   ]});
+  idlArray.add_dependency_idls(dedicated);
   idlArray.add_dependency_idls(dom);
+  idlArray.add_dependency_idls(html);
   idlArray.add_objects({
     ServiceWorkerGlobalScope: ['self'],
     Clients: ['self.clients'],
diff --git a/third_party/WebKit/LayoutTests/external/wpt/svg/linking/reftests/use-descendant-combinator-003.html b/third_party/WebKit/LayoutTests/external/wpt/svg/linking/reftests/use-descendant-combinator-003.html
new file mode 100644
index 0000000..14bf5bd1
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/svg/linking/reftests/use-descendant-combinator-003.html
@@ -0,0 +1,37 @@
+<!doctype html>
+<meta charset=utf-8>
+<title>CSS Test: use element doesn't cross shadow tree boundaries in selector-matching, and is invalidated properly</title>
+<link rel="author" title="Emilio Cobos Álvarez" href="mailto:emilio@crisal.io">
+<link rel="help" href="https://svgwg.org/svg2-draft/struct.html#UseElement">
+<link rel="match" href="/svg/linking/reftests/use-descendant-combinator-ref.html">
+<style>
+#test rect {
+  stroke: red;
+  stroke-width: 10px;
+}
+.inside-use rect {
+  fill: red;
+}
+defs .inside-use rect {
+  fill: red;
+}
+</style>
+<p>
+  You should see a green square, and no red.
+</p>
+<svg>
+  <defs>
+    <g id="square">
+      <g class="inside-use">
+        <rect width="100" height="100"/>
+      </g>
+    </g>
+  </defs>
+  <g id="test">
+    <use href="#square" />
+  </g>
+</svg>
+<script>
+  document.body.offsetTop;
+  document.styleSheets[0].cssRules[1].style.fill = 'green';
+</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001-ref.svg b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001-ref.svg
new file mode 100644
index 0000000..dcd39a8
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001-ref.svg
@@ -0,0 +1,32 @@
+<svg id="svg-root"
+  width="100%" height="100%" viewBox="0 0 480 360"
+  xmlns="http://www.w3.org/2000/svg"
+  xmlns:xlink="http://www.w3.org/1999/xlink"
+  xmlns:html="http://www.w3.org/1999/xhtml">
+  <g id="testmeta">
+    <title>'mix-blend-mode' with 'isolation'</title>
+    <html:link rel="author"
+          title="Tavmjong Bah"
+          href="http://tavmjong.free.fr"/>
+  </g>
+
+  <g id="test-body-content">
+    <g>
+      <rect x="120" y="80"  width="160" height="160" fill="red"/>
+      <rect x="200" y="80"  width="160" height="160" fill="lime"/>
+      <rect x="160" y="160" width="160" height="160" fill="blue"/>
+      <rect x="200" y="160" width="80"  height="80"  fill="#ffffff"/>
+      <rect x="200" y="80"  width="80"  height="80"  fill="#ffff00"/>
+      <rect x="160" y="160" width="40"  height="80"  fill="#ff00ff"/>
+      <rect x="280" y="160" width="40"  height="80"  fill="#00ffff"/>
+    </g>
+
+    <!-- Stroke to prevent aliasing from effecting results. -->
+    <g style="fill:none;stroke:black;stroke-width:2px">
+      <rect x="120" y="80"  width="160" height="160"/>
+      <rect x="200" y="80"  width="160" height="160"/>
+      <rect x="160" y="160" width="160" height="160"/>
+    </g>
+  </g>
+
+</svg>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001.svg b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001.svg
new file mode 100644
index 0000000..2278019c
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-001.svg
@@ -0,0 +1,43 @@
+<svg id="svg-root"
+  width="100%" height="100%" viewBox="0 0 480 360"
+  xmlns="http://www.w3.org/2000/svg"
+  xmlns:xlink="http://www.w3.org/1999/xlink"
+  xmlns:html="http://www.w3.org/1999/xhtml">
+  <g id="testmeta">
+    <title>'mix-blend-mode'</title>
+    <html:link rel="author"
+          title="Tavmjong Bah"
+          href="http://tavmjong.free.fr"/>
+    <html:link rel="help"
+          href="https://www.w3.org/TR/SVG2/render.html#PaintersModel"/>
+    <html:link rel="match"  href="blending-001-ref.svg" />
+  </g>
+
+  <style id="test-font" type="text/css">
+    rect {
+      mix-blend-mode: screen;
+    }
+
+    g {
+      isolation: isolate;
+    }
+  </style>
+
+  <g id="test-body-content">
+
+    <g>
+      <rect x="120" y="80"  width="160" height="160" fill="red"/>
+      <rect x="200" y="80"  width="160" height="160" fill="lime"/>
+      <rect x="160" y="160" width="160" height="160" fill="blue"/>
+    </g>
+
+    <!-- Stroke to prevent aliasing from effecting results. -->
+    <g style="mix-blend-mode:normal;fill:none;stroke:black;stroke-width:2px">
+      <rect x="120" y="80"  width="160" height="160"/>
+      <rect x="200" y="80"  width="160" height="160"/>
+      <rect x="160" y="160" width="160" height="160"/>
+    </g>
+
+  </g>
+
+</svg>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002-ref.svg b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002-ref.svg
new file mode 100644
index 0000000..df6f29f
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002-ref.svg
@@ -0,0 +1,32 @@
+<svg id="svg-root"
+  width="100%" height="100%" viewBox="0 0 480 360"
+  xmlns="http://www.w3.org/2000/svg"
+  xmlns:xlink="http://www.w3.org/1999/xlink"
+  xmlns:html="http://www.w3.org/1999/xhtml">
+  <g id="testmeta">
+    <title>'mix-blend-mode' with 'isolation'</title>
+    <html:link rel="author"
+          title="Tavmjong Bah"
+          href="http://tavmjong.free.fr"/>
+  </g>
+
+  <g id="test-body-content">
+    <g>
+      <rect x="120" y="80"  width="160" height="160" fill="red"/>
+      <rect x="200" y="80"  width="160" height="160" fill="lime"/>
+      <rect x="160" y="160" width="160" height="160" fill="blue"/>
+      <rect x="200" y="160" width="80"  height="80"  fill="#00ffff"/>
+      <rect x="200" y="80"  width="80"  height="80"  fill="#00ff00"/>
+      <rect x="160" y="160" width="40"  height="80"  fill="#0000ff"/>
+      <rect x="280" y="160" width="40"  height="80"  fill="#00ffff"/>
+    </g>
+
+    <!-- Stroke to prevent aliasing from effecting results. -->
+    <g style="fill:none;stroke:black;stroke-width:2px">
+      <rect x="120" y="80"  width="160" height="160"/>
+      <rect x="200" y="80"  width="160" height="160"/>
+      <rect x="160" y="160" width="160" height="160"/>
+    </g>
+  </g>
+
+</svg>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002.svg b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002.svg
new file mode 100644
index 0000000..01e180f
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/svg/render/reftests/blending-002.svg
@@ -0,0 +1,43 @@
+<svg id="svg-root"
+  width="100%" height="100%" viewBox="0 0 480 360"
+  xmlns="http://www.w3.org/2000/svg"
+  xmlns:xlink="http://www.w3.org/1999/xlink"
+  xmlns:html="http://www.w3.org/1999/xhtml">
+  <g id="testmeta">
+    <title>'mix-blend-mode' with 'isolation'</title>
+    <html:link rel="author"
+          title="Tavmjong Bah"
+          href="http://tavmjong.free.fr"/>
+    <html:link rel="help"
+          href="https://www.w3.org/TR/SVG2/render.html#PaintersModel"/>
+    <html:link rel="match"  href="blending-002-ref.svg" />
+  </g>
+
+  <style id="test-font" type="text/css">
+    rect {
+      mix-blend-mode: screen;
+    }
+
+    g {
+      isolation: isolate;
+    }
+  </style>
+
+  <g id="test-body-content">
+
+    <rect x="120" y="80"  width="160" height="160" fill="red"/>
+    <g>
+      <rect x="200" y="80"  width="160" height="160" fill="lime"/>
+      <rect x="160" y="160" width="160" height="160" fill="blue"/>
+    </g>
+
+    <!-- Stroke to prevent aliasing from effecting results. -->
+    <g style="mix-blend-mode:normal;fill:none;stroke:black;stroke-width:2px">
+      <rect x="120" y="80"  width="160" height="160"/>
+      <rect x="200" y="80"  width="160" height="160"/>
+      <rect x="160" y="160" width="160" height="160"/>
+    </g>
+
+  </g>
+
+</svg>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/touch-events/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/touch-events/idlharness.window.js
index 6a95892..dbea07a 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/touch-events/idlharness.window.js
+++ b/third_party/WebKit/LayoutTests/external/wpt/touch-events/idlharness.window.js
@@ -5,21 +5,16 @@
 
 'use strict';
 
-promise_test(async () => {
-  const srcs = ['touch-events', 'uievents', 'dom', 'html'];
-  const [idl, uievents, dom, html] = await Promise.all(
-      srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
-
-  const idl_array = new IdlArray();
-  idl_array.add_idls(idl);
-  idl_array.add_dependency_idls(uievents);
-  idl_array.add_dependency_idls(dom);
-  idl_array.add_dependency_idls(html);
-  idl_array.add_objects({
-    Document: ['document'],
-    GlobalEventHandlers: ['window', 'document', 'document.body'],
-    Touch: ['new Touch({identifier: 1, target: document})'],
-    TouchEvent: ['new TouchEvent("name")'],
-  });
-  idl_array.test();
-}, 'Test IDL implementation of touch-events API');
+idl_test(
+  ['touch-events'],
+  ['uievents', 'dom', 'html'],
+  idl_array => {
+    idl_array.add_objects({
+      Document: ['document'],
+      GlobalEventHandlers: ['window', 'document', 'document.body'],
+      Touch: ['new Touch({identifier: 1, target: document})'],
+      TouchEvent: ['new TouchEvent("name")'],
+    });
+  },
+  'Test IDL implementation of touch-events API'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.html b/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.html
deleted file mode 100644
index f68e51e5e..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.html
+++ /dev/null
@@ -1,32 +0,0 @@
-<!DOCTYPE html>
-<html>
-  <head>
-    <meta charset="utf-8">
-    <title>WebShare IDL test</title>
-    <link rel="help" href="https://wicg.github.io/web-share/">
-    <script src=/resources/testharness.js></script>
-    <script src=/resources/testharnessreport.js></script>
-    <script src=/resources/WebIDLParser.js></script>
-    <script src=/resources/idlharness.js></script>
-  </head>
-  <body>
-    <script>
-        "use strict";
-        var idl_array = new IdlArray();
-
-        function doTest(idl) {
-          idl_array.add_untested_idls('interface Navigator {};');
-          idl_array.add_idls(idl);
-          idl_array.add_objects({
-            Navigator: ['navigator']
-          });
-          idl_array.test();
-        }
-
-        promise_test(async () => {
-          const response = await fetch('../interfaces/web-share.idl');
-          doTest(await response.text());
-        }, 'Test driver');
-    </script>
-  </body>
-</html>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.window.js b/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.window.js
new file mode 100644
index 0000000..6a69c8d
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/web-share/idlharness.https.window.js
@@ -0,0 +1,17 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+// https://wicg.github.io/web-share/
+
+'use strict';
+
+idl_test(
+  ['web-share'],
+  ['html'],
+  idl_array => {
+    idl_array.add_objects({
+      Navigator: ['navigator']
+    });
+  },
+  'Test driver'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webaudio/historical.html b/third_party/WebKit/LayoutTests/external/wpt/webaudio/historical.html
index 2407a87..1f3146c 100644
--- a/third_party/WebKit/LayoutTests/external/wpt/webaudio/historical.html
+++ b/third_party/WebKit/LayoutTests/external/wpt/webaudio/historical.html
@@ -12,4 +12,18 @@
     assert_false(name in window);
   }, name + " interface should not exist");
 });
+
+[
+  "dopplerFactor",
+  "speedOfSound",
+  "setVelocity"
+].forEach(name => {
+  test(function() {
+    assert_false(name in AudioListener.prototype);
+  }, name + " member should not exist on the AudioListener.");
+});
+
+test(function() {
+  assert_false("setVelocity" in PannerNode.prototype);
+}, "setVelocity should not exist on PannerNodes.");
 </script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.html b/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.html
deleted file mode 100644
index bd1df62..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.html
+++ /dev/null
@@ -1,97 +0,0 @@
-<!DOCTYPE html>
-<meta charset="utf-8">
-<title>WebAudio IDL tests</title>
-<link rel="help" href="https://webaudio.github.io/web-audio-api/"/>
-<meta name="timeout" content="long"><!-- Heavy computation of the idl_objects -->
-<script src="/resources/testharness.js"></script>
-<script src="/resources/testharnessreport.js"></script>
-<script src="/resources/WebIDLParser.js"></script>
-<script src="/resources/idlharness.js"></script>
-<script>
-'use strict';
-
-let sample_rate, context, buffer, worklet_node;
-
-promise_test(async t => {
-  const srcs = ['cssom', 'dom', 'html', 'uievents', 'mediacapture-streams', 'webaudio'];
-  const [cssom, dom, html, uievents, mediacapture, webaudio] =
-      await Promise.all(
-          srcs.map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
-
-  const idl_array = new IdlArray();
-  idl_array.add_idls(webaudio);
-
-  // Dependencies of HTML
-  idl_array.add_dependency_idls(html);
-  idl_array.add_dependency_idls(uievents);
-  idl_array.add_dependency_idls(dom);
-  idl_array.add_dependency_idls(mediacapture);
-  idl_array.add_dependency_idls(cssom);
-
-  idl_array.add_untested_idls('interface SVGElement {};');
-  idl_array.add_untested_idls('interface WorkletGlobalScope {};');
-  idl_array.add_untested_idls('interface Worklet {};');
-
-  try {
-    sample_rate = 44100;
-    context = new AudioContext;
-    buffer = new AudioBuffer({length: 1, sampleRate: sample_rate});
-    await context.audioWorklet.addModule(
-      'the-audio-api/the-audioworklet-interface/processors/dummy-processor.js');
-    worklet_node = new AudioWorkletNode(context, 'dummy');
-  } catch (e) {
-    // Ignore - errors will surface in each test_object below.
-  }
-
-  idl_array.add_objects({
-    BaseAudioContext: [],
-    AudioContext: ['context'],
-    OfflineAudioContext: ['new OfflineAudioContext(1, 1, sample_rate)'],
-    OfflineAudioCompletionEvent: [
-      'new OfflineAudioCompletionEvent("", {renderedBuffer: buffer})'],
-    AudioBuffer: ['buffer'],
-    AudioNode: [],
-    AudioParam: ['new AudioBufferSourceNode(context).playbackRate'],
-    AudioScheduledSourceNode: [],
-    AnalyserNode: ['new AnalyserNode(context)'],
-    AudioBufferSourceNode: ['new AudioBufferSourceNode(context)'],
-    AudioDestinationNode: ['context.destination'],
-    AudioListener: ['context.listener'],
-    AudioProcessingEvent: [`new AudioProcessingEvent('', {
-      playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
-    })`],
-    BiquadFilterNode: ['new BiquadFilterNode(context)'],
-    ChannelMergerNode: ['new ChannelMergerNode(context)'],
-    ChannelSplitterNode: ['new ChannelSplitterNode(context)'],
-    ConstantSourceNode: ['new ConstantSourceNode(context)'],
-    ConvolverNode: ['new ConvolverNode(context)'],
-    DelayNode: ['new DelayNode(context)'],
-    DynamicsCompressorNode: ['new DynamicsCompressorNode(context)'],
-    GainNode: ['new GainNode(context)'],
-    IIRFilterNode: [
-      'new IIRFilterNode(context, {feedforward: [1], feedback: [1]})'
-    ],
-    MediaElementAudioSourceNode: [
-      'new MediaElementAudioSourceNode(context, {mediaElement: new Audio})'
-    ],
-    MediaStreamAudioDestinationNode: [
-      'new MediaStreamAudioDestinationNode(context)'
-    ],
-    MediaStreamAudioSourceNode: [],
-    MediaStreamTrackAudioSourceNode: [],
-    OscillatorNode: ['new OscillatorNode(context)'],
-    PannerNode: ['new PannerNode(context)'],
-    PeriodicWave: ['new PeriodicWave(context)'],
-    ScriptProcessorNode: ['context.createScriptProcessor()'],
-    StereoPannerNode: ['new StereoPannerNode(context)'],
-    WaveShaperNode: ['new WaveShaperNode(context)'],
-    AudioWorklet: ['context.audioWorklet'],
-    AudioWorkletGlobalScope: [],
-    AudioParamMap: ['worklet_node.parameters'],
-    AudioWorkletNode: ['worklet_node'],
-    AudioWorkletProcessor: [],
-  });
-  idl_array.test();
-
-}, 'Test driver');
-</script>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window-expected.txt b/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window-expected.txt
new file mode 100644
index 0000000..5bdb954
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window-expected.txt
@@ -0,0 +1,1285 @@
+This is a testharness.js-based test.
+PASS Test driver
+PASS BaseAudioContext interface: existence and properties of interface object
+PASS BaseAudioContext interface object length
+PASS BaseAudioContext interface object name
+PASS BaseAudioContext interface: existence and properties of interface prototype object
+PASS BaseAudioContext interface: existence and properties of interface prototype object's "constructor" property
+PASS BaseAudioContext interface: existence and properties of interface prototype object's @@unscopables property
+PASS BaseAudioContext interface: attribute destination
+PASS Unscopable handled correctly for destination property on BaseAudioContext
+PASS BaseAudioContext interface: attribute sampleRate
+PASS Unscopable handled correctly for sampleRate property on BaseAudioContext
+PASS BaseAudioContext interface: attribute currentTime
+PASS Unscopable handled correctly for currentTime property on BaseAudioContext
+PASS BaseAudioContext interface: attribute listener
+PASS Unscopable handled correctly for listener property on BaseAudioContext
+PASS BaseAudioContext interface: attribute state
+PASS Unscopable handled correctly for state property on BaseAudioContext
+PASS BaseAudioContext interface: attribute audioWorklet
+PASS Unscopable handled correctly for audioWorklet property on BaseAudioContext
+PASS BaseAudioContext interface: attribute onstatechange
+PASS Unscopable handled correctly for onstatechange property on BaseAudioContext
+PASS BaseAudioContext interface: operation createAnalyser()
+PASS Unscopable handled correctly for createAnalyser() on BaseAudioContext
+PASS BaseAudioContext interface: operation createBiquadFilter()
+PASS Unscopable handled correctly for createBiquadFilter() on BaseAudioContext
+PASS BaseAudioContext interface: operation createBuffer(unsigned long, unsigned long, float)
+PASS Unscopable handled correctly for createBuffer(unsigned long, unsigned long, float) on BaseAudioContext
+PASS BaseAudioContext interface: operation createBufferSource()
+PASS Unscopable handled correctly for createBufferSource() on BaseAudioContext
+PASS BaseAudioContext interface: operation createChannelMerger(unsigned long)
+PASS Unscopable handled correctly for createChannelMerger(unsigned long) on BaseAudioContext
+PASS BaseAudioContext interface: operation createChannelSplitter(unsigned long)
+PASS Unscopable handled correctly for createChannelSplitter(unsigned long) on BaseAudioContext
+PASS BaseAudioContext interface: operation createConstantSource()
+PASS Unscopable handled correctly for createConstantSource() on BaseAudioContext
+PASS BaseAudioContext interface: operation createConvolver()
+PASS Unscopable handled correctly for createConvolver() on BaseAudioContext
+PASS BaseAudioContext interface: operation createDelay(double)
+PASS Unscopable handled correctly for createDelay(double) on BaseAudioContext
+PASS BaseAudioContext interface: operation createDynamicsCompressor()
+PASS Unscopable handled correctly for createDynamicsCompressor() on BaseAudioContext
+PASS BaseAudioContext interface: operation createGain()
+PASS Unscopable handled correctly for createGain() on BaseAudioContext
+PASS BaseAudioContext interface: operation createIIRFilter([object Object], [object Object])
+PASS Unscopable handled correctly for createIIRFilter([object Object], [object Object]) on BaseAudioContext
+PASS BaseAudioContext interface: operation createOscillator()
+PASS Unscopable handled correctly for createOscillator() on BaseAudioContext
+PASS BaseAudioContext interface: operation createPanner()
+PASS Unscopable handled correctly for createPanner() on BaseAudioContext
+PASS BaseAudioContext interface: operation createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints)
+PASS Unscopable handled correctly for createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints) on BaseAudioContext
+PASS BaseAudioContext interface: operation createScriptProcessor(unsigned long, unsigned long, unsigned long)
+PASS Unscopable handled correctly for createScriptProcessor(unsigned long, unsigned long, unsigned long) on BaseAudioContext
+PASS BaseAudioContext interface: operation createStereoPanner()
+PASS Unscopable handled correctly for createStereoPanner() on BaseAudioContext
+PASS BaseAudioContext interface: operation createWaveShaper()
+PASS Unscopable handled correctly for createWaveShaper() on BaseAudioContext
+PASS BaseAudioContext interface: operation decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback)
+PASS Unscopable handled correctly for decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback) on BaseAudioContext
+PASS BaseAudioContext interface: operation resume()
+PASS Unscopable handled correctly for resume() on BaseAudioContext
+PASS AudioContext interface: existence and properties of interface object
+PASS AudioContext interface object length
+PASS AudioContext interface object name
+PASS AudioContext interface: existence and properties of interface prototype object
+PASS AudioContext interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioContext interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioContext interface: attribute baseLatency
+PASS Unscopable handled correctly for baseLatency property on AudioContext
+FAIL AudioContext interface: attribute outputLatency assert_true: The prototype object must have a property "outputLatency" expected true got false
+PASS Unscopable handled correctly for outputLatency property on AudioContext
+PASS AudioContext interface: operation getOutputTimestamp()
+PASS Unscopable handled correctly for getOutputTimestamp() on AudioContext
+PASS AudioContext interface: operation suspend()
+PASS Unscopable handled correctly for suspend() on AudioContext
+PASS AudioContext interface: operation close()
+PASS Unscopable handled correctly for close() on AudioContext
+FAIL AudioContext interface: operation createMediaElementSource(HTMLMediaElement) assert_own_property: interface prototype object missing non-static operation expected property "createMediaElementSource" missing
+PASS Unscopable handled correctly for createMediaElementSource(HTMLMediaElement) on AudioContext
+FAIL AudioContext interface: operation createMediaStreamSource(MediaStream) assert_own_property: interface prototype object missing non-static operation expected property "createMediaStreamSource" missing
+PASS Unscopable handled correctly for createMediaStreamSource(MediaStream) on AudioContext
+FAIL AudioContext interface: operation createMediaStreamTrackSource(MediaStreamTrack) assert_own_property: interface prototype object missing non-static operation expected property "createMediaStreamTrackSource" missing
+PASS Unscopable handled correctly for createMediaStreamTrackSource(MediaStreamTrack) on AudioContext
+FAIL AudioContext interface: operation createMediaStreamDestination() assert_own_property: interface prototype object missing non-static operation expected property "createMediaStreamDestination" missing
+PASS Unscopable handled correctly for createMediaStreamDestination() on AudioContext
+PASS AudioContext must be primary interface of context
+PASS Stringification of context
+PASS AudioContext interface: context must inherit property "baseLatency" with the proper type
+FAIL AudioContext interface: context must inherit property "outputLatency" with the proper type assert_inherits: property "outputLatency" not found in prototype chain
+PASS AudioContext interface: context must inherit property "getOutputTimestamp()" with the proper type
+PASS AudioContext interface: context must inherit property "suspend()" with the proper type
+PASS AudioContext interface: context must inherit property "close()" with the proper type
+PASS AudioContext interface: context must inherit property "createMediaElementSource(HTMLMediaElement)" with the proper type
+PASS AudioContext interface: calling createMediaElementSource(HTMLMediaElement) on context with too few arguments must throw TypeError
+PASS AudioContext interface: context must inherit property "createMediaStreamSource(MediaStream)" with the proper type
+PASS AudioContext interface: calling createMediaStreamSource(MediaStream) on context with too few arguments must throw TypeError
+FAIL AudioContext interface: context must inherit property "createMediaStreamTrackSource(MediaStreamTrack)" with the proper type assert_inherits: property "createMediaStreamTrackSource" not found in prototype chain
+FAIL AudioContext interface: calling createMediaStreamTrackSource(MediaStreamTrack) on context with too few arguments must throw TypeError assert_inherits: property "createMediaStreamTrackSource" not found in prototype chain
+PASS AudioContext interface: context must inherit property "createMediaStreamDestination()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "destination" with the proper type
+PASS BaseAudioContext interface: context must inherit property "sampleRate" with the proper type
+PASS BaseAudioContext interface: context must inherit property "currentTime" with the proper type
+PASS BaseAudioContext interface: context must inherit property "listener" with the proper type
+PASS BaseAudioContext interface: context must inherit property "state" with the proper type
+PASS BaseAudioContext interface: context must inherit property "audioWorklet" with the proper type
+PASS BaseAudioContext interface: context must inherit property "onstatechange" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createAnalyser()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createBiquadFilter()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createBuffer(unsigned long, unsigned long, float)" with the proper type
+PASS BaseAudioContext interface: calling createBuffer(unsigned long, unsigned long, float) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createBufferSource()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createChannelMerger(unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createChannelMerger(unsigned long) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createChannelSplitter(unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createChannelSplitter(unsigned long) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createConstantSource()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createConvolver()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createDelay(double)" with the proper type
+PASS BaseAudioContext interface: calling createDelay(double) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createDynamicsCompressor()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createGain()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createIIRFilter([object Object], [object Object])" with the proper type
+PASS BaseAudioContext interface: calling createIIRFilter([object Object], [object Object]) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createOscillator()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createPanner()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints)" with the proper type
+PASS BaseAudioContext interface: calling createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createScriptProcessor(unsigned long, unsigned long, unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createScriptProcessor(unsigned long, unsigned long, unsigned long) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "createStereoPanner()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "createWaveShaper()" with the proper type
+PASS BaseAudioContext interface: context must inherit property "decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback)" with the proper type
+PASS BaseAudioContext interface: calling decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback) on context with too few arguments must throw TypeError
+PASS BaseAudioContext interface: context must inherit property "resume()" with the proper type
+PASS OfflineAudioContext interface: existence and properties of interface object
+PASS OfflineAudioContext interface object length
+PASS OfflineAudioContext interface object name
+PASS OfflineAudioContext interface: existence and properties of interface prototype object
+PASS OfflineAudioContext interface: existence and properties of interface prototype object's "constructor" property
+PASS OfflineAudioContext interface: existence and properties of interface prototype object's @@unscopables property
+PASS OfflineAudioContext interface: operation startRendering()
+PASS Unscopable handled correctly for startRendering() on OfflineAudioContext
+PASS OfflineAudioContext interface: operation suspend(double)
+PASS Unscopable handled correctly for suspend(double) on OfflineAudioContext
+PASS OfflineAudioContext interface: attribute length
+PASS Unscopable handled correctly for length property on OfflineAudioContext
+PASS OfflineAudioContext interface: attribute oncomplete
+PASS Unscopable handled correctly for oncomplete property on OfflineAudioContext
+PASS OfflineAudioContext must be primary interface of new OfflineAudioContext(1, 1, sample_rate)
+PASS Stringification of new OfflineAudioContext(1, 1, sample_rate)
+PASS OfflineAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "startRendering()" with the proper type
+PASS OfflineAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "suspend(double)" with the proper type
+PASS OfflineAudioContext interface: calling suspend(double) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS OfflineAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "length" with the proper type
+PASS OfflineAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "oncomplete" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "destination" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "sampleRate" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "currentTime" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "listener" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "state" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "audioWorklet" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "onstatechange" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createAnalyser()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createBiquadFilter()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createBuffer(unsigned long, unsigned long, float)" with the proper type
+PASS BaseAudioContext interface: calling createBuffer(unsigned long, unsigned long, float) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createBufferSource()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createChannelMerger(unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createChannelMerger(unsigned long) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createChannelSplitter(unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createChannelSplitter(unsigned long) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createConstantSource()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createConvolver()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createDelay(double)" with the proper type
+PASS BaseAudioContext interface: calling createDelay(double) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createDynamicsCompressor()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createGain()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createIIRFilter([object Object], [object Object])" with the proper type
+PASS BaseAudioContext interface: calling createIIRFilter([object Object], [object Object]) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createOscillator()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createPanner()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints)" with the proper type
+PASS BaseAudioContext interface: calling createPeriodicWave([object Object], [object Object], PeriodicWaveConstraints) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createScriptProcessor(unsigned long, unsigned long, unsigned long)" with the proper type
+PASS BaseAudioContext interface: calling createScriptProcessor(unsigned long, unsigned long, unsigned long) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createStereoPanner()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "createWaveShaper()" with the proper type
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback)" with the proper type
+PASS BaseAudioContext interface: calling decodeAudioData(ArrayBuffer, DecodeSuccessCallback, DecodeErrorCallback) on new OfflineAudioContext(1, 1, sample_rate) with too few arguments must throw TypeError
+PASS BaseAudioContext interface: new OfflineAudioContext(1, 1, sample_rate) must inherit property "resume()" with the proper type
+PASS OfflineAudioCompletionEvent interface: existence and properties of interface object
+PASS OfflineAudioCompletionEvent interface object length
+PASS OfflineAudioCompletionEvent interface object name
+PASS OfflineAudioCompletionEvent interface: existence and properties of interface prototype object
+PASS OfflineAudioCompletionEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS OfflineAudioCompletionEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS OfflineAudioCompletionEvent interface: attribute renderedBuffer
+PASS Unscopable handled correctly for renderedBuffer property on OfflineAudioCompletionEvent
+PASS OfflineAudioCompletionEvent must be primary interface of new OfflineAudioCompletionEvent("", {renderedBuffer: buffer})
+PASS Stringification of new OfflineAudioCompletionEvent("", {renderedBuffer: buffer})
+PASS OfflineAudioCompletionEvent interface: new OfflineAudioCompletionEvent("", {renderedBuffer: buffer}) must inherit property "renderedBuffer" with the proper type
+PASS AudioBuffer interface: existence and properties of interface object
+PASS AudioBuffer interface object length
+PASS AudioBuffer interface object name
+PASS AudioBuffer interface: existence and properties of interface prototype object
+PASS AudioBuffer interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioBuffer interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioBuffer interface: attribute sampleRate
+PASS Unscopable handled correctly for sampleRate property on AudioBuffer
+PASS AudioBuffer interface: attribute length
+PASS Unscopable handled correctly for length property on AudioBuffer
+PASS AudioBuffer interface: attribute duration
+PASS Unscopable handled correctly for duration property on AudioBuffer
+PASS AudioBuffer interface: attribute numberOfChannels
+PASS Unscopable handled correctly for numberOfChannels property on AudioBuffer
+PASS AudioBuffer interface: operation getChannelData(unsigned long)
+PASS Unscopable handled correctly for getChannelData(unsigned long) on AudioBuffer
+PASS AudioBuffer interface: operation copyFromChannel(Float32Array, unsigned long, unsigned long)
+PASS Unscopable handled correctly for copyFromChannel(Float32Array, unsigned long, unsigned long) on AudioBuffer
+PASS AudioBuffer interface: operation copyToChannel(Float32Array, unsigned long, unsigned long)
+PASS Unscopable handled correctly for copyToChannel(Float32Array, unsigned long, unsigned long) on AudioBuffer
+PASS AudioBuffer must be primary interface of buffer
+PASS Stringification of buffer
+PASS AudioBuffer interface: buffer must inherit property "sampleRate" with the proper type
+PASS AudioBuffer interface: buffer must inherit property "length" with the proper type
+PASS AudioBuffer interface: buffer must inherit property "duration" with the proper type
+PASS AudioBuffer interface: buffer must inherit property "numberOfChannels" with the proper type
+PASS AudioBuffer interface: buffer must inherit property "getChannelData(unsigned long)" with the proper type
+PASS AudioBuffer interface: calling getChannelData(unsigned long) on buffer with too few arguments must throw TypeError
+PASS AudioBuffer interface: buffer must inherit property "copyFromChannel(Float32Array, unsigned long, unsigned long)" with the proper type
+PASS AudioBuffer interface: calling copyFromChannel(Float32Array, unsigned long, unsigned long) on buffer with too few arguments must throw TypeError
+PASS AudioBuffer interface: buffer must inherit property "copyToChannel(Float32Array, unsigned long, unsigned long)" with the proper type
+PASS AudioBuffer interface: calling copyToChannel(Float32Array, unsigned long, unsigned long) on buffer with too few arguments must throw TypeError
+PASS AudioNode interface: existence and properties of interface object
+PASS AudioNode interface object length
+PASS AudioNode interface object name
+PASS AudioNode interface: existence and properties of interface prototype object
+PASS AudioNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioNode interface: operation connect(AudioNode, unsigned long, unsigned long)
+PASS Unscopable handled correctly for connect(AudioNode, unsigned long, unsigned long) on AudioNode
+PASS AudioNode interface: operation connect(AudioParam, unsigned long)
+PASS Unscopable handled correctly for connect(AudioParam, unsigned long) on AudioNode
+PASS AudioNode interface: operation disconnect()
+PASS Unscopable handled correctly for disconnect() on AudioNode
+PASS AudioNode interface: operation disconnect(unsigned long)
+PASS Unscopable handled correctly for disconnect(unsigned long) on AudioNode
+PASS AudioNode interface: operation disconnect(AudioNode)
+PASS Unscopable handled correctly for disconnect(AudioNode) on AudioNode
+PASS AudioNode interface: operation disconnect(AudioNode, unsigned long)
+PASS Unscopable handled correctly for disconnect(AudioNode, unsigned long) on AudioNode
+PASS AudioNode interface: operation disconnect(AudioNode, unsigned long, unsigned long)
+PASS Unscopable handled correctly for disconnect(AudioNode, unsigned long, unsigned long) on AudioNode
+PASS AudioNode interface: operation disconnect(AudioParam)
+PASS Unscopable handled correctly for disconnect(AudioParam) on AudioNode
+PASS AudioNode interface: operation disconnect(AudioParam, unsigned long)
+PASS Unscopable handled correctly for disconnect(AudioParam, unsigned long) on AudioNode
+PASS AudioNode interface: attribute context
+PASS Unscopable handled correctly for context property on AudioNode
+PASS AudioNode interface: attribute numberOfInputs
+PASS Unscopable handled correctly for numberOfInputs property on AudioNode
+PASS AudioNode interface: attribute numberOfOutputs
+PASS Unscopable handled correctly for numberOfOutputs property on AudioNode
+PASS AudioNode interface: attribute channelCount
+PASS Unscopable handled correctly for channelCount property on AudioNode
+PASS AudioNode interface: attribute channelCountMode
+PASS Unscopable handled correctly for channelCountMode property on AudioNode
+PASS AudioNode interface: attribute channelInterpretation
+PASS Unscopable handled correctly for channelInterpretation property on AudioNode
+PASS AudioParam interface: existence and properties of interface object
+PASS AudioParam interface object length
+PASS AudioParam interface object name
+PASS AudioParam interface: existence and properties of interface prototype object
+PASS AudioParam interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioParam interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioParam interface: attribute value
+PASS Unscopable handled correctly for value property on AudioParam
+PASS AudioParam interface: attribute automationRate
+PASS Unscopable handled correctly for automationRate property on AudioParam
+PASS AudioParam interface: attribute defaultValue
+PASS Unscopable handled correctly for defaultValue property on AudioParam
+PASS AudioParam interface: attribute minValue
+PASS Unscopable handled correctly for minValue property on AudioParam
+PASS AudioParam interface: attribute maxValue
+PASS Unscopable handled correctly for maxValue property on AudioParam
+PASS AudioParam interface: operation setValueAtTime(float, double)
+PASS Unscopable handled correctly for setValueAtTime(float, double) on AudioParam
+PASS AudioParam interface: operation linearRampToValueAtTime(float, double)
+PASS Unscopable handled correctly for linearRampToValueAtTime(float, double) on AudioParam
+PASS AudioParam interface: operation exponentialRampToValueAtTime(float, double)
+PASS Unscopable handled correctly for exponentialRampToValueAtTime(float, double) on AudioParam
+PASS AudioParam interface: operation setTargetAtTime(float, double, float)
+PASS Unscopable handled correctly for setTargetAtTime(float, double, float) on AudioParam
+PASS AudioParam interface: operation setValueCurveAtTime([object Object], double, double)
+PASS Unscopable handled correctly for setValueCurveAtTime([object Object], double, double) on AudioParam
+PASS AudioParam interface: operation cancelScheduledValues(double)
+PASS Unscopable handled correctly for cancelScheduledValues(double) on AudioParam
+PASS AudioParam interface: operation cancelAndHoldAtTime(double)
+PASS Unscopable handled correctly for cancelAndHoldAtTime(double) on AudioParam
+PASS AudioParam must be primary interface of new AudioBufferSourceNode(context).playbackRate
+PASS Stringification of new AudioBufferSourceNode(context).playbackRate
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "value" with the proper type
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "automationRate" with the proper type
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "defaultValue" with the proper type
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "minValue" with the proper type
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "maxValue" with the proper type
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "setValueAtTime(float, double)" with the proper type
+PASS AudioParam interface: calling setValueAtTime(float, double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "linearRampToValueAtTime(float, double)" with the proper type
+PASS AudioParam interface: calling linearRampToValueAtTime(float, double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "exponentialRampToValueAtTime(float, double)" with the proper type
+PASS AudioParam interface: calling exponentialRampToValueAtTime(float, double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "setTargetAtTime(float, double, float)" with the proper type
+PASS AudioParam interface: calling setTargetAtTime(float, double, float) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "setValueCurveAtTime([object Object], double, double)" with the proper type
+PASS AudioParam interface: calling setValueCurveAtTime([object Object], double, double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "cancelScheduledValues(double)" with the proper type
+PASS AudioParam interface: calling cancelScheduledValues(double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioParam interface: new AudioBufferSourceNode(context).playbackRate must inherit property "cancelAndHoldAtTime(double)" with the proper type
+PASS AudioParam interface: calling cancelAndHoldAtTime(double) on new AudioBufferSourceNode(context).playbackRate with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: existence and properties of interface object
+PASS AudioScheduledSourceNode interface object length
+PASS AudioScheduledSourceNode interface object name
+PASS AudioScheduledSourceNode interface: existence and properties of interface prototype object
+PASS AudioScheduledSourceNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioScheduledSourceNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioScheduledSourceNode interface: attribute onended
+PASS Unscopable handled correctly for onended property on AudioScheduledSourceNode
+PASS AudioScheduledSourceNode interface: operation start(double)
+PASS Unscopable handled correctly for start(double) on AudioScheduledSourceNode
+PASS AudioScheduledSourceNode interface: operation stop(double)
+PASS Unscopable handled correctly for stop(double) on AudioScheduledSourceNode
+PASS AnalyserNode interface: existence and properties of interface object
+PASS AnalyserNode interface object length
+PASS AnalyserNode interface object name
+PASS AnalyserNode interface: existence and properties of interface prototype object
+PASS AnalyserNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AnalyserNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AnalyserNode interface: operation getFloatFrequencyData(Float32Array)
+PASS Unscopable handled correctly for getFloatFrequencyData(Float32Array) on AnalyserNode
+PASS AnalyserNode interface: operation getByteFrequencyData(Uint8Array)
+PASS Unscopable handled correctly for getByteFrequencyData(Uint8Array) on AnalyserNode
+PASS AnalyserNode interface: operation getFloatTimeDomainData(Float32Array)
+PASS Unscopable handled correctly for getFloatTimeDomainData(Float32Array) on AnalyserNode
+PASS AnalyserNode interface: operation getByteTimeDomainData(Uint8Array)
+PASS Unscopable handled correctly for getByteTimeDomainData(Uint8Array) on AnalyserNode
+PASS AnalyserNode interface: attribute fftSize
+PASS Unscopable handled correctly for fftSize property on AnalyserNode
+PASS AnalyserNode interface: attribute frequencyBinCount
+PASS Unscopable handled correctly for frequencyBinCount property on AnalyserNode
+PASS AnalyserNode interface: attribute minDecibels
+PASS Unscopable handled correctly for minDecibels property on AnalyserNode
+PASS AnalyserNode interface: attribute maxDecibels
+PASS Unscopable handled correctly for maxDecibels property on AnalyserNode
+PASS AnalyserNode interface: attribute smoothingTimeConstant
+PASS Unscopable handled correctly for smoothingTimeConstant property on AnalyserNode
+PASS AnalyserNode must be primary interface of new AnalyserNode(context)
+PASS Stringification of new AnalyserNode(context)
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "getFloatFrequencyData(Float32Array)" with the proper type
+PASS AnalyserNode interface: calling getFloatFrequencyData(Float32Array) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "getByteFrequencyData(Uint8Array)" with the proper type
+PASS AnalyserNode interface: calling getByteFrequencyData(Uint8Array) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "getFloatTimeDomainData(Float32Array)" with the proper type
+PASS AnalyserNode interface: calling getFloatTimeDomainData(Float32Array) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "getByteTimeDomainData(Uint8Array)" with the proper type
+PASS AnalyserNode interface: calling getByteTimeDomainData(Uint8Array) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "fftSize" with the proper type
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "frequencyBinCount" with the proper type
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "minDecibels" with the proper type
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "maxDecibels" with the proper type
+PASS AnalyserNode interface: new AnalyserNode(context) must inherit property "smoothingTimeConstant" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new AnalyserNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new AnalyserNode(context) must inherit property "channelInterpretation" with the proper type
+PASS AudioBufferSourceNode interface: existence and properties of interface object
+PASS AudioBufferSourceNode interface object length
+PASS AudioBufferSourceNode interface object name
+PASS AudioBufferSourceNode interface: existence and properties of interface prototype object
+PASS AudioBufferSourceNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioBufferSourceNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioBufferSourceNode interface: attribute buffer
+PASS Unscopable handled correctly for buffer property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: attribute playbackRate
+PASS Unscopable handled correctly for playbackRate property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: attribute detune
+PASS Unscopable handled correctly for detune property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: attribute loop
+PASS Unscopable handled correctly for loop property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: attribute loopStart
+PASS Unscopable handled correctly for loopStart property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: attribute loopEnd
+PASS Unscopable handled correctly for loopEnd property on AudioBufferSourceNode
+PASS AudioBufferSourceNode interface: operation start(double, double, double)
+PASS Unscopable handled correctly for start(double, double, double) on AudioBufferSourceNode
+PASS AudioBufferSourceNode must be primary interface of new AudioBufferSourceNode(context)
+PASS Stringification of new AudioBufferSourceNode(context)
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "buffer" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "playbackRate" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "detune" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "loop" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "loopStart" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "loopEnd" with the proper type
+PASS AudioBufferSourceNode interface: new AudioBufferSourceNode(context) must inherit property "start(double, double, double)" with the proper type
+PASS AudioBufferSourceNode interface: calling start(double, double, double) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: new AudioBufferSourceNode(context) must inherit property "onended" with the proper type
+PASS AudioScheduledSourceNode interface: new AudioBufferSourceNode(context) must inherit property "start(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling start(double) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: new AudioBufferSourceNode(context) must inherit property "stop(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling stop(double) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new AudioBufferSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new AudioBufferSourceNode(context) must inherit property "channelInterpretation" with the proper type
+PASS AudioDestinationNode interface: existence and properties of interface object
+PASS AudioDestinationNode interface object length
+PASS AudioDestinationNode interface object name
+PASS AudioDestinationNode interface: existence and properties of interface prototype object
+PASS AudioDestinationNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioDestinationNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioDestinationNode interface: attribute maxChannelCount
+PASS Unscopable handled correctly for maxChannelCount property on AudioDestinationNode
+PASS AudioDestinationNode must be primary interface of context.destination
+PASS Stringification of context.destination
+PASS AudioDestinationNode interface: context.destination must inherit property "maxChannelCount" with the proper type
+PASS AudioNode interface: context.destination must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: context.destination must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on context.destination with too few arguments must throw TypeError
+PASS AudioNode interface: context.destination must inherit property "context" with the proper type
+PASS AudioNode interface: context.destination must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: context.destination must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: context.destination must inherit property "channelCount" with the proper type
+PASS AudioNode interface: context.destination must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: context.destination must inherit property "channelInterpretation" with the proper type
+PASS AudioListener interface: existence and properties of interface object
+PASS AudioListener interface object length
+PASS AudioListener interface object name
+PASS AudioListener interface: existence and properties of interface prototype object
+PASS AudioListener interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioListener interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioListener interface: attribute positionX
+PASS Unscopable handled correctly for positionX property on AudioListener
+PASS AudioListener interface: attribute positionY
+PASS Unscopable handled correctly for positionY property on AudioListener
+PASS AudioListener interface: attribute positionZ
+PASS Unscopable handled correctly for positionZ property on AudioListener
+PASS AudioListener interface: attribute forwardX
+PASS Unscopable handled correctly for forwardX property on AudioListener
+PASS AudioListener interface: attribute forwardY
+PASS Unscopable handled correctly for forwardY property on AudioListener
+PASS AudioListener interface: attribute forwardZ
+PASS Unscopable handled correctly for forwardZ property on AudioListener
+PASS AudioListener interface: attribute upX
+PASS Unscopable handled correctly for upX property on AudioListener
+PASS AudioListener interface: attribute upY
+PASS Unscopable handled correctly for upY property on AudioListener
+PASS AudioListener interface: attribute upZ
+PASS Unscopable handled correctly for upZ property on AudioListener
+PASS AudioListener interface: operation setPosition(float, float, float)
+PASS Unscopable handled correctly for setPosition(float, float, float) on AudioListener
+PASS AudioListener interface: operation setOrientation(float, float, float, float, float, float)
+PASS Unscopable handled correctly for setOrientation(float, float, float, float, float, float) on AudioListener
+PASS AudioListener must be primary interface of context.listener
+PASS Stringification of context.listener
+PASS AudioListener interface: context.listener must inherit property "positionX" with the proper type
+PASS AudioListener interface: context.listener must inherit property "positionY" with the proper type
+PASS AudioListener interface: context.listener must inherit property "positionZ" with the proper type
+PASS AudioListener interface: context.listener must inherit property "forwardX" with the proper type
+PASS AudioListener interface: context.listener must inherit property "forwardY" with the proper type
+PASS AudioListener interface: context.listener must inherit property "forwardZ" with the proper type
+PASS AudioListener interface: context.listener must inherit property "upX" with the proper type
+PASS AudioListener interface: context.listener must inherit property "upY" with the proper type
+PASS AudioListener interface: context.listener must inherit property "upZ" with the proper type
+PASS AudioListener interface: context.listener must inherit property "setPosition(float, float, float)" with the proper type
+PASS AudioListener interface: calling setPosition(float, float, float) on context.listener with too few arguments must throw TypeError
+PASS AudioListener interface: context.listener must inherit property "setOrientation(float, float, float, float, float, float)" with the proper type
+PASS AudioListener interface: calling setOrientation(float, float, float, float, float, float) on context.listener with too few arguments must throw TypeError
+PASS AudioProcessingEvent interface: existence and properties of interface object
+PASS AudioProcessingEvent interface object length
+PASS AudioProcessingEvent interface object name
+PASS AudioProcessingEvent interface: existence and properties of interface prototype object
+PASS AudioProcessingEvent interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioProcessingEvent interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioProcessingEvent interface: attribute playbackTime
+PASS Unscopable handled correctly for playbackTime property on AudioProcessingEvent
+PASS AudioProcessingEvent interface: attribute inputBuffer
+PASS Unscopable handled correctly for inputBuffer property on AudioProcessingEvent
+PASS AudioProcessingEvent interface: attribute outputBuffer
+PASS Unscopable handled correctly for outputBuffer property on AudioProcessingEvent
+PASS AudioProcessingEvent must be primary interface of new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      })
+PASS Stringification of new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      })
+PASS AudioProcessingEvent interface: new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      }) must inherit property "playbackTime" with the proper type
+PASS AudioProcessingEvent interface: new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      }) must inherit property "inputBuffer" with the proper type
+PASS AudioProcessingEvent interface: new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      }) must inherit property "outputBuffer" with the proper type
+PASS BiquadFilterNode interface: existence and properties of interface object
+PASS BiquadFilterNode interface object length
+PASS BiquadFilterNode interface object name
+PASS BiquadFilterNode interface: existence and properties of interface prototype object
+PASS BiquadFilterNode interface: existence and properties of interface prototype object's "constructor" property
+PASS BiquadFilterNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS BiquadFilterNode interface: attribute type
+PASS Unscopable handled correctly for type property on BiquadFilterNode
+PASS BiquadFilterNode interface: attribute frequency
+PASS Unscopable handled correctly for frequency property on BiquadFilterNode
+PASS BiquadFilterNode interface: attribute detune
+PASS Unscopable handled correctly for detune property on BiquadFilterNode
+PASS BiquadFilterNode interface: attribute Q
+PASS Unscopable handled correctly for Q property on BiquadFilterNode
+PASS BiquadFilterNode interface: attribute gain
+PASS Unscopable handled correctly for gain property on BiquadFilterNode
+PASS BiquadFilterNode interface: operation getFrequencyResponse(Float32Array, Float32Array, Float32Array)
+PASS Unscopable handled correctly for getFrequencyResponse(Float32Array, Float32Array, Float32Array) on BiquadFilterNode
+PASS BiquadFilterNode must be primary interface of new BiquadFilterNode(context)
+PASS Stringification of new BiquadFilterNode(context)
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "type" with the proper type
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "frequency" with the proper type
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "detune" with the proper type
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "Q" with the proper type
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "gain" with the proper type
+PASS BiquadFilterNode interface: new BiquadFilterNode(context) must inherit property "getFrequencyResponse(Float32Array, Float32Array, Float32Array)" with the proper type
+PASS BiquadFilterNode interface: calling getFrequencyResponse(Float32Array, Float32Array, Float32Array) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new BiquadFilterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new BiquadFilterNode(context) must inherit property "channelInterpretation" with the proper type
+PASS ChannelMergerNode interface: existence and properties of interface object
+PASS ChannelMergerNode interface object length
+PASS ChannelMergerNode interface object name
+PASS ChannelMergerNode interface: existence and properties of interface prototype object
+PASS ChannelMergerNode interface: existence and properties of interface prototype object's "constructor" property
+PASS ChannelMergerNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS ChannelMergerNode must be primary interface of new ChannelMergerNode(context)
+PASS Stringification of new ChannelMergerNode(context)
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new ChannelMergerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new ChannelMergerNode(context) must inherit property "channelInterpretation" with the proper type
+PASS ChannelSplitterNode interface: existence and properties of interface object
+PASS ChannelSplitterNode interface object length
+PASS ChannelSplitterNode interface object name
+PASS ChannelSplitterNode interface: existence and properties of interface prototype object
+PASS ChannelSplitterNode interface: existence and properties of interface prototype object's "constructor" property
+PASS ChannelSplitterNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS ChannelSplitterNode must be primary interface of new ChannelSplitterNode(context)
+PASS Stringification of new ChannelSplitterNode(context)
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new ChannelSplitterNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new ChannelSplitterNode(context) must inherit property "channelInterpretation" with the proper type
+PASS ConstantSourceNode interface: existence and properties of interface object
+PASS ConstantSourceNode interface object length
+PASS ConstantSourceNode interface object name
+PASS ConstantSourceNode interface: existence and properties of interface prototype object
+PASS ConstantSourceNode interface: existence and properties of interface prototype object's "constructor" property
+PASS ConstantSourceNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS ConstantSourceNode interface: attribute offset
+PASS Unscopable handled correctly for offset property on ConstantSourceNode
+PASS ConstantSourceNode must be primary interface of new ConstantSourceNode(context)
+PASS Stringification of new ConstantSourceNode(context)
+PASS ConstantSourceNode interface: new ConstantSourceNode(context) must inherit property "offset" with the proper type
+PASS AudioScheduledSourceNode interface: new ConstantSourceNode(context) must inherit property "onended" with the proper type
+PASS AudioScheduledSourceNode interface: new ConstantSourceNode(context) must inherit property "start(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling start(double) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: new ConstantSourceNode(context) must inherit property "stop(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling stop(double) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new ConstantSourceNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new ConstantSourceNode(context) must inherit property "channelInterpretation" with the proper type
+PASS ConvolverNode interface: existence and properties of interface object
+PASS ConvolverNode interface object length
+PASS ConvolverNode interface object name
+PASS ConvolverNode interface: existence and properties of interface prototype object
+PASS ConvolverNode interface: existence and properties of interface prototype object's "constructor" property
+PASS ConvolverNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS ConvolverNode interface: attribute buffer
+PASS Unscopable handled correctly for buffer property on ConvolverNode
+PASS ConvolverNode interface: attribute normalize
+PASS Unscopable handled correctly for normalize property on ConvolverNode
+PASS ConvolverNode must be primary interface of new ConvolverNode(context)
+PASS Stringification of new ConvolverNode(context)
+PASS ConvolverNode interface: new ConvolverNode(context) must inherit property "buffer" with the proper type
+PASS ConvolverNode interface: new ConvolverNode(context) must inherit property "normalize" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new ConvolverNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new ConvolverNode(context) must inherit property "channelInterpretation" with the proper type
+PASS DelayNode interface: existence and properties of interface object
+PASS DelayNode interface object length
+PASS DelayNode interface object name
+PASS DelayNode interface: existence and properties of interface prototype object
+PASS DelayNode interface: existence and properties of interface prototype object's "constructor" property
+PASS DelayNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS DelayNode interface: attribute delayTime
+PASS Unscopable handled correctly for delayTime property on DelayNode
+PASS DelayNode must be primary interface of new DelayNode(context)
+PASS Stringification of new DelayNode(context)
+PASS DelayNode interface: new DelayNode(context) must inherit property "delayTime" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new DelayNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DelayNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new DelayNode(context) must inherit property "channelInterpretation" with the proper type
+PASS DynamicsCompressorNode interface: existence and properties of interface object
+PASS DynamicsCompressorNode interface object length
+PASS DynamicsCompressorNode interface object name
+PASS DynamicsCompressorNode interface: existence and properties of interface prototype object
+PASS DynamicsCompressorNode interface: existence and properties of interface prototype object's "constructor" property
+PASS DynamicsCompressorNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS DynamicsCompressorNode interface: attribute threshold
+PASS Unscopable handled correctly for threshold property on DynamicsCompressorNode
+PASS DynamicsCompressorNode interface: attribute knee
+PASS Unscopable handled correctly for knee property on DynamicsCompressorNode
+PASS DynamicsCompressorNode interface: attribute ratio
+PASS Unscopable handled correctly for ratio property on DynamicsCompressorNode
+PASS DynamicsCompressorNode interface: attribute reduction
+PASS Unscopable handled correctly for reduction property on DynamicsCompressorNode
+PASS DynamicsCompressorNode interface: attribute attack
+PASS Unscopable handled correctly for attack property on DynamicsCompressorNode
+PASS DynamicsCompressorNode interface: attribute release
+PASS Unscopable handled correctly for release property on DynamicsCompressorNode
+PASS DynamicsCompressorNode must be primary interface of new DynamicsCompressorNode(context)
+PASS Stringification of new DynamicsCompressorNode(context)
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "threshold" with the proper type
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "knee" with the proper type
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "ratio" with the proper type
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "reduction" with the proper type
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "attack" with the proper type
+PASS DynamicsCompressorNode interface: new DynamicsCompressorNode(context) must inherit property "release" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new DynamicsCompressorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new DynamicsCompressorNode(context) must inherit property "channelInterpretation" with the proper type
+PASS GainNode interface: existence and properties of interface object
+PASS GainNode interface object length
+PASS GainNode interface object name
+PASS GainNode interface: existence and properties of interface prototype object
+PASS GainNode interface: existence and properties of interface prototype object's "constructor" property
+PASS GainNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS GainNode interface: attribute gain
+PASS Unscopable handled correctly for gain property on GainNode
+PASS GainNode must be primary interface of new GainNode(context)
+PASS Stringification of new GainNode(context)
+PASS GainNode interface: new GainNode(context) must inherit property "gain" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new GainNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new GainNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new GainNode(context) must inherit property "channelInterpretation" with the proper type
+PASS IIRFilterNode interface: existence and properties of interface object
+PASS IIRFilterNode interface object length
+PASS IIRFilterNode interface object name
+PASS IIRFilterNode interface: existence and properties of interface prototype object
+PASS IIRFilterNode interface: existence and properties of interface prototype object's "constructor" property
+PASS IIRFilterNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS IIRFilterNode interface: operation getFrequencyResponse(Float32Array, Float32Array, Float32Array)
+PASS Unscopable handled correctly for getFrequencyResponse(Float32Array, Float32Array, Float32Array) on IIRFilterNode
+PASS IIRFilterNode must be primary interface of new IIRFilterNode(context, {feedforward: [1], feedback: [1]})
+PASS Stringification of new IIRFilterNode(context, {feedforward: [1], feedback: [1]})
+PASS IIRFilterNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "getFrequencyResponse(Float32Array, Float32Array, Float32Array)" with the proper type
+PASS IIRFilterNode interface: calling getFrequencyResponse(Float32Array, Float32Array, Float32Array) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) with too few arguments must throw TypeError
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "context" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new IIRFilterNode(context, {feedforward: [1], feedback: [1]}) must inherit property "channelInterpretation" with the proper type
+PASS MediaElementAudioSourceNode interface: existence and properties of interface object
+PASS MediaElementAudioSourceNode interface object length
+PASS MediaElementAudioSourceNode interface object name
+PASS MediaElementAudioSourceNode interface: existence and properties of interface prototype object
+PASS MediaElementAudioSourceNode interface: existence and properties of interface prototype object's "constructor" property
+PASS MediaElementAudioSourceNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS MediaElementAudioSourceNode interface: attribute mediaElement
+PASS Unscopable handled correctly for mediaElement property on MediaElementAudioSourceNode
+PASS MediaElementAudioSourceNode must be primary interface of new MediaElementAudioSourceNode(context, {mediaElement: new Audio})
+PASS Stringification of new MediaElementAudioSourceNode(context, {mediaElement: new Audio})
+PASS MediaElementAudioSourceNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "mediaElement" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "context" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new MediaElementAudioSourceNode(context, {mediaElement: new Audio}) must inherit property "channelInterpretation" with the proper type
+PASS MediaStreamAudioDestinationNode interface: existence and properties of interface object
+PASS MediaStreamAudioDestinationNode interface object length
+PASS MediaStreamAudioDestinationNode interface object name
+PASS MediaStreamAudioDestinationNode interface: existence and properties of interface prototype object
+PASS MediaStreamAudioDestinationNode interface: existence and properties of interface prototype object's "constructor" property
+PASS MediaStreamAudioDestinationNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS MediaStreamAudioDestinationNode interface: attribute stream
+PASS Unscopable handled correctly for stream property on MediaStreamAudioDestinationNode
+PASS MediaStreamAudioDestinationNode must be primary interface of new MediaStreamAudioDestinationNode(context)
+PASS Stringification of new MediaStreamAudioDestinationNode(context)
+PASS MediaStreamAudioDestinationNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "stream" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new MediaStreamAudioDestinationNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new MediaStreamAudioDestinationNode(context) must inherit property "channelInterpretation" with the proper type
+PASS MediaStreamAudioSourceNode interface: existence and properties of interface object
+PASS MediaStreamAudioSourceNode interface object length
+PASS MediaStreamAudioSourceNode interface object name
+PASS MediaStreamAudioSourceNode interface: existence and properties of interface prototype object
+PASS MediaStreamAudioSourceNode interface: existence and properties of interface prototype object's "constructor" property
+PASS MediaStreamAudioSourceNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS MediaStreamAudioSourceNode interface: attribute mediaStream
+PASS Unscopable handled correctly for mediaStream property on MediaStreamAudioSourceNode
+FAIL MediaStreamTrackAudioSourceNode interface: existence and properties of interface object assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+FAIL MediaStreamTrackAudioSourceNode interface object length assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+FAIL MediaStreamTrackAudioSourceNode interface object name assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+FAIL MediaStreamTrackAudioSourceNode interface: existence and properties of interface prototype object assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+FAIL MediaStreamTrackAudioSourceNode interface: existence and properties of interface prototype object's "constructor" property assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+FAIL MediaStreamTrackAudioSourceNode interface: existence and properties of interface prototype object's @@unscopables property assert_own_property: self does not have own property "MediaStreamTrackAudioSourceNode" expected property "MediaStreamTrackAudioSourceNode" missing
+PASS OscillatorNode interface: existence and properties of interface object
+PASS OscillatorNode interface object length
+PASS OscillatorNode interface object name
+PASS OscillatorNode interface: existence and properties of interface prototype object
+PASS OscillatorNode interface: existence and properties of interface prototype object's "constructor" property
+PASS OscillatorNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS OscillatorNode interface: attribute type
+PASS Unscopable handled correctly for type property on OscillatorNode
+PASS OscillatorNode interface: attribute frequency
+PASS Unscopable handled correctly for frequency property on OscillatorNode
+PASS OscillatorNode interface: attribute detune
+PASS Unscopable handled correctly for detune property on OscillatorNode
+PASS OscillatorNode interface: operation setPeriodicWave(PeriodicWave)
+PASS Unscopable handled correctly for setPeriodicWave(PeriodicWave) on OscillatorNode
+PASS OscillatorNode must be primary interface of new OscillatorNode(context)
+PASS Stringification of new OscillatorNode(context)
+PASS OscillatorNode interface: new OscillatorNode(context) must inherit property "type" with the proper type
+PASS OscillatorNode interface: new OscillatorNode(context) must inherit property "frequency" with the proper type
+PASS OscillatorNode interface: new OscillatorNode(context) must inherit property "detune" with the proper type
+PASS OscillatorNode interface: new OscillatorNode(context) must inherit property "setPeriodicWave(PeriodicWave)" with the proper type
+PASS OscillatorNode interface: calling setPeriodicWave(PeriodicWave) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: new OscillatorNode(context) must inherit property "onended" with the proper type
+PASS AudioScheduledSourceNode interface: new OscillatorNode(context) must inherit property "start(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling start(double) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioScheduledSourceNode interface: new OscillatorNode(context) must inherit property "stop(double)" with the proper type
+PASS AudioScheduledSourceNode interface: calling stop(double) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new OscillatorNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new OscillatorNode(context) must inherit property "channelInterpretation" with the proper type
+PASS PannerNode interface: existence and properties of interface object
+PASS PannerNode interface object length
+PASS PannerNode interface object name
+PASS PannerNode interface: existence and properties of interface prototype object
+PASS PannerNode interface: existence and properties of interface prototype object's "constructor" property
+PASS PannerNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS PannerNode interface: attribute panningModel
+PASS Unscopable handled correctly for panningModel property on PannerNode
+PASS PannerNode interface: attribute positionX
+PASS Unscopable handled correctly for positionX property on PannerNode
+PASS PannerNode interface: attribute positionY
+PASS Unscopable handled correctly for positionY property on PannerNode
+PASS PannerNode interface: attribute positionZ
+PASS Unscopable handled correctly for positionZ property on PannerNode
+PASS PannerNode interface: attribute orientationX
+PASS Unscopable handled correctly for orientationX property on PannerNode
+PASS PannerNode interface: attribute orientationY
+PASS Unscopable handled correctly for orientationY property on PannerNode
+PASS PannerNode interface: attribute orientationZ
+PASS Unscopable handled correctly for orientationZ property on PannerNode
+PASS PannerNode interface: attribute distanceModel
+PASS Unscopable handled correctly for distanceModel property on PannerNode
+PASS PannerNode interface: attribute refDistance
+PASS Unscopable handled correctly for refDistance property on PannerNode
+PASS PannerNode interface: attribute maxDistance
+PASS Unscopable handled correctly for maxDistance property on PannerNode
+PASS PannerNode interface: attribute rolloffFactor
+PASS Unscopable handled correctly for rolloffFactor property on PannerNode
+PASS PannerNode interface: attribute coneInnerAngle
+PASS Unscopable handled correctly for coneInnerAngle property on PannerNode
+PASS PannerNode interface: attribute coneOuterAngle
+PASS Unscopable handled correctly for coneOuterAngle property on PannerNode
+PASS PannerNode interface: attribute coneOuterGain
+PASS Unscopable handled correctly for coneOuterGain property on PannerNode
+PASS PannerNode interface: operation setPosition(float, float, float)
+PASS Unscopable handled correctly for setPosition(float, float, float) on PannerNode
+PASS PannerNode interface: operation setOrientation(float, float, float)
+PASS Unscopable handled correctly for setOrientation(float, float, float) on PannerNode
+PASS PannerNode must be primary interface of new PannerNode(context)
+PASS Stringification of new PannerNode(context)
+PASS PannerNode interface: new PannerNode(context) must inherit property "panningModel" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "positionX" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "positionY" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "positionZ" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "orientationX" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "orientationY" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "orientationZ" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "distanceModel" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "refDistance" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "maxDistance" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "rolloffFactor" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "coneInnerAngle" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "coneOuterAngle" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "coneOuterGain" with the proper type
+PASS PannerNode interface: new PannerNode(context) must inherit property "setPosition(float, float, float)" with the proper type
+PASS PannerNode interface: calling setPosition(float, float, float) on new PannerNode(context) with too few arguments must throw TypeError
+PASS PannerNode interface: new PannerNode(context) must inherit property "setOrientation(float, float, float)" with the proper type
+PASS PannerNode interface: calling setOrientation(float, float, float) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new PannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new PannerNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new PannerNode(context) must inherit property "channelInterpretation" with the proper type
+PASS PeriodicWave interface: existence and properties of interface object
+PASS PeriodicWave interface object length
+PASS PeriodicWave interface object name
+PASS PeriodicWave interface: existence and properties of interface prototype object
+PASS PeriodicWave interface: existence and properties of interface prototype object's "constructor" property
+PASS PeriodicWave interface: existence and properties of interface prototype object's @@unscopables property
+PASS PeriodicWave must be primary interface of new PeriodicWave(context)
+PASS Stringification of new PeriodicWave(context)
+PASS ScriptProcessorNode interface: existence and properties of interface object
+PASS ScriptProcessorNode interface object length
+PASS ScriptProcessorNode interface object name
+PASS ScriptProcessorNode interface: existence and properties of interface prototype object
+PASS ScriptProcessorNode interface: existence and properties of interface prototype object's "constructor" property
+PASS ScriptProcessorNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS ScriptProcessorNode interface: attribute onaudioprocess
+PASS Unscopable handled correctly for onaudioprocess property on ScriptProcessorNode
+PASS ScriptProcessorNode interface: attribute bufferSize
+PASS Unscopable handled correctly for bufferSize property on ScriptProcessorNode
+PASS ScriptProcessorNode must be primary interface of context.createScriptProcessor()
+PASS Stringification of context.createScriptProcessor()
+PASS ScriptProcessorNode interface: context.createScriptProcessor() must inherit property "onaudioprocess" with the proper type
+PASS ScriptProcessorNode interface: context.createScriptProcessor() must inherit property "bufferSize" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on context.createScriptProcessor() with too few arguments must throw TypeError
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "context" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "channelCount" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: context.createScriptProcessor() must inherit property "channelInterpretation" with the proper type
+PASS StereoPannerNode interface: existence and properties of interface object
+PASS StereoPannerNode interface object length
+PASS StereoPannerNode interface object name
+PASS StereoPannerNode interface: existence and properties of interface prototype object
+PASS StereoPannerNode interface: existence and properties of interface prototype object's "constructor" property
+PASS StereoPannerNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS StereoPannerNode interface: attribute pan
+PASS Unscopable handled correctly for pan property on StereoPannerNode
+PASS StereoPannerNode must be primary interface of new StereoPannerNode(context)
+PASS Stringification of new StereoPannerNode(context)
+PASS StereoPannerNode interface: new StereoPannerNode(context) must inherit property "pan" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new StereoPannerNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new StereoPannerNode(context) must inherit property "channelInterpretation" with the proper type
+PASS WaveShaperNode interface: existence and properties of interface object
+PASS WaveShaperNode interface object length
+PASS WaveShaperNode interface object name
+PASS WaveShaperNode interface: existence and properties of interface prototype object
+PASS WaveShaperNode interface: existence and properties of interface prototype object's "constructor" property
+PASS WaveShaperNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS WaveShaperNode interface: attribute curve
+PASS Unscopable handled correctly for curve property on WaveShaperNode
+PASS WaveShaperNode interface: attribute oversample
+PASS Unscopable handled correctly for oversample property on WaveShaperNode
+PASS WaveShaperNode must be primary interface of new WaveShaperNode(context)
+PASS Stringification of new WaveShaperNode(context)
+PASS WaveShaperNode interface: new WaveShaperNode(context) must inherit property "curve" with the proper type
+PASS WaveShaperNode interface: new WaveShaperNode(context) must inherit property "oversample" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on new WaveShaperNode(context) with too few arguments must throw TypeError
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "context" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "channelCount" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: new WaveShaperNode(context) must inherit property "channelInterpretation" with the proper type
+PASS AudioWorklet interface: existence and properties of interface object
+PASS AudioWorklet interface object length
+PASS AudioWorklet interface object name
+PASS AudioWorklet interface: existence and properties of interface prototype object
+PASS AudioWorklet interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioWorklet interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioWorklet must be primary interface of context.audioWorklet
+PASS Stringification of context.audioWorklet
+PASS AudioWorkletGlobalScope interface: existence and properties of interface object
+PASS AudioParamMap interface: existence and properties of interface object
+PASS AudioParamMap interface object length
+PASS AudioParamMap interface object name
+PASS AudioParamMap interface: existence and properties of interface prototype object
+PASS AudioParamMap interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioParamMap interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioParamMap must be primary interface of worklet_node.parameters
+PASS Stringification of worklet_node.parameters
+PASS AudioWorkletNode interface: existence and properties of interface object
+PASS AudioWorkletNode interface object length
+PASS AudioWorkletNode interface object name
+PASS AudioWorkletNode interface: existence and properties of interface prototype object
+PASS AudioWorkletNode interface: existence and properties of interface prototype object's "constructor" property
+PASS AudioWorkletNode interface: existence and properties of interface prototype object's @@unscopables property
+PASS AudioWorkletNode interface: attribute parameters
+PASS Unscopable handled correctly for parameters property on AudioWorkletNode
+PASS AudioWorkletNode interface: attribute port
+PASS Unscopable handled correctly for port property on AudioWorkletNode
+PASS AudioWorkletNode interface: attribute onprocessorerror
+PASS Unscopable handled correctly for onprocessorerror property on AudioWorkletNode
+PASS AudioWorkletNode must be primary interface of worklet_node
+PASS Stringification of worklet_node
+PASS AudioWorkletNode interface: worklet_node must inherit property "parameters" with the proper type
+PASS AudioWorkletNode interface: worklet_node must inherit property "port" with the proper type
+PASS AudioWorkletNode interface: worklet_node must inherit property "onprocessorerror" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "connect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioNode, unsigned long, unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "connect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling connect(AudioParam, unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect()" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "disconnect(unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect(AudioNode)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect(AudioNode, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect(AudioNode, unsigned long, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioNode, unsigned long, unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect(AudioParam)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "disconnect(AudioParam, unsigned long)" with the proper type
+PASS AudioNode interface: calling disconnect(AudioParam, unsigned long) on worklet_node with too few arguments must throw TypeError
+PASS AudioNode interface: worklet_node must inherit property "context" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "numberOfInputs" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "numberOfOutputs" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "channelCount" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "channelCountMode" with the proper type
+PASS AudioNode interface: worklet_node must inherit property "channelInterpretation" with the proper type
+PASS AudioWorkletProcessor interface: existence and properties of interface object
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window.js b/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window.js
new file mode 100644
index 0000000..eb4734b0
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/webaudio/idlharness.https.window.js
@@ -0,0 +1,75 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+// META: timeout=long
+
+// https://webaudio.github.io/web-audio-api/
+
+'use strict';
+
+idl_test(
+  ['webaudio'],
+  ['cssom', 'uievents', 'mediacapture-streams', 'html', 'dom'],
+  async idl_array => {
+    idl_array.add_untested_idls('interface SVGElement {};');
+    idl_array.add_untested_idls('interface WorkletGlobalScope {};');
+    idl_array.add_untested_idls('interface Worklet {};');
+
+    idl_array.add_objects({
+      BaseAudioContext: [],
+      AudioContext: ['context'],
+      OfflineAudioContext: ['new OfflineAudioContext(1, 1, sample_rate)'],
+      OfflineAudioCompletionEvent: [
+        'new OfflineAudioCompletionEvent("", {renderedBuffer: buffer})'
+      ],
+      AudioBuffer: ['buffer'],
+      AudioNode: [],
+      AudioParam: ['new AudioBufferSourceNode(context).playbackRate'],
+      AudioScheduledSourceNode: [],
+      AnalyserNode: ['new AnalyserNode(context)'],
+      AudioBufferSourceNode: ['new AudioBufferSourceNode(context)'],
+      AudioDestinationNode: ['context.destination'],
+      AudioListener: ['context.listener'],
+      AudioProcessingEvent: [`new AudioProcessingEvent('', {
+        playbackTime: 0, inputBuffer: buffer, outputBuffer: buffer
+      })`],
+      BiquadFilterNode: ['new BiquadFilterNode(context)'],
+      ChannelMergerNode: ['new ChannelMergerNode(context)'],
+      ChannelSplitterNode: ['new ChannelSplitterNode(context)'],
+      ConstantSourceNode: ['new ConstantSourceNode(context)'],
+      ConvolverNode: ['new ConvolverNode(context)'],
+      DelayNode: ['new DelayNode(context)'],
+      DynamicsCompressorNode: ['new DynamicsCompressorNode(context)'],
+      GainNode: ['new GainNode(context)'],
+      IIRFilterNode: [
+        'new IIRFilterNode(context, {feedforward: [1], feedback: [1]})'
+      ],
+      MediaElementAudioSourceNode: [
+        'new MediaElementAudioSourceNode(context, {mediaElement: new Audio})'
+      ],
+      MediaStreamAudioDestinationNode: [
+        'new MediaStreamAudioDestinationNode(context)'
+      ],
+      MediaStreamAudioSourceNode: [],
+      MediaStreamTrackAudioSourceNode: [],
+      OscillatorNode: ['new OscillatorNode(context)'],
+      PannerNode: ['new PannerNode(context)'],
+      PeriodicWave: ['new PeriodicWave(context)'],
+      ScriptProcessorNode: ['context.createScriptProcessor()'],
+      StereoPannerNode: ['new StereoPannerNode(context)'],
+      WaveShaperNode: ['new WaveShaperNode(context)'],
+      AudioWorklet: ['context.audioWorklet'],
+      AudioWorkletGlobalScope: [],
+      AudioParamMap: ['worklet_node.parameters'],
+      AudioWorkletNode: ['worklet_node'],
+      AudioWorkletProcessor: [],
+    });
+
+    self.sample_rate = 44100;
+    self.context = new AudioContext;
+    self.buffer = new AudioBuffer({length: 1, sampleRate: sample_rate});
+    await context.audioWorklet.addModule(
+      'the-audio-api/the-audioworklet-interface/processors/dummy-processor.js');
+    self.worklet_node = new AudioWorkletNode(context, 'dummy');
+  },
+  'Test driver'
+);
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.html b/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.html
deleted file mode 100644
index efff8302..0000000
--- a/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.html
+++ /dev/null
@@ -1,83 +0,0 @@
-<!DOCTYPE html>
-<html>
-<head>
-<meta charset="utf-8">
-<title>Web Storage IDL tests</title>
-<link rel="author" title="W3C" href="http://www.w3.org/" />
-<link rel="help" href="http://www.w3.org/TR/webstorage/#storage"/>
-<script src="/resources/testharness.js"></script>
-<script src="/resources/testharnessreport.js"></script>
-<script src="/resources/WebIDLParser.js"></script>
-<script src="/resources/idlharness.js"></script>
-</head>
-<body>
-<h1>Web Storage IDL tests</h1>
-<div id="log"></div>
-
-<pre id='untested_idl' style='display:none'>
-[Global=Window, Exposed=Window]
-interface Window {
-};
-</pre>
-
-<pre id='idl'>
-interface Storage {
-  readonly attribute unsigned long length;
-  DOMString? key(unsigned long index);
-  getter DOMString? getItem(DOMString key);
-  setter void setItem(DOMString key, DOMString value);
-  deleter void removeItem(DOMString key);
-  void clear();
-};
-[NoInterfaceObject]
-interface WindowSessionStorage {
-  readonly attribute Storage sessionStorage;
-};
-Window implements WindowSessionStorage;
-[NoInterfaceObject]
-interface WindowLocalStorage {
-  readonly attribute Storage localStorage;
-};
-Window implements WindowLocalStorage;
-[Constructor(DOMString type, optional StorageEventInit eventInitDict)]
-interface StorageEvent : Event {
-  readonly attribute DOMString? key;
-  readonly attribute DOMString? oldValue;
-  readonly attribute DOMString? newValue;
-  readonly attribute DOMString url;
-  readonly attribute Storage? storageArea;
-};
-
-dictionary StorageEventInit : EventInit {
-  DOMString? key;
-  DOMString? oldValue;
-  DOMString? newValue;
-  DOMString url;
-  Storage? storageArea;
-};
-</pre>
-
-<script>
-function do_test([html, dom]) {
-  var idl_array = new IdlArray();
-
-  idl_array.add_untested_idls(dom, { only: ['Event', 'EventInit'] });
-  idl_array.add_untested_idls(html, { only: ['EventStorageInit'] });
-  idl_array.add_untested_idls(document.getElementById("untested_idl").textContent);
-  idl_array.add_idls(document.getElementById("idl").textContent);
-
-  idl_array.add_objects({Storage: ["window.localStorage"]});
-
-  idl_array.test();
-}
-
-function fetch_text(url) {
-  return fetch(url).then(response => response.text());
-}
-
-promise_test(function () {
-  return Promise.all(['/interfaces/html.idl', '/interfaces/dom.idl'].map(fetch_text)).then(do_test);
-}, 'webstorage interfaces');
-</script>
-</body>
-</html>
diff --git a/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.window.js b/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.window.js
new file mode 100644
index 0000000..7a6f16a
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/external/wpt/webstorage/idlharness.window.js
@@ -0,0 +1,30 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+
+// http://www.w3.org/TR/webstorage/#storage
+
+idl_test(
+  [], [], // Srcs + deps manually handled below.
+  async idl_array => {
+    const [html, dom] = await Promise.all(['html', 'dom']
+        .map(i => fetch(`/interfaces/${i}.idl`).then(r => r.text())));
+    idl_array.add_idls(html, {
+      only: [
+        'Storage',
+        'WindowSessionStorage',
+        'WindowLocalStorage',
+        'StorageEvent',
+        'StorageEventInit',
+      ]});
+    idl_array.add_dependency_idls(dom);
+
+    idl_array.add_objects({
+      Storage: [
+        'localStorage',
+        'sessionStorage',
+      ],
+      StorageEvent: ['new StorageEvent("storage")']
+    });
+  },
+  'webstorage interfaces'
+);
diff --git a/third_party/WebKit/LayoutTests/fast/backgrounds/bug420029.html b/third_party/WebKit/LayoutTests/fast/backgrounds/bug420029.html
index 20e5c0c..4bcfe4f 100644
--- a/third_party/WebKit/LayoutTests/fast/backgrounds/bug420029.html
+++ b/third_party/WebKit/LayoutTests/fast/backgrounds/bug420029.html
@@ -1,6 +1,6 @@
 <style>
 *{
-    -webkit-padding-start: 6801059;
+    padding-inline-start: 6801059;
     background-image:-webkit-cross-fade(url(foo1), url(foo2), 50%);
 }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/block/float/float-list-changed-before-layout-crash.html b/third_party/WebKit/LayoutTests/fast/block/float/float-list-changed-before-layout-crash.html
index ee2f26e..820e7bc8 100644
--- a/third_party/WebKit/LayoutTests/fast/block/float/float-list-changed-before-layout-crash.html
+++ b/third_party/WebKit/LayoutTests/fast/block/float/float-list-changed-before-layout-crash.html
@@ -26,7 +26,7 @@
         display: table-header-group;
       }
       #el6 {
-        -webkit-border-after: solid;
+        border-block-end: solid;
         float: left;
       }
       #el6:last-of-type {
diff --git a/third_party/WebKit/LayoutTests/fast/block/float/intruding-float-not-removed-from-next-sibling-crash.html b/third_party/WebKit/LayoutTests/fast/block/float/intruding-float-not-removed-from-next-sibling-crash.html
index 544e62e..3960955f 100644
--- a/third_party/WebKit/LayoutTests/fast/block/float/intruding-float-not-removed-from-next-sibling-crash.html
+++ b/third_party/WebKit/LayoutTests/fast/block/float/intruding-float-not-removed-from-next-sibling-crash.html
@@ -9,7 +9,7 @@
     padding-top: 1em;
     padding-bottom: 1em;
     margin-bottom: 1em;
-    -webkit-margin-before: -100px; 
+    margin-block-start: -100px; 
 }
 </style>
 <script>
diff --git a/third_party/WebKit/LayoutTests/fast/block/float/overhanging-float-crashes-when-sibling-becomes-formatting-context.html b/third_party/WebKit/LayoutTests/fast/block/float/overhanging-float-crashes-when-sibling-becomes-formatting-context.html
index 1e39e89..7216f0e8 100644
--- a/third_party/WebKit/LayoutTests/fast/block/float/overhanging-float-crashes-when-sibling-becomes-formatting-context.html
+++ b/third_party/WebKit/LayoutTests/fast/block/float/overhanging-float-crashes-when-sibling-becomes-formatting-context.html
@@ -20,7 +20,7 @@
         document.documentElement.style.webkitWritingMode='vertical-rl';
         iframe.style.all='unset';
         document.documentElement.clientWidth;
-        iframe.style.webkitMarginAfter='22vmin';
+        iframe.style.marginBlockEnd='22vmin';
         if (window.testRunner)
             testRunner.notifyDone();
     }
diff --git a/third_party/WebKit/LayoutTests/fast/block/margin-collapse/self-collapsing-block-discards-margin.html b/third_party/WebKit/LayoutTests/fast/block/margin-collapse/self-collapsing-block-discards-margin.html
index 2b73efef..0ea6273 100644
--- a/third_party/WebKit/LayoutTests/fast/block/margin-collapse/self-collapsing-block-discards-margin.html
+++ b/third_party/WebKit/LayoutTests/fast/block/margin-collapse/self-collapsing-block-discards-margin.html
@@ -3,7 +3,7 @@
 <style>
 * { clear:both; }
 .discard { clear:none; -webkit-margin-after-collapse:discard; }
-.float { display:inherit; -webkit-border-before-style:dashed; float:left; margin-top:2%; }
+.float { display:inherit; border-block-start-style:dashed; float:left; margin-top:2%; }
 </style>
 <div class="float"></div>
 <div class="discard"></div>
diff --git a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/003.html b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/003.html
index 9ace83e2f..c8f83de 100644
--- a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/003.html
+++ b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/003.html
@@ -24,7 +24,7 @@
 
     div#two {
         background: green;
-        -webkit-margin-before: -100px;
+        margin-block-start: -100px;
     }
 
     body { -webkit-writing-mode: vertical-lr }
diff --git a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/004.html b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/004.html
index 77c51fa..fffcdb2 100644
--- a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/004.html
+++ b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-lr/004.html
@@ -24,7 +24,7 @@
 
     div#two {
         background: green;
-        -webkit-margin-before: -100px;
+        margin-block-start: -100px;
     }
 
     body { -webkit-writing-mode: vertical-lr }
diff --git a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/003.html b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/003.html
index ff29337..b938d603 100644
--- a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/003.html
+++ b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/003.html
@@ -24,7 +24,7 @@
 
     div#two {
         background: green;
-        -webkit-margin-before: -100px;
+        margin-block-start: -100px;
     }
 
     body { -webkit-writing-mode: vertical-rl }
diff --git a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/004.html b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/004.html
index 74757f8..31c3618 100644
--- a/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/004.html
+++ b/third_party/WebKit/LayoutTests/fast/block/positioning/vertical-rl/004.html
@@ -24,7 +24,7 @@
 
     div#two {
         background: green;
-        -webkit-margin-before: -100px;
+        margin-block-start: -100px;
     }
 
     body { -webkit-writing-mode: vertical-rl }
diff --git a/third_party/WebKit/LayoutTests/fast/borders/border-image-border-radius.html b/third_party/WebKit/LayoutTests/fast/borders/border-image-border-radius.html
index 1504fc1..ef3bf20 100644
--- a/third_party/WebKit/LayoutTests/fast/borders/border-image-border-radius.html
+++ b/third_party/WebKit/LayoutTests/fast/borders/border-image-border-radius.html
@@ -1,3 +1,4 @@
+<!DOCTYPE html>
 <html>
 <head>
     <style>
diff --git a/third_party/WebKit/LayoutTests/fast/borders/border-styles-split.html b/third_party/WebKit/LayoutTests/fast/borders/border-styles-split.html
index 6a71032..427ca95 100644
--- a/third_party/WebKit/LayoutTests/fast/borders/border-styles-split.html
+++ b/third_party/WebKit/LayoutTests/fast/borders/border-styles-split.html
@@ -3,7 +3,7 @@
     display: inline-block;
   }
   span {
-    -webkit-padding-start: 15px; -webkit-padding-end:15px;
+    padding-inline-start: 15px; padding-inline-end:15px;
     font-size: 48px;
     background-color: #DDD;
     line-height: 2em;
diff --git a/third_party/WebKit/LayoutTests/fast/css/border-start-end-expected.txt b/third_party/WebKit/LayoutTests/fast/css/border-start-end-expected.txt
index fe7f945..da92363 100644
--- a/third_party/WebKit/LayoutTests/fast/css/border-start-end-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/css/border-start-end-expected.txt
@@ -3,46 +3,46 @@
 On success, you will see a series of "PASS" messages, followed by "TEST COMPLETE".
 
 
-PASS testWidth("ltr", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid") is 110
-PASS testWidth("ltr", "-webkit-border-end-width: 20px; -webkit-border-end-style:  solid") is 120
-PASS testWidth("rtl", "-webkit-border-start-width: 10px; -webkit-border-start-style:  solid") is 110
-PASS testWidth("rtl", "-webkit-border-end-width: 20px; -webkit-border-end-style:  solid") is 120
-PASS test("ltr", "-webkit-border-start-color: rgb(255, 0, 0)", "border-left-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "border-left-color: rgb(255, 0, 0)", "-webkit-border-start-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "-webkit-border-end-color: rgb(255, 0, 0)", "border-right-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "border-right-color: rgb(255, 0, 0)", "-webkit-border-end-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "-webkit-border-start-color: rgb(255, 0, 0)", "border-right-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "border-right-color: rgb(255, 0, 0)", "-webkit-border-start-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "-webkit-border-end-color: rgb(255, 0, 0)", "border-left-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "border-left-color: rgb(255, 0, 0)", "-webkit-border-end-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "-webkit-border-start-style: dotted", "border-left-style") is "dotted"
-PASS test("ltr", "border-left-style: dotted", "-webkit-border-start-style") is "dotted"
-PASS test("ltr", "-webkit-border-end-style: dotted", "border-right-style") is "dotted"
-PASS test("ltr", "border-right-style: dotted", "-webkit-border-end-style") is "dotted"
-PASS test("rtl", "-webkit-border-start-style: dotted", "border-right-style") is "dotted"
-PASS test("rtl", "border-right-style: dotted", "-webkit-border-start-style") is "dotted"
-PASS test("rtl", "-webkit-border-end-style: dotted", "border-left-style") is "dotted"
-PASS test("rtl", "border-left-style: dotted", "-webkit-border-end-style") is "dotted"
-PASS test("ltr", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid", "border-left-width") is "10px"
-PASS test("ltr", "-webkit-border-end-width: 10px; -webkit-border-end-style: solid", "border-right-width") is "10px"
-PASS test("rtl", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid", "border-right-width") is "10px"
-PASS test("rtl", "-webkit-border-end-width: 10px; -webkit-border-end-style: solid", "border-left-width") is "10px"
-PASS test("ltr", "border-left: 10px solid", "-webkit-border-start-width") is "10px"
-PASS test("ltr", "border-right: 10px solid", "-webkit-border-end-width") is "10px"
-PASS test("rtl", "border-left: 10px solid", "-webkit-border-end-width") is "10px"
-PASS test("rtl", "border-right: 10px solid", "-webkit-border-start-width") is "10px"
-PASS test("ltr", "-webkit-border-start: 10px solid red", "border-left-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "-webkit-border-start: 10px solid red", "border-left-style") is "solid"
-PASS test("ltr", "-webkit-border-start: 10px solid red", "border-left-width") is "10px"
-PASS test("rtl", "-webkit-border-start: 10px solid red", "border-right-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "-webkit-border-start: 10px solid red", "border-right-style") is "solid"
-PASS test("rtl", "-webkit-border-start: 10px solid red", "border-right-width") is "10px"
-PASS test("ltr", "-webkit-border-end: 10px solid red", "border-right-color") is "rgb(255, 0, 0)"
-PASS test("ltr", "-webkit-border-end: 10px solid red", "border-right-style") is "solid"
-PASS test("ltr", "-webkit-border-end: 10px solid red", "border-right-width") is "10px"
-PASS test("rtl", "-webkit-border-end: 10px solid red", "border-left-color") is "rgb(255, 0, 0)"
-PASS test("rtl", "-webkit-border-end: 10px solid red", "border-left-style") is "solid"
-PASS test("rtl", "-webkit-border-end: 10px solid red", "border-left-width") is "10px"
+PASS testWidth("ltr", "border-inline-start-width: 10px; border-inline-start-style: solid") is 110
+PASS testWidth("ltr", "border-inline-end-width: 20px; border-inline-end-style:  solid") is 120
+PASS testWidth("rtl", "border-inline-start-width: 10px; border-inline-start-style:  solid") is 110
+PASS testWidth("rtl", "border-inline-end-width: 20px; border-inline-end-style:  solid") is 120
+PASS test("ltr", "border-inline-start-color: rgb(255, 0, 0)", "border-left-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-left-color: rgb(255, 0, 0)", "border-inline-start-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-inline-end-color: rgb(255, 0, 0)", "border-right-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-right-color: rgb(255, 0, 0)", "border-inline-end-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-inline-start-color: rgb(255, 0, 0)", "border-right-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-right-color: rgb(255, 0, 0)", "border-inline-start-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-inline-end-color: rgb(255, 0, 0)", "border-left-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-left-color: rgb(255, 0, 0)", "border-inline-end-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-inline-start-style: dotted", "border-left-style") is "dotted"
+PASS test("ltr", "border-left-style: dotted", "border-inline-start-style") is "dotted"
+PASS test("ltr", "border-inline-end-style: dotted", "border-right-style") is "dotted"
+PASS test("ltr", "border-right-style: dotted", "border-inline-end-style") is "dotted"
+PASS test("rtl", "border-inline-start-style: dotted", "border-right-style") is "dotted"
+PASS test("rtl", "border-right-style: dotted", "border-inline-start-style") is "dotted"
+PASS test("rtl", "border-inline-end-style: dotted", "border-left-style") is "dotted"
+PASS test("rtl", "border-left-style: dotted", "border-inline-end-style") is "dotted"
+PASS test("ltr", "border-inline-start-width: 10px; border-inline-start-style: solid", "border-left-width") is "10px"
+PASS test("ltr", "border-inline-end-width: 10px; border-inline-end-style: solid", "border-right-width") is "10px"
+PASS test("rtl", "border-inline-start-width: 10px; border-inline-start-style: solid", "border-right-width") is "10px"
+PASS test("rtl", "border-inline-end-width: 10px; border-inline-end-style: solid", "border-left-width") is "10px"
+PASS test("ltr", "border-left: 10px solid", "border-inline-start-width") is "10px"
+PASS test("ltr", "border-right: 10px solid", "border-inline-end-width") is "10px"
+PASS test("rtl", "border-left: 10px solid", "border-inline-end-width") is "10px"
+PASS test("rtl", "border-right: 10px solid", "border-inline-start-width") is "10px"
+PASS test("ltr", "border-inline-start: 10px solid red", "border-left-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-inline-start: 10px solid red", "border-left-style") is "solid"
+PASS test("ltr", "border-inline-start: 10px solid red", "border-left-width") is "10px"
+PASS test("rtl", "border-inline-start: 10px solid red", "border-right-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-inline-start: 10px solid red", "border-right-style") is "solid"
+PASS test("rtl", "border-inline-start: 10px solid red", "border-right-width") is "10px"
+PASS test("ltr", "border-inline-end: 10px solid red", "border-right-color") is "rgb(255, 0, 0)"
+PASS test("ltr", "border-inline-end: 10px solid red", "border-right-style") is "solid"
+PASS test("ltr", "border-inline-end: 10px solid red", "border-right-width") is "10px"
+PASS test("rtl", "border-inline-end: 10px solid red", "border-left-color") is "rgb(255, 0, 0)"
+PASS test("rtl", "border-inline-end: 10px solid red", "border-left-style") is "solid"
+PASS test("rtl", "border-inline-end: 10px solid red", "border-left-width") is "10px"
 PASS successfullyParsed is true
 
 TEST COMPLETE
diff --git a/third_party/WebKit/LayoutTests/fast/css/crash-inherit-value-font-family.html b/third_party/WebKit/LayoutTests/fast/css/crash-inherit-value-font-family.html
index 100a041..48a6bae 100644
--- a/third_party/WebKit/LayoutTests/fast/css/crash-inherit-value-font-family.html
+++ b/third_party/WebKit/LayoutTests/fast/css/crash-inherit-value-font-family.html
@@ -1,6 +1,6 @@
 <!DOCTYPE html>
 <body>
-<var style="float: right; -webkit-border-before: groove; font-family: inherit"></var>
+<var style="float: right; border-block-start: groove; font-family: inherit"></var>
 
 <script>
     if (window.testRunner)
diff --git a/third_party/WebKit/LayoutTests/fast/css/fontMetric-webkit-border-end-width-null-crash.html b/third_party/WebKit/LayoutTests/fast/css/fontMetric-webkit-border-end-width-null-crash.html
index a3aa5d4..b0538f8 100644
--- a/third_party/WebKit/LayoutTests/fast/css/fontMetric-webkit-border-end-width-null-crash.html
+++ b/third_party/WebKit/LayoutTests/fast/css/fontMetric-webkit-border-end-width-null-crash.html
@@ -2,7 +2,7 @@
     if (window.testRunner)
         testRunner.dumpAsText();
     document.writeln("<v>");
-    document.body.innerHTML="<style>*{-webkit-border-end-width:0ex;}</style>";
+    document.body.innerHTML="<style>*{border-inline-end-width:0ex;}</style>";
     document.write("<title>x");
     document.body.innerHTML = "<a href='https://bugs.webkit.org/show_bug.cgi?id=57756'>chrome.dll!WebCore::ComputedStyle::fontMetrics ReadAV@NULL (two crashes)<br>PASSED: This test did not crash!";
 </script>
diff --git a/third_party/WebKit/LayoutTests/fast/css/identical-logical-height-decl.html b/third_party/WebKit/LayoutTests/fast/css/identical-logical-height-decl.html
index 20888a1..bd5d8910 100644
--- a/third_party/WebKit/LayoutTests/fast/css/identical-logical-height-decl.html
+++ b/third_party/WebKit/LayoutTests/fast/css/identical-logical-height-decl.html
@@ -5,9 +5,9 @@
   </head>
   <body style="overflow:hidden;">
     <p>There should be no red on this page.</p>
-    <div style="-webkit-logical-height:10px;"></div> <!-- set height -->
+    <div style="block-size:10px;"></div> <!-- set height -->
     <div style="-webkit-writing-mode:vertical-lr; width:100px; height:100px; background:red;">
-      <div style="-webkit-logical-height:10px;"> <!-- set width! -->
+      <div style="block-size:10px;"> <!-- set width! -->
         <div style="width:100px; background:white;"></div>
       </div>
     </div>
diff --git a/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values-expected.txt b/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values-expected.txt
index 83374a0..adda70d8 100644
--- a/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values-expected.txt
@@ -2,8 +2,12 @@
 PASS test('background-position') is "inherit, initial"
 PASS test('background-repeat') is "inherit, initial"
 PASS test('border') is "inherit, initial"
+PASS test('border-block-end') is "inherit, initial"
+PASS test('border-block-start') is "inherit, initial"
 PASS test('border-bottom') is "inherit, initial"
 PASS test('border-color') is "inherit, initial"
+PASS test('border-inline-end') is "inherit, initial"
+PASS test('border-inline-start') is "inherit, initial"
 PASS test('border-left') is "inherit, initial"
 PASS test('border-radius') is "inherit, initial"
 PASS test('border-right') is "inherit, initial"
@@ -18,10 +22,6 @@
 PASS test('overflow') is "inherit, initial"
 PASS test('padding') is "inherit, initial"
 PASS test('webkit-animation') is "inherit, initial"
-PASS test('webkit-border-after') is "inherit, initial"
-PASS test('webkit-border-before') is "inherit, initial"
-PASS test('webkit-border-end') is "inherit, initial"
-PASS test('webkit-border-start') is "inherit, initial"
 PASS test('webkit-columns') is "inherit, initial"
 PASS test('webkit-column-rule') is "inherit, initial"
 PASS test('webkit-flex-flow') is "inherit, initial"
diff --git a/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values.html b/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values.html
index fbc93530..6e49ec44 100644
--- a/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values.html
+++ b/third_party/WebKit/LayoutTests/fast/css/inherit-initial-shorthand-values.html
@@ -18,10 +18,14 @@
     "background-position",
     "background-repeat",
     "border",
+    "border-block-end",
+    "border-block-start",
     "border-bottom",
     "border-color",
     // getPropertyValue() functionality not supported, see http://webkit.org/b/103245.
     // "border-image",
+    "border-inline-end",
+    "border-inline-start",
     "border-left",
     "border-radius",
     "border-right",
@@ -36,10 +40,6 @@
     "overflow",
     "padding",
     "webkit-animation",
-    "webkit-border-after",
-    "webkit-border-before",
-    "webkit-border-end",
-    "webkit-border-start",
     "webkit-columns",
     "webkit-column-rule",
     "webkit-flex-flow",
diff --git a/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix-expected.txt b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix-expected.txt
new file mode 100644
index 0000000..88f7c87
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix-expected.txt
@@ -0,0 +1,38 @@
+This test has incorrect results but reflects what's happening as we bring block-flow online.
+Block #1:
+Margins: 1px 2px 3px 4px
+Padding: 4px 3px 2px 1px
+Borders: 1px solid rgb(0, 0, 0), 2px dotted rgb(0, 128, 0), 3px dashed rgb(255, 255, 0), 4px double rgb(128, 0, 128)
+Width: 100px
+Height: 50px
+Block #2:
+Margins: 4px 3px 2px 1px
+Padding: 1px 2px 3px 4px
+Borders: 4px double rgb(128, 0, 128), 3px dashed rgb(255, 255, 0), 2px dotted rgb(0, 128, 0), 1px solid rgb(0, 0, 0)
+Width: 50px
+Height: 100px
+Block #3:
+Margins: 4px 1px 2px 3px
+Padding: 1px 4px 3px 2px
+Borders: 4px double rgb(128, 0, 128), 1px solid rgb(0, 0, 0), 2px dotted rgb(0, 128, 0), 3px dashed rgb(255, 255, 0)
+Width: 50px
+Height: 100px
+Block #4:
+Margins: 1px 4px 3px 2px
+Padding: 4px 1px 2px 3px
+Borders: 1px solid rgb(0, 0, 0), 4px double rgb(128, 0, 128), 3px dashed rgb(255, 255, 0), 2px dotted rgb(0, 128, 0)
+Width: 100px
+Height: 50px
+Block #5:
+Margins: 2px 3px 4px 1px
+Padding: 3px 2px 1px 4px
+Borders: 2px dotted rgb(0, 128, 0), 3px dashed rgb(255, 255, 0), 4px double rgb(128, 0, 128), 1px solid rgb(0, 0, 0)
+Width: 50px
+Height: 100px
+Block #6:
+Margins: 2px 1px 4px 3px
+Padding: 3px 4px 1px 2px
+Borders: 2px dotted rgb(0, 128, 0), 1px solid rgb(0, 0, 0), 4px double rgb(128, 0, 128), 3px dashed rgb(255, 255, 0)
+Width: 50px
+Height: 100px
+
diff --git a/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix.html b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix.html
new file mode 100644
index 0000000..48d1570
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution-webkit-prefix.html
@@ -0,0 +1,93 @@
+<html>
+<head>
+<style>
+.ltr-text {
+  direction: ltr
+}
+
+.rtl-text {
+  direction: rtl
+}
+
+.tb-block {
+  -webkit-writing-mode: horizontal-tb;
+}
+
+.lr-block {
+  -webkit-writing-mode: vertical-lr
+}
+
+.rl-block {
+  -webkit-writing-mode: vertical-rl
+}
+
+.test {
+  -webkit-margin-before: 1px;
+  -webkit-margin-end: 2px;
+  -webkit-margin-after: 3px;
+  -webkit-margin-start: 4px;
+  -webkit-padding-before:4px;
+  -webkit-padding-end:3px;
+  -webkit-padding-after:2px;
+  -webkit-padding-start:1px;
+  -webkit-logical-width:100px;
+  -webkit-logical-height:50px;
+  -webkit-border-before: 1px solid black;
+  -webkit-border-end: 2px dotted green;
+  -webkit-border-after: 3px dashed yellow;
+  -webkit-border-start: 4px double purple;
+  float:left;
+}
+</style>
+<script>
+if (window.testRunner)
+  testRunner.dumpAsText()
+  
+function dumpBlockProperties()
+{
+   result = document.getElementById('result');
+   blocks = document.getElementsByTagName('div');
+   resultStr = "This test has incorrect results but reflects what's happening as we bring block-flow online.<br>";
+   for (i = 0; i < blocks.length; ++i) {
+        resultStr += "<b>Block #" + (i+1) + ":</b><br>";
+        resultStr += "Margins: ";
+        c = getComputedStyle(blocks[i], null);
+        resultStr += c.getPropertyValue("margin-top") + " "
+                     + c.getPropertyValue("margin-right") + " "
+                     + c.getPropertyValue("margin-bottom") + " "
+                     + c.getPropertyValue("margin-left") + "<br>";
+                     
+        resultStr += "Padding: ";
+        c = getComputedStyle(blocks[i], null);
+        resultStr += c.getPropertyValue("padding-top") + " "
+                     + c.getPropertyValue("padding-right") + " "
+                     + c.getPropertyValue("padding-bottom") + " "
+                     + c.getPropertyValue("padding-left") + "<br>";
+        
+        resultStr += "Borders: "
+        resultStr += c.getPropertyValue("border-top-width") + " " + c.getPropertyValue("border-top-style") + " " + c.getPropertyValue("border-top-color") + ", "
+                     + c.getPropertyValue("border-right-width") + " " + c.getPropertyValue("border-right-style") + " " + c.getPropertyValue("border-right-color") + ", "
+                     + c.getPropertyValue("border-bottom-width") + " " + c.getPropertyValue("border-bottom-style") + " " + c.getPropertyValue("border-bottom-color") + ", "
+                     + c.getPropertyValue("border-left-width") + " " + c.getPropertyValue("border-left-style") + " " + c.getPropertyValue("border-left-color") + "<br>"
+                     
+        resultStr += "Width: "
+        resultStr += c.getPropertyValue("width") + "<br>";
+        
+        resultStr += "Height: "
+        resultStr += c.getPropertyValue("height") + "<br>";
+   }
+   result.innerHTML = resultStr;
+}
+</script>
+</head>
+<body onload="dumpBlockProperties()">
+<div class="ltr-text tb-block test"></div>
+<div class="ltr-text lr-block test"></div>
+<div class="ltr-text rl-block test"></div>
+<div class="rtl-text tb-block test"></div>
+<div class="rtl-text lr-block test"></div>
+<div class="rtl-text rl-block test"></div>
+<p style="clear:both" id="result"></p>
+</body>
+</html>
+
diff --git a/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution.html b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution.html
index 48d1570..2257b34 100644
--- a/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution.html
+++ b/third_party/WebKit/LayoutTests/fast/css/logical-property-resolution.html
@@ -22,20 +22,20 @@
 }
 
 .test {
-  -webkit-margin-before: 1px;
-  -webkit-margin-end: 2px;
-  -webkit-margin-after: 3px;
-  -webkit-margin-start: 4px;
-  -webkit-padding-before:4px;
-  -webkit-padding-end:3px;
-  -webkit-padding-after:2px;
-  -webkit-padding-start:1px;
-  -webkit-logical-width:100px;
-  -webkit-logical-height:50px;
-  -webkit-border-before: 1px solid black;
-  -webkit-border-end: 2px dotted green;
-  -webkit-border-after: 3px dashed yellow;
-  -webkit-border-start: 4px double purple;
+  margin-block-start: 1px;
+  margin-inline-end: 2px;
+  margin-block-end: 3px;
+  margin-inline-start: 4px;
+  padding-block-start:4px;
+  padding-inline-end:3px;
+  padding-block-end:2px;
+  padding-inline-start:1px;
+  inline-size:100px;
+  block-size:50px;
+  border-block-start: 1px solid black;
+  border-inline-end: 2px dotted green;
+  border-block-end: 3px dashed yellow;
+  border-inline-start: 4px double purple;
   float:left;
 }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/css/margin-start-end-expected.txt b/third_party/WebKit/LayoutTests/fast/css/margin-start-end-expected.txt
index 121c5c33..def03d1f 100644
--- a/third_party/WebKit/LayoutTests/fast/css/margin-start-end-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/css/margin-start-end-expected.txt
@@ -3,18 +3,18 @@
 On success, you will see a series of "PASS" messages, followed by "TEST COMPLETE".
 
 
-PASS test("ltr", "-webkit-margin-start", "10px", "width") is "90px"
-PASS test("ltr", "-webkit-margin-end", "20px", "width") is "80px"
-PASS test("ltr", "-webkit-margin-start", "10px", "margin-left") is "10px"
-PASS test("ltr", "-webkit-margin-end", "20px", "margin-right") is "20px"
-PASS test("ltr", "margin-left", "10px", "-webkit-margin-start") is "10px"
-PASS test("ltr", "margin-right", "20px", "-webkit-margin-end") is "20px"
-PASS test("rtl", "-webkit-margin-start", "10px", "width") is "90px"
-PASS test("rtl", "-webkit-margin-end", "20px", "width") is "80px"
-PASS test("rtl", "-webkit-margin-start", "10px", "margin-right") is "10px"
-PASS test("rtl", "-webkit-margin-end", "20px", "margin-left") is "20px"
-PASS test("rtl", "margin-right", "10px", "-webkit-margin-start") is "10px"
-PASS test("rtl", "margin-left", "20px", "-webkit-margin-end") is "20px"
+PASS test("ltr", "margin-inline-start", "10px", "width") is "90px"
+PASS test("ltr", "margin-inline-end", "20px", "width") is "80px"
+PASS test("ltr", "margin-inline-start", "10px", "margin-left") is "10px"
+PASS test("ltr", "margin-inline-end", "20px", "margin-right") is "20px"
+PASS test("ltr", "margin-left", "10px", "margin-inline-start") is "10px"
+PASS test("ltr", "margin-right", "20px", "margin-inline-end") is "20px"
+PASS test("rtl", "margin-inline-start", "10px", "width") is "90px"
+PASS test("rtl", "margin-inline-end", "20px", "width") is "80px"
+PASS test("rtl", "margin-inline-start", "10px", "margin-right") is "10px"
+PASS test("rtl", "margin-inline-end", "20px", "margin-left") is "20px"
+PASS test("rtl", "margin-right", "10px", "margin-inline-start") is "10px"
+PASS test("rtl", "margin-left", "20px", "margin-inline-end") is "20px"
 PASS successfullyParsed is true
 
 TEST COMPLETE
diff --git a/third_party/WebKit/LayoutTests/fast/css/padding-start-end-expected.txt b/third_party/WebKit/LayoutTests/fast/css/padding-start-end-expected.txt
index 8dce0c4..f80303c 100644
--- a/third_party/WebKit/LayoutTests/fast/css/padding-start-end-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/css/padding-start-end-expected.txt
@@ -3,18 +3,18 @@
 On success, you will see a series of "PASS" messages, followed by "TEST COMPLETE".
 
 
-PASS testWidth("ltr", "-webkit-padding-start", "10px") is 110
-PASS testWidth("ltr", "-webkit-padding-end", "20px") is 120
-PASS test("ltr", "-webkit-padding-start", "10px", "padding-left") is "10px"
-PASS test("ltr", "-webkit-padding-end", "20px", "padding-right") is "20px"
-PASS test("ltr", "padding-left", "10px", "-webkit-padding-start") is "10px"
-PASS test("ltr", "padding-right", "20px", "-webkit-padding-end") is "20px"
-PASS testWidth("rtl", "-webkit-padding-start", "10px") is 110
-PASS testWidth("rtl", "-webkit-padding-end", "20px") is 120
-PASS test("rtl", "-webkit-padding-start", "10px", "padding-right") is "10px"
-PASS test("rtl", "-webkit-padding-end", "20px", "padding-left") is "20px"
-PASS test("rtl", "padding-right", "10px", "-webkit-padding-start") is "10px"
-PASS test("rtl", "padding-left", "20px", "-webkit-padding-end") is "20px"
+PASS testWidth("ltr", "padding-inline-start", "10px") is 110
+PASS testWidth("ltr", "padding-inline-end", "20px") is 120
+PASS test("ltr", "padding-inline-start", "10px", "padding-left") is "10px"
+PASS test("ltr", "padding-inline-end", "20px", "padding-right") is "20px"
+PASS test("ltr", "padding-left", "10px", "padding-inline-start") is "10px"
+PASS test("ltr", "padding-right", "20px", "padding-inline-end") is "20px"
+PASS testWidth("rtl", "padding-inline-start", "10px") is 110
+PASS testWidth("rtl", "padding-inline-end", "20px") is 120
+PASS test("rtl", "padding-inline-start", "10px", "padding-right") is "10px"
+PASS test("rtl", "padding-inline-end", "20px", "padding-left") is "20px"
+PASS test("rtl", "padding-right", "10px", "padding-inline-start") is "10px"
+PASS test("rtl", "padding-left", "20px", "padding-inline-end") is "20px"
 PASS successfullyParsed is true
 
 TEST COMPLETE
diff --git a/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk-expected.txt b/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk-expected.txt
index 63567ae3..c33ec5bb 100644
--- a/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk-expected.txt
@@ -13,10 +13,10 @@
 PASS No hashless color quirk for text-shadow color
 PASS No hashless color quirk for box-shadow color
 PASS No hashless color quirk for linear-gradient colors
-PASS No hashless color quirk for -webkit-border-start-color property
-PASS No hashless color quirk for -webkit-border-end-color property
-PASS No hashless color quirk for -webkit-border-before-property
-PASS No hashless color quirk for -webkit-border-after-color property
+PASS No hashless color quirk for border-inline-start-color property
+PASS No hashless color quirk for border-inline-end-color property
+PASS No hashless color quirk for border-block-start-property
+PASS No hashless color quirk for border-block-end-color property
 PASS No hashless color quirk for -webkit-column-rule-color property
 PASS No hashless color quirk for -webkit-text-emphasis-color property
 PASS No hashless color quirk for -webkit-text-fill-color property
diff --git a/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk.html b/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk.html
index a77f9438..c0efa1e 100644
--- a/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk.html
+++ b/third_party/WebKit/LayoutTests/fast/css/parsing-color-quirk.html
@@ -14,10 +14,10 @@
 #t10 { box-shadow: 1px 1px ff0000 }
 #t11 { background-image: linear-gradient(ff0000, blue); }
 
-#t12 { -webkit-border-start-color: ff0000 }
-#t13 { -webkit-border-end-color: ff0000 }
-#t14 { -webkit-border-before-color: ff0000 }
-#t15 { -webkit-border-after-color: ff0000 }
+#t12 { border-inline-start-color: ff0000 }
+#t13 { border-inline-end-color: ff0000 }
+#t14 { border-block-start-color: ff0000 }
+#t15 { border-block-end-color: ff0000 }
 
 #t16 { -webkit-column-rule-color: ff0000 }
 
@@ -113,20 +113,20 @@
 }, "No hashless color quirk for linear-gradient colors");
 
 test(function(){
-    assert_equals(sheet.cssRules[11].style.WebkitBorderStartColor, "");
-}, "No hashless color quirk for -webkit-border-start-color property");
+    assert_equals(sheet.cssRules[11].style.borderInlineStartColor, "");
+}, "No hashless color quirk for border-inline-start-color property");
 
 test(function(){
-    assert_equals(sheet.cssRules[12].style.WebkitBorderEndColor, "");
-}, "No hashless color quirk for -webkit-border-end-color property");
+    assert_equals(sheet.cssRules[12].style.borderInlineEndColor, "");
+}, "No hashless color quirk for border-inline-end-color property");
 
 test(function(){
-    assert_equals(sheet.cssRules[13].style.WebkitBorderBeforeColor, "");
-}, "No hashless color quirk for -webkit-border-before-property");
+    assert_equals(sheet.cssRules[13].style.borderBlockStartColor, "");
+}, "No hashless color quirk for border-block-start-property");
 
 test(function(){
-    assert_equals(sheet.cssRules[14].style.WebkitBorderAfterColor, "");
-}, "No hashless color quirk for -webkit-border-after-color property");
+    assert_equals(sheet.cssRules[14].style.borderBlockEndColor, "");
+}, "No hashless color quirk for border-block-end-color property");
 
 test(function(){
     assert_equals(sheet.cssRules[15].style.WebkitColumnRuleColor, "");
diff --git a/third_party/WebKit/LayoutTests/fast/css/script-tests/border-start-end.js b/third_party/WebKit/LayoutTests/fast/css/script-tests/border-start-end.js
index 0ed26443..ddc6d672 100644
--- a/third_party/WebKit/LayoutTests/fast/css/script-tests/border-start-end.js
+++ b/third_party/WebKit/LayoutTests/fast/css/script-tests/border-start-end.js
@@ -30,43 +30,43 @@
     return result;
 }
 
-shouldBe('testWidth("ltr", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid")', '110');
-shouldBe('testWidth("ltr", "-webkit-border-end-width: 20px; -webkit-border-end-style:  solid")', '120');
-shouldBe('testWidth("rtl", "-webkit-border-start-width: 10px; -webkit-border-start-style:  solid")', '110');
-shouldBe('testWidth("rtl", "-webkit-border-end-width: 20px; -webkit-border-end-style:  solid")', '120');
+shouldBe('testWidth("ltr", "border-inline-start-width: 10px; border-inline-start-style: solid")', '110');
+shouldBe('testWidth("ltr", "border-inline-end-width: 20px; border-inline-end-style:  solid")', '120');
+shouldBe('testWidth("rtl", "border-inline-start-width: 10px; border-inline-start-style:  solid")', '110');
+shouldBe('testWidth("rtl", "border-inline-end-width: 20px; border-inline-end-style:  solid")', '120');
 
-testSame('ltr', '-webkit-border-start-color', 'border-left-color', 'rgb(255, 0, 0)');
-testSame('ltr', '-webkit-border-end-color', 'border-right-color', 'rgb(255, 0, 0)');
-testSame('rtl', '-webkit-border-start-color', 'border-right-color', 'rgb(255, 0, 0)');
-testSame('rtl', '-webkit-border-end-color', 'border-left-color', 'rgb(255, 0, 0)');
+testSame('ltr', 'border-inline-start-color', 'border-left-color', 'rgb(255, 0, 0)');
+testSame('ltr', 'border-inline-end-color', 'border-right-color', 'rgb(255, 0, 0)');
+testSame('rtl', 'border-inline-start-color', 'border-right-color', 'rgb(255, 0, 0)');
+testSame('rtl', 'border-inline-end-color', 'border-left-color', 'rgb(255, 0, 0)');
 
-testSame('ltr', '-webkit-border-start-style', 'border-left-style', 'dotted');
-testSame('ltr', '-webkit-border-end-style', 'border-right-style', 'dotted');
-testSame('rtl', '-webkit-border-start-style', 'border-right-style', 'dotted');
-testSame('rtl', '-webkit-border-end-style', 'border-left-style', 'dotted');
+testSame('ltr', 'border-inline-start-style', 'border-left-style', 'dotted');
+testSame('ltr', 'border-inline-end-style', 'border-right-style', 'dotted');
+testSame('rtl', 'border-inline-start-style', 'border-right-style', 'dotted');
+testSame('rtl', 'border-inline-end-style', 'border-left-style', 'dotted');
 
-shouldBeEqualToString('test("ltr", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid", "border-left-width")', '10px');
-shouldBeEqualToString('test("ltr", "-webkit-border-end-width: 10px; -webkit-border-end-style: solid", "border-right-width")', '10px');
-shouldBeEqualToString('test("rtl", "-webkit-border-start-width: 10px; -webkit-border-start-style: solid", "border-right-width")', '10px');
-shouldBeEqualToString('test("rtl", "-webkit-border-end-width: 10px; -webkit-border-end-style: solid", "border-left-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-inline-start-width: 10px; border-inline-start-style: solid", "border-left-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-inline-end-width: 10px; border-inline-end-style: solid", "border-right-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-inline-start-width: 10px; border-inline-start-style: solid", "border-right-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-inline-end-width: 10px; border-inline-end-style: solid", "border-left-width")', '10px');
 
-shouldBeEqualToString('test("ltr", "border-left: 10px solid", "-webkit-border-start-width")', '10px');
-shouldBeEqualToString('test("ltr", "border-right: 10px solid", "-webkit-border-end-width")', '10px');
-shouldBeEqualToString('test("rtl", "border-left: 10px solid", "-webkit-border-end-width")', '10px');
-shouldBeEqualToString('test("rtl", "border-right: 10px solid", "-webkit-border-start-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-left: 10px solid", "border-inline-start-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-right: 10px solid", "border-inline-end-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-left: 10px solid", "border-inline-end-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-right: 10px solid", "border-inline-start-width")', '10px');
 
-shouldBeEqualToString('test("ltr", "-webkit-border-start: 10px solid red", "border-left-color")', 'rgb(255, 0, 0)');
-shouldBeEqualToString('test("ltr", "-webkit-border-start: 10px solid red", "border-left-style")', 'solid');
-shouldBeEqualToString('test("ltr", "-webkit-border-start: 10px solid red", "border-left-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-inline-start: 10px solid red", "border-left-color")', 'rgb(255, 0, 0)');
+shouldBeEqualToString('test("ltr", "border-inline-start: 10px solid red", "border-left-style")', 'solid');
+shouldBeEqualToString('test("ltr", "border-inline-start: 10px solid red", "border-left-width")', '10px');
 
-shouldBeEqualToString('test("rtl", "-webkit-border-start: 10px solid red", "border-right-color")', 'rgb(255, 0, 0)');
-shouldBeEqualToString('test("rtl", "-webkit-border-start: 10px solid red", "border-right-style")', 'solid');
-shouldBeEqualToString('test("rtl", "-webkit-border-start: 10px solid red", "border-right-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-inline-start: 10px solid red", "border-right-color")', 'rgb(255, 0, 0)');
+shouldBeEqualToString('test("rtl", "border-inline-start: 10px solid red", "border-right-style")', 'solid');
+shouldBeEqualToString('test("rtl", "border-inline-start: 10px solid red", "border-right-width")', '10px');
 
-shouldBeEqualToString('test("ltr", "-webkit-border-end: 10px solid red", "border-right-color")', 'rgb(255, 0, 0)');
-shouldBeEqualToString('test("ltr", "-webkit-border-end: 10px solid red", "border-right-style")', 'solid');
-shouldBeEqualToString('test("ltr", "-webkit-border-end: 10px solid red", "border-right-width")', '10px');
+shouldBeEqualToString('test("ltr", "border-inline-end: 10px solid red", "border-right-color")', 'rgb(255, 0, 0)');
+shouldBeEqualToString('test("ltr", "border-inline-end: 10px solid red", "border-right-style")', 'solid');
+shouldBeEqualToString('test("ltr", "border-inline-end: 10px solid red", "border-right-width")', '10px');
 
-shouldBeEqualToString('test("rtl", "-webkit-border-end: 10px solid red", "border-left-color")', 'rgb(255, 0, 0)');
-shouldBeEqualToString('test("rtl", "-webkit-border-end: 10px solid red", "border-left-style")', 'solid');
-shouldBeEqualToString('test("rtl", "-webkit-border-end: 10px solid red", "border-left-width")', '10px');
+shouldBeEqualToString('test("rtl", "border-inline-end: 10px solid red", "border-left-color")', 'rgb(255, 0, 0)');
+shouldBeEqualToString('test("rtl", "border-inline-end: 10px solid red", "border-left-style")', 'solid');
+shouldBeEqualToString('test("rtl", "border-inline-end: 10px solid red", "border-left-width")', '10px');
diff --git a/third_party/WebKit/LayoutTests/fast/css/script-tests/margin-start-end.js b/third_party/WebKit/LayoutTests/fast/css/script-tests/margin-start-end.js
index 970f1ef..df26c3c 100644
--- a/third_party/WebKit/LayoutTests/fast/css/script-tests/margin-start-end.js
+++ b/third_party/WebKit/LayoutTests/fast/css/script-tests/margin-start-end.js
@@ -17,16 +17,16 @@
     return result;
 }
 
-shouldBeEqualToString('test("ltr", "-webkit-margin-start", "10px", "width")', '90px');
-shouldBeEqualToString('test("ltr", "-webkit-margin-end", "20px", "width")', '80px');
-shouldBeEqualToString('test("ltr", "-webkit-margin-start", "10px", "margin-left")', '10px');
-shouldBeEqualToString('test("ltr", "-webkit-margin-end", "20px", "margin-right")', '20px');
-shouldBeEqualToString('test("ltr", "margin-left", "10px", "-webkit-margin-start")', '10px');
-shouldBeEqualToString('test("ltr", "margin-right", "20px", "-webkit-margin-end")', '20px');
+shouldBeEqualToString('test("ltr", "margin-inline-start", "10px", "width")', '90px');
+shouldBeEqualToString('test("ltr", "margin-inline-end", "20px", "width")', '80px');
+shouldBeEqualToString('test("ltr", "margin-inline-start", "10px", "margin-left")', '10px');
+shouldBeEqualToString('test("ltr", "margin-inline-end", "20px", "margin-right")', '20px');
+shouldBeEqualToString('test("ltr", "margin-left", "10px", "margin-inline-start")', '10px');
+shouldBeEqualToString('test("ltr", "margin-right", "20px", "margin-inline-end")', '20px');
 
-shouldBeEqualToString('test("rtl", "-webkit-margin-start", "10px", "width")', '90px');
-shouldBeEqualToString('test("rtl", "-webkit-margin-end", "20px", "width")', '80px');
-shouldBeEqualToString('test("rtl", "-webkit-margin-start", "10px", "margin-right")', '10px');
-shouldBeEqualToString('test("rtl", "-webkit-margin-end", "20px", "margin-left")', '20px');
-shouldBeEqualToString('test("rtl", "margin-right", "10px", "-webkit-margin-start")', '10px');
-shouldBeEqualToString('test("rtl", "margin-left", "20px", "-webkit-margin-end")', '20px');
+shouldBeEqualToString('test("rtl", "margin-inline-start", "10px", "width")', '90px');
+shouldBeEqualToString('test("rtl", "margin-inline-end", "20px", "width")', '80px');
+shouldBeEqualToString('test("rtl", "margin-inline-start", "10px", "margin-right")', '10px');
+shouldBeEqualToString('test("rtl", "margin-inline-end", "20px", "margin-left")', '20px');
+shouldBeEqualToString('test("rtl", "margin-right", "10px", "margin-inline-start")', '10px');
+shouldBeEqualToString('test("rtl", "margin-left", "20px", "margin-inline-end")', '20px');
diff --git a/third_party/WebKit/LayoutTests/fast/css/script-tests/padding-start-end.js b/third_party/WebKit/LayoutTests/fast/css/script-tests/padding-start-end.js
index 91c07b3..0e2bd0e4 100644
--- a/third_party/WebKit/LayoutTests/fast/css/script-tests/padding-start-end.js
+++ b/third_party/WebKit/LayoutTests/fast/css/script-tests/padding-start-end.js
@@ -24,16 +24,16 @@
     return result;
 }
 
-shouldBe('testWidth("ltr", "-webkit-padding-start", "10px")', '110');
-shouldBe('testWidth("ltr", "-webkit-padding-end", "20px")', '120');
-shouldBeEqualToString('test("ltr", "-webkit-padding-start", "10px", "padding-left")', '10px');
-shouldBeEqualToString('test("ltr", "-webkit-padding-end", "20px", "padding-right")', '20px');
-shouldBeEqualToString('test("ltr", "padding-left", "10px", "-webkit-padding-start")', '10px');
-shouldBeEqualToString('test("ltr", "padding-right", "20px", "-webkit-padding-end")', '20px');
+shouldBe('testWidth("ltr", "padding-inline-start", "10px")', '110');
+shouldBe('testWidth("ltr", "padding-inline-end", "20px")', '120');
+shouldBeEqualToString('test("ltr", "padding-inline-start", "10px", "padding-left")', '10px');
+shouldBeEqualToString('test("ltr", "padding-inline-end", "20px", "padding-right")', '20px');
+shouldBeEqualToString('test("ltr", "padding-left", "10px", "padding-inline-start")', '10px');
+shouldBeEqualToString('test("ltr", "padding-right", "20px", "padding-inline-end")', '20px');
 
-shouldBe('testWidth("rtl", "-webkit-padding-start", "10px")', '110');
-shouldBe('testWidth("rtl", "-webkit-padding-end", "20px")', '120');
-shouldBeEqualToString('test("rtl", "-webkit-padding-start", "10px", "padding-right")', '10px');
-shouldBeEqualToString('test("rtl", "-webkit-padding-end", "20px", "padding-left")', '20px');
-shouldBeEqualToString('test("rtl", "padding-right", "10px", "-webkit-padding-start")', '10px');
-shouldBeEqualToString('test("rtl", "padding-left", "20px", "-webkit-padding-end")', '20px');
+shouldBe('testWidth("rtl", "padding-inline-start", "10px")', '110');
+shouldBe('testWidth("rtl", "padding-inline-end", "20px")', '120');
+shouldBeEqualToString('test("rtl", "padding-inline-start", "10px", "padding-right")', '10px');
+shouldBeEqualToString('test("rtl", "padding-inline-end", "20px", "padding-left")', '20px');
+shouldBeEqualToString('test("rtl", "padding-right", "10px", "padding-inline-start")', '10px');
+shouldBeEqualToString('test("rtl", "padding-left", "20px", "padding-inline-end")', '20px');
diff --git a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-1.html b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-1.html
index 94fcfd38..70eaf2e 100644
--- a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-1.html
+++ b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-1.html
@@ -22,21 +22,21 @@
     <p>This test the style is not copied from the wrong source when the direction is defined.</p>
     <!-- The inline style strings must be strictly equal. Each tested div must also match the exact same set of rules. -->
     <div class="ltr">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="rtl">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="default">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
 </body>
diff --git a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-2.html b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-2.html
index fa774d49..6d7db5d 100644
--- a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-2.html
+++ b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-2.html
@@ -22,21 +22,21 @@
     <p>This test the style is not copied from the wrong source when the direction is defined.</p>
     <!-- The inline style strings must be strictly equal. Each tested div must also match the exact same set of rules. -->
     <div class="rtl">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="ltr">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="default">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
 </body>
diff --git a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-3.html b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-3.html
index 121316e..3a0dd3ed 100644
--- a/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-3.html
+++ b/third_party/WebKit/LayoutTests/fast/css/style-resolver-cache-direction-3.html
@@ -22,21 +22,21 @@
     <p>This test the style is not copied from the wrong source when the direction is defined.</p>
     <!-- The inline style strings must be strictly equal. Each tested div must also match the exact same set of rules. -->
     <div class="default">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="ltr">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
     <div class="rtl">
-        <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+        <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         <div>
-            <div style="height: 25px; width: 50px; background-color: red; -webkit-margin-start: 200px; -moz-margin-start: 200px;"></div>
+            <div style="height: 25px; width: 50px; background-color: red; margin-inline-start: 200px; -moz-margin-start: 200px;"></div>
         </div>
     </div>
 </body>
diff --git a/third_party/WebKit/LayoutTests/fast/css/window-internals-isCSSPropertyUseCounted.html b/third_party/WebKit/LayoutTests/fast/css/window-internals-isCSSPropertyUseCounted.html
index 65356b6..a47559cb 100644
--- a/third_party/WebKit/LayoutTests/fast/css/window-internals-isCSSPropertyUseCounted.html
+++ b/third_party/WebKit/LayoutTests/fast/css/window-internals-isCSSPropertyUseCounted.html
@@ -11,7 +11,7 @@
 	opacity: invalid value;
 	border: black solid 5px;
 	border-left: thick dashed lightgreen;
-	-webkit-border-end-color: pink;
+	border-inline-end-color: pink;
 }
 #target2 {
   -webkit-align-content: space-between;
@@ -28,7 +28,7 @@
 	assert_true(internals.isCSSPropertyUseCounted(document, "background-color"));
 	assert_true(internals.isCSSPropertyUseCounted(document, "padding-bottom"));
 	assert_true(internals.isCSSPropertyUseCounted(document, "padding-left"));
-	assert_true(internals.isCSSPropertyUseCounted(document, "-webkit-border-end-color"));
+	assert_true(internals.isCSSPropertyUseCounted(document, "border-inline-end-color"));
 }, "Test setting and reading css properties");
 
 test(function() {
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-before-text-node.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-before-text-node.html
index 81630fba..4ef84a0 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-before-text-node.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-before-text-node.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-child-node.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-child-node.html
index 3b20976..65d5c43 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-child-node.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-child-node.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text-form-control.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text-form-control.html
index 65586c9..9838e1c 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text-form-control.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text-form-control.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testElement {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testElement {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text.html
index 05729e1..a7e1d2c 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-change-text.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-children.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-children.html
index 9017f02..b66a6ba3 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-children.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-children.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-remove-add-children.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-remove-add-children.html
index 757c2d4..5fa80e17 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-remove-add-children.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-remove-add-children.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control-child.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control-child.html
index 5d7f9c8..6a1f22b 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control-child.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control-child.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testElement {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testElement {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control.html
index 429174e1f..0d08bd8 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto-text-form-control.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testElement {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testElement {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto.html
index 1c419af..3a3de45 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-auto.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-value-change.html b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-value-change.html
index 10a12b1..8df849a 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-value-change.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/HTMLElement/attr-dir-value-change.html
@@ -4,7 +4,7 @@
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 <script src="../../../resources/js-test.js"></script>
 <style>
-.testDiv {-webkit-border-start: 5px solid green; -webkit-border-end: 5px solid red; }
+.testDiv {border-inline-start: 5px solid green; border-inline-end: 5px solid red; }
 </style>
 </head>
 <body>
diff --git a/third_party/WebKit/LayoutTests/fast/dom/shadow/attr-dir-inherit.html b/third_party/WebKit/LayoutTests/fast/dom/shadow/attr-dir-inherit.html
index 608067b..477d17a 100644
--- a/third_party/WebKit/LayoutTests/fast/dom/shadow/attr-dir-inherit.html
+++ b/third_party/WebKit/LayoutTests/fast/dom/shadow/attr-dir-inherit.html
@@ -115,8 +115,8 @@
 }, "dir=auto the first character in the distributed content with multiple levels of shadow roots");
 
 function assert_direction(description, elementToTest) {
-    elementToTest.style.webkitBorderStart = "5px solid green";
-    elementToTest.style.webkitBorderEnd = "5px solid red";
+    elementToTest.style.borderInlineStart = "5px solid green";
+    elementToTest.style.borderInlineEnd = "5px solid red";
     var actualColor = document.defaultView.getComputedStyle(elementToTest).getPropertyValue("border-right-color");
     var dirExpected = elementToTest.dataset.dirExpected;
     switch (dirExpected) {
diff --git a/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer-expected.txt b/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer-expected.txt
index ff24d7c..24ab3b4 100644
--- a/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer-expected.txt
+++ b/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer-expected.txt
@@ -1 +1 @@
-*{-webkit-border-before-style:groove}
+*{border-block-start-style:groove}
diff --git a/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer.html-disabled b/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer.html-disabled
index 080e4c4..37c8660 100644
--- a/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer.html-disabled
+++ b/third_party/WebKit/LayoutTests/fast/dynamic/crash-paint-no-documentElement-renderer.html-disabled
@@ -11,7 +11,7 @@
         document.open();
         var oUElement = document.createElement("U");
         oUElement.hidden=true;
-        oUElement.innerHTML="<style>*{-webkit-border-before-style:groove}";
+        oUElement.innerHTML="<style>*{border-block-start-style:groove}";
         document.appendChild(oUElement);
         document.close();
     }
diff --git a/third_party/WebKit/LayoutTests/fast/html/layout-runs-and-floats-crash.html b/third_party/WebKit/LayoutTests/fast/html/layout-runs-and-floats-crash.html
index 9d35e5a..6ab355e 100644
--- a/third_party/WebKit/LayoutTests/fast/html/layout-runs-and-floats-crash.html
+++ b/third_party/WebKit/LayoutTests/fast/html/layout-runs-and-floats-crash.html
@@ -1,7 +1,7 @@
 <style>
 * {
     -webkit-columns: 10ex;
-    -webkit-margin-after: 60;
+    margin-block-end: 60;
 }
 </style>
 <main></main>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/hit-test-end-of-column-with-line-height.html b/third_party/WebKit/LayoutTests/fast/multicol/hit-test-end-of-column-with-line-height.html
index aaec74f5a..49b6f63 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/hit-test-end-of-column-with-line-height.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/hit-test-end-of-column-with-line-height.html
@@ -6,8 +6,8 @@
     columns: 2;
     column-gap: 0;
     column-fill: auto;
-    -webkit-logical-width: 400px;
-    -webkit-logical-height: 80px;
+    inline-size: 400px;
+    block-size: 80px;
     font: 20px Ahem;
     line-height: 2;
 ">Lorem ipsum dolor sit amet</div>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr-expected.html b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr-expected.html
index 3c360e6..b9253cc 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr-expected.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr-expected.html
@@ -7,9 +7,9 @@
         <p>The word PASS should be seen below.</p>
         <div style="position:relative; width:30em; background:black;">
             <div style="width:20em; height:20em; -webkit-writing-mode:vertical-lr;">
-                <div style="-webkit-logical-height:6em; line-height:2em; background:cyan;">
+                <div style="block-size:6em; line-height:2em; background:cyan;">
                     <br>
-                    <span style="-webkit-margin-start:5em;">&nbsp;PASS</span>
+                    <span style="margin-inline-start:5em;">&nbsp;PASS</span>
                 </div>
             </div>
         </div>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr.html b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr.html
index 62cf792..dc0c86e 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-lr.html
@@ -22,7 +22,7 @@
         <div style="position:relative; width:30em; background:black;">
             <div id="hider" style="position:absolute; z-index:1; top:5.1em; left:2em; width:3em; height:3em; background:red;"></div>
             <div style="width:20em; height:20em; -webkit-writing-mode:vertical-lr;">
-                <div style="-webkit-columns:4; -webkit-column-gap:0; columns:4; column-fill:auto; column-gap:0; orphans:1; widows:1; -webkit-logical-height:6em; line-height:2em; background:cyan;">
+                <div style="-webkit-columns:4; -webkit-column-gap:0; columns:4; column-fill:auto; column-gap:0; orphans:1; widows:1; block-size:6em; line-height:2em; background:cyan;">
                     <br>
                     <br>
                     <br>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl-expected.html b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl-expected.html
index 4f86922..76e7190 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl-expected.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl-expected.html
@@ -7,9 +7,9 @@
         <p>The word PASS should be seen below.</p>
         <div style="position:relative; width:30em; background:black;">
             <div style="width:20em; height:20em; -webkit-writing-mode:vertical-rl;">
-                <div style="-webkit-logical-height:6em; line-height:2em; background:cyan;">
+                <div style="block-size:6em; line-height:2em; background:cyan;">
                     <br>
-                    <span style="-webkit-margin-start:5em;">&nbsp;PASS</span>
+                    <span style="margin-inline-start:5em;">&nbsp;PASS</span>
                 </div>
             </div>
         </div>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl.html b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl.html
index f1b96d8..950c20af 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/newmulticol/hide-box-vertical-rl.html
@@ -22,7 +22,7 @@
         <div style="position:relative; width:30em; background:black;">
             <div id="hider" style="position:absolute; z-index:1; top:5.1em; right:12em; width:3em; height:3em; background:red;"></div>
             <div style="width:20em; height:20em; -webkit-writing-mode:vertical-rl;">
-                <div style="-webkit-columns:4; -webkit-column-gap:0; columns:4; column-fill:auto; column-gap:0; orphans:1; widows:1; -webkit-logical-height:6em; line-height:2em; background:cyan;">
+                <div style="-webkit-columns:4; -webkit-column-gap:0; columns:4; column-fill:auto; column-gap:0; orphans:1; widows:1; block-size:6em; line-height:2em; background:cyan;">
                     <br>
                     <br>
                     <br>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/overflow-content-expected.html b/third_party/WebKit/LayoutTests/fast/multicol/overflow-content-expected.html
index 1fb1979e..11ce40d 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/overflow-content-expected.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/overflow-content-expected.html
@@ -1,8 +1,8 @@
 <style>
     div.test {
-        -webkit-logical-width: 100px;
-        -webkit-border-before: 5px solid;
-        -webkit-border-after: 15px solid;
+        inline-size: 100px;
+        border-block-start: 5px solid;
+        border-block-end: 15px solid;
         -webkit-column-count: 2;
         -webkit-column-gap: 0;
         column-count: 2;
@@ -11,17 +11,17 @@
         background-color: lightblue;
     }
     div.container {
-        -webkit-logical-height: 120px;
+        block-size: 120px;
         background-color: red;
-        -webkit-logical-width: 100px;
-        -webkit-margin-after: 8px;
+        inline-size: 100px;
+        margin-block-end: 8px;
     }
 </style>
 <div>
     <div class="container">
-        <div class="test" style="-webkit-logical-height: 100px;">
+        <div class="test" style="block-size: 100px;">
             <div>
-                <div style="-webkit-logical-height: 300px; background-color: yellow;"></div>
+                <div style="block-size: 300px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
@@ -29,16 +29,16 @@
     <div class="container">
         <div class="test">
             <div>
-                <div style="-webkit-logical-height: 200px; background-color: yellow;"></div>
+                <div style="block-size: 200px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 10px;"></div>
+            <div style="block-size: 10px;"></div>
             <div style="-webkit-column-break-before: always;">
-                <div style="-webkit-logical-height: 100px; background-color: yellow;"></div>
+                <div style="block-size: 100px; background-color: yellow;"></div>
             </div>
             <div></div>
         </div>
@@ -46,9 +46,9 @@
 </div>
 <div style="-webkit-writing-mode: vertical-rl;">
     <div class="container">
-        <div class="test" style="-webkit-logical-height: 100px;">
+        <div class="test" style="block-size: 100px;">
             <div>
-                <div style="-webkit-logical-height: 300px; background-color: yellow;"></div>
+                <div style="block-size: 300px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
@@ -56,16 +56,16 @@
     <div class="container">
         <div class="test">
             <div>
-                <div style="-webkit-logical-height: 200px; background-color: yellow;"></div>
+                <div style="block-size: 200px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 10px;"></div>
+            <div style="block-size: 10px;"></div>
             <div style="-webkit-column-break-before: always;">
-                <div style="-webkit-logical-height: 100px; background-color: yellow;"></div>
+                <div style="block-size: 100px; background-color: yellow;"></div>
             </div>
             <div></div>
         </div>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/overflow-content.html b/third_party/WebKit/LayoutTests/fast/multicol/overflow-content.html
index fc5de7b..65d090d 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/overflow-content.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/overflow-content.html
@@ -1,8 +1,8 @@
 <style>
     div.test {
-        -webkit-logical-width: 100px;
-        -webkit-border-before: 5px solid;
-        -webkit-border-after: 15px solid;
+        inline-size: 100px;
+        border-block-start: 5px solid;
+        border-block-end: 15px solid;
         -webkit-column-count: 2;
         -webkit-column-gap: 0;
         column-count: 2;
@@ -11,63 +11,63 @@
         background-color: lightblue;
     }
     div.container {
-        -webkit-logical-height: 120px;
+        block-size: 120px;
         background-color: red;
-        -webkit-logical-width: 100px;
-        -webkit-margin-after: 8px;
+        inline-size: 100px;
+        margin-block-end: 8px;
     }
 </style>
 <div>
     <div class="container">
-        <div class="test" style="-webkit-logical-height: 100px;">
-            <div style="-webkit-logical-height: 20px;">
-                <div style="-webkit-logical-height: 300px; background-color: yellow;"></div>
+        <div class="test" style="block-size: 100px;">
+            <div style="block-size: 20px;">
+                <div style="block-size: 300px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 20px;">
-                <div style="-webkit-logical-height: 200px; background-color: yellow;"></div>
+            <div style="block-size: 20px;">
+                <div style="block-size: 200px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 10px;"></div>
-            <div style="-webkit-column-break-before: always; -webkit-logical-height: 10px;">
-                <div style="-webkit-logical-height: 100px; background-color: yellow;"></div>
+            <div style="block-size: 10px;"></div>
+            <div style="-webkit-column-break-before: always; block-size: 10px;">
+                <div style="block-size: 100px; background-color: yellow;"></div>
             </div>
-            <div style="-webkit-logical-height: 20px;"></div>
+            <div style="block-size: 20px;"></div>
         </div>
     </div>
 </div>
 <div style="-webkit-writing-mode: vertical-rl;">
     <div class="container">
-        <div class="test" style="-webkit-logical-height: 100px;">
-            <div style="-webkit-logical-height: 20px;">
-                <div style="-webkit-logical-height: 300px; background-color: yellow;"></div>
+        <div class="test" style="block-size: 100px;">
+            <div style="block-size: 20px;">
+                <div style="block-size: 300px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 20px;">
-                <div style="-webkit-logical-height: 200px; background-color: yellow;"></div>
+            <div style="block-size: 20px;">
+                <div style="block-size: 200px; background-color: yellow;"></div>
             </div>
         </div>
     </div>
 
     <div class="container">
         <div class="test">
-            <div style="-webkit-logical-height: 10px;"></div>
-            <div style="-webkit-column-break-before: always; -webkit-logical-height: 10px;">
-                <div style="-webkit-logical-height: 100px; background-color: yellow;"></div>
+            <div style="block-size: 10px;"></div>
+            <div style="-webkit-column-break-before: always; block-size: 10px;">
+                <div style="block-size: 100px; background-color: yellow;"></div>
             </div>
-            <div style="-webkit-logical-height: 20px;"></div>
+            <div style="block-size: 20px;"></div>
         </div>
     </div>
 </div>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/pageLogicalOffset-vertical.html b/third_party/WebKit/LayoutTests/fast/multicol/pageLogicalOffset-vertical.html
index c3fe08b..2c53bfbb3 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/pageLogicalOffset-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/pageLogicalOffset-vertical.html
@@ -12,7 +12,7 @@
     }
 
     .margin-before {
-        -webkit-margin-before: 30px;
+        margin-block-start: 30px;
         width: 60px;
     }
 
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/float-truncation.html b/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/float-truncation.html
index 6e96866..20f1643 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/float-truncation.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/float-truncation.html
@@ -19,7 +19,7 @@
     div.float {
         float: left;
         height: 51px;
-        -webkit-margin-before: 5px;
+        margin-block-start: 5px;
         color: silver;
     }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/rules-with-border-before.html b/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/rules-with-border-before.html
index 9a364acf..9421a96 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/rules-with-border-before.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/vertical-lr/rules-with-border-before.html
@@ -9,8 +9,8 @@
         -webkit-column-gap: 50px;
     }
 
-    .before { -webkit-border-before: 50px solid blue; }
-    .after { -webkit-border-after: 50px solid blue; }
+    .before { border-block-start: 50px solid blue; }
+    .after { border-block-end: 50px solid blue; }
 
     .test div { width: 200px; }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/float-truncation.html b/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/float-truncation.html
index 4ac1f46..f82e8759 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/float-truncation.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/float-truncation.html
@@ -20,7 +20,7 @@
     div.float {
         float: left;
         height: 51px;
-        -webkit-margin-before: 5px;
+        margin-block-start: 5px;
         color: silver;
     }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/rules-with-border-before.html b/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/rules-with-border-before.html
index e5b536d..900714f 100644
--- a/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/rules-with-border-before.html
+++ b/third_party/WebKit/LayoutTests/fast/multicol/vertical-rl/rules-with-border-before.html
@@ -9,8 +9,8 @@
         -webkit-column-gap: 50px;
     }
 
-    .before { -webkit-border-before: 50px solid blue; }
-    .after { -webkit-border-after: 50px solid blue; }
+    .before { border-block-start: 50px solid blue; }
+    .after { border-block-end: 50px solid blue; }
 
     .test div { width: 200px; }
 </style>
diff --git a/third_party/WebKit/LayoutTests/fast/ruby/position-after.html b/third_party/WebKit/LayoutTests/fast/ruby/position-after.html
index 5b32ff0..df6e570f 100644
--- a/third_party/WebKit/LayoutTests/fast/ruby/position-after.html
+++ b/third_party/WebKit/LayoutTests/fast/ruby/position-after.html
@@ -1,7 +1,7 @@
 <script src="../../resources/ahem.js"></script>
 <style>
     div.container {
-        -webkit-logical-width: 8em;
+        inline-size: 8em;
         font: 20px Ahem;
         -webkit-font-smoothing: none;
         outline: thin dashed lightblue;
diff --git a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/001-vertical.html b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/001-vertical.html
index 2f655b0..e094f83 100644
--- a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/001-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/001-vertical.html
@@ -10,6 +10,6 @@
 </div>
 
 
-<div style="border:1px solid green; -webkit-margin-before:10em;">
+<div style="border:1px solid green; margin-block-start:10em;">
 <table style="border-collapse:collapse;"><tr><td style="border:2px solid black">Test</td></tr></table>
 </div>
diff --git a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/border-collapsing-head-foot-vertical.html b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/border-collapsing-head-foot-vertical.html
index a14d63a5..fb2454c1 100644
--- a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/border-collapsing-head-foot-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/border-collapsing-head-foot-vertical.html
@@ -7,12 +7,12 @@
         .test table { border-collapse: collapse; margin: 10px; }
         .test td, th { border: 1px lightgray solid; }
         
-        .example1 tfoot { -webkit-border-before: 2px solid blue;
-            -webkit-border-after: 2px solid green;
+        .example1 tfoot { border-block-start: 2px solid blue;
+            border-block-end: 2px solid green;
         }
 
-        .example2 thead { -webkit-border-before: 2px solid blue;
-            -webkit-border-after: 2px solid green;
+        .example2 thead { border-block-start: 2px solid blue;
+            border-block-end: 2px solid green;
         }
 
         table table { -webkit-writing-mode: vertical-rl; }
diff --git a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/equal-precedence-resolution-vertical.html b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/equal-precedence-resolution-vertical.html
index a3ba34d..32963794 100644
--- a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/equal-precedence-resolution-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/equal-precedence-resolution-vertical.html
@@ -5,50 +5,50 @@
 </style>
 <div style="-webkit-writing-mode: vertical-rl; height: 200px; border: solid;">
 <table>
-    <colgroup span="2"><colgroup span="1" style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="-webkit-border-start: 5px solid red">
+    <colgroup span="2"><colgroup span="1" style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="border-inline-start: 5px solid red">
     <tr>
-        <td style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5); -webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);"></td>
-        <td style="-webkit-border-start: 5px solid red;"></td>
+        <td style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5); border-block-end: 5px solid rgba(0, 127, 0, 0.5);"></td>
+        <td style="border-inline-start: 5px solid red;"></td>
         <td></td>
         <td></td>
     </tr>
     <tr>
-        <td style="-webkit-border-before: 5px solid red;"></td>
+        <td style="border-block-start: 5px solid red;"></td>
         <td></td>
         <td></td>
         <td></td>
     </tr>
 </table>
 <table style="direction: rtl;">
-    <colgroup span="2"><colgroup span="1" style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="-webkit-border-start: 5px solid red">
+    <colgroup span="2"><colgroup span="1" style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="border-inline-start: 5px solid red">
     <tr>
-        <td style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5); -webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);"></td>
-        <td style="-webkit-border-start: 5px solid red;"></td>
+        <td style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5); border-block-end: 5px solid rgba(0, 127, 0, 0.5);"></td>
+        <td style="border-inline-start: 5px solid red;"></td>
         <td></td>
         <td></td>
     </tr>
     <tr>
-        <td style="-webkit-border-before: 5px solid red;"></td>
+        <td style="border-block-start: 5px solid red;"></td>
         <td></td>
         <td></td>
         <td></td>
     </tr>
 </table>
 <table>
-    <tr style="-webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);">
+    <tr style="border-block-end: 5px solid rgba(0, 127, 0, 0.5);">
         <td></td>
     </tr>
-    <tr style="-webkit-border-before: 5px solid red;">
+    <tr style="border-block-start: 5px solid red;">
         <td></td>
     </tr>
 </table>
 <table>
-    <tbody style="-webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);">
+    <tbody style="border-block-end: 5px solid rgba(0, 127, 0, 0.5);">
         <tr>
             <td></td>
         </tr>
     </tbody>
-    <tbody style="-webkit-border-before: 5px solid red;">
+    <tbody style="border-block-start: 5px solid red;">
         <tr>
             <td></td>
         </tr>
@@ -57,50 +57,50 @@
 </div>
 <div style="-webkit-writing-mode: vertical-lr; height: 200px; border: solid;">
 <table>
-    <colgroup span="2"><colgroup span="1" style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="-webkit-border-start: 5px solid red">
+    <colgroup span="2"><colgroup span="1" style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="border-inline-start: 5px solid red">
     <tr>
-        <td style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5); -webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);"></td>
-        <td style="-webkit-border-start: 5px solid red;"></td>
+        <td style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5); border-block-end: 5px solid rgba(0, 127, 0, 0.5);"></td>
+        <td style="border-inline-start: 5px solid red;"></td>
         <td></td>
         <td></td>
     </tr>
     <tr>
-        <td style="-webkit-border-before: 5px solid red;"></td>
+        <td style="border-block-start: 5px solid red;"></td>
         <td></td>
         <td></td>
         <td></td>
     </tr>
 </table>
 <table style="direction: rtl;">
-    <colgroup span="2"><colgroup span="1" style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="-webkit-border-start: 5px solid red">
+    <colgroup span="2"><colgroup span="1" style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5);"><colgroup span="1" style="border-inline-start: 5px solid red">
     <tr>
-        <td style="-webkit-border-end: 5px solid rgba(0, 127, 0, 0.5); -webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);"></td>
-        <td style="-webkit-border-start: 5px solid red;"></td>
+        <td style="border-inline-end: 5px solid rgba(0, 127, 0, 0.5); border-block-end: 5px solid rgba(0, 127, 0, 0.5);"></td>
+        <td style="border-inline-start: 5px solid red;"></td>
         <td></td>
         <td></td>
     </tr>
     <tr>
-        <td style="-webkit-border-before: 5px solid red;"></td>
+        <td style="border-block-start: 5px solid red;"></td>
         <td></td>
         <td></td>
         <td></td>
     </tr>
 </table>
 <table>
-    <tr style="-webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);">
+    <tr style="border-block-end: 5px solid rgba(0, 127, 0, 0.5);">
         <td></td>
     </tr>
-    <tr style="-webkit-border-before: 5px solid red;">
+    <tr style="border-block-start: 5px solid red;">
         <td></td>
     </tr>
 </table>
 <table>
-    <tbody style="-webkit-border-after: 5px solid rgba(0, 127, 0, 0.5);">
+    <tbody style="border-block-end: 5px solid rgba(0, 127, 0, 0.5);">
         <tr>
             <td></td>
         </tr>
     </tbody>
-    <tbody style="-webkit-border-before: 5px solid red;">
+    <tbody style="border-block-start: 5px solid red;">
         <tr>
             <td></td>
         </tr>
diff --git a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/rtl-border-collapsing-vertical.html b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/rtl-border-collapsing-vertical.html
index 101790f..819235fa 100644
--- a/third_party/WebKit/LayoutTests/fast/table/border-collapsing/rtl-border-collapsing-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/table/border-collapsing/rtl-border-collapsing-vertical.html
@@ -9,15 +9,15 @@
 <hr>
 <table cellpadding="4" cellspacing="0" style="text-align: center; -webkit-writing-mode: vertical-rl;">
 <tr>
-<td style="-webkit-border-end: 1px solid black;">RTL</td>
+<td style="border-inline-end: 1px solid black;">RTL</td>
 <td>LTR</td>
 </tr>
 <tr>
-<td style="-webkit-border-end: 1px solid black;">
-    <table style="direction:rtl; border-collapse: collapse; -webkit-border-start: 2px solid red; -webkit-border-end: 2px solid blue;">
+<td style="border-inline-end: 1px solid black;">
+    <table style="direction:rtl; border-collapse: collapse; border-inline-start: 2px solid red; border-inline-end: 2px solid blue;">
         <col>
         <col>
-        <col style="background: yellow; -webkit-border-end: 2px green dotted; -webkit-border-start: solid 1px;">
+        <col style="background: yellow; border-inline-end: 2px green dotted; border-inline-start: solid 1px;">
         <col style="background: silver;">
         <tr>
             <td>&nbsp;&nbsp;</td>
@@ -26,7 +26,7 @@
             <td>&nbsp;&nbsp;</td>
         </tr>
         <tr>
-            <td style="-webkit-border-end: 4px solid maroon;">&nbsp;&nbsp;</td>
+            <td style="border-inline-end: 4px solid maroon;">&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
@@ -35,10 +35,10 @@
     </table>
 </td>
 <td>    
-    <table style="direction:ltr; border-collapse: collapse; -webkit-border-start: 2px solid red; -webkit-border-end: 2px solid blue;">
+    <table style="direction:ltr; border-collapse: collapse; border-inline-start: 2px solid red; border-inline-end: 2px solid blue;">
         <col>
         <col>
-        <col style="background: yellow; -webkit-border-end: 2px green dotted; -webkit-border-start: solid 1px;">
+        <col style="background: yellow; border-inline-end: 2px green dotted; border-inline-start: solid 1px;">
         <col style="background: silver;">
         <tr>
             <td>&nbsp;&nbsp;</td>
@@ -47,7 +47,7 @@
             <td>&nbsp;&nbsp;</td>
         </tr>
         <tr>
-            <td style="-webkit-border-end: 4px solid maroon;">&nbsp;&nbsp;</td>
+            <td style="border-inline-end: 4px solid maroon;">&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
             <td>&nbsp;&nbsp;</td>
diff --git a/third_party/WebKit/LayoutTests/fast/table/fixed-with-auto-with-colspan-vertical.html b/third_party/WebKit/LayoutTests/fast/table/fixed-with-auto-with-colspan-vertical.html
index be67a15..d4ee2746 100644
--- a/third_party/WebKit/LayoutTests/fast/table/fixed-with-auto-with-colspan-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/table/fixed-with-auto-with-colspan-vertical.html
@@ -1,5 +1,5 @@
 <style>
-    table { border-spacing: 5px 0; table-layout: fixed; height: 445px; -webkit-margin-before: 5px; }
+    table { border-spacing: 5px 0; table-layout: fixed; height: 445px; margin-block-start: 5px; }
     td { width: 50px; }
     #ref td { background-color: red; }
     #test td { background-color: green; }
diff --git a/third_party/WebKit/LayoutTests/fast/text/decorations-with-text-combine.html b/third_party/WebKit/LayoutTests/fast/text/decorations-with-text-combine.html
index 0dc1e6dce..732033a 100644
--- a/third_party/WebKit/LayoutTests/fast/text/decorations-with-text-combine.html
+++ b/third_party/WebKit/LayoutTests/fast/text/decorations-with-text-combine.html
@@ -7,7 +7,7 @@
     }
 
     div {
-        -webkit-margin-after: 10px;
+        margin-block-end: 10px;
     }
 
     .combine {
diff --git a/third_party/WebKit/LayoutTests/fast/text/dirty-inline-textbox-crash.html b/third_party/WebKit/LayoutTests/fast/text/dirty-inline-textbox-crash.html
index 6a861a9..0ea9cfe 100644
--- a/third_party/WebKit/LayoutTests/fast/text/dirty-inline-textbox-crash.html
+++ b/third_party/WebKit/LayoutTests/fast/text/dirty-inline-textbox-crash.html
@@ -4,7 +4,7 @@
             .box
             {
                 display: run-in; 
-                -webkit-padding-start: 10000;
+                padding-inline-start: 10000;
             }
         </style>
     </head>
diff --git a/third_party/WebKit/LayoutTests/fast/text/justify-vertical-expected.html b/third_party/WebKit/LayoutTests/fast/text/justify-vertical-expected.html
index b87fa976..a8c8f710 100644
--- a/third_party/WebKit/LayoutTests/fast/text/justify-vertical-expected.html
+++ b/third_party/WebKit/LayoutTests/fast/text/justify-vertical-expected.html
@@ -16,20 +16,20 @@
 </style>
 <dl>
 <dt>Ideographic, Kana, CJK symbols</dt>
-<dd style="-webkit-logical-width:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; W<br>WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x20B9F; &#x20B9F; &#x20B9F; &#x20B9F; &#x20B9F;<br>&#x20B9F;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; W<br>WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x20B9F; &#x20B9F; &#x20B9F; &#x20B9F; &#x20B9F;<br>&#x20B9F;&#xFEFF;WWWWWWWW</dd>
 <dt>Ideographic, Kana, CJK symbols with spans</dt>
-<dd style="-webkit-logical-width:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD; &#x56FD; &#x3042; &#xFF01; &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWW</dd>
 <dt>Ideographic, Latin, and spaces</dt>
-<dd style="-webkit-logical-width:8em;">WW &#x56FD; W &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW &#x56FD;   W  &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW &#x56FD;   W  &#x56FD;<br>WWWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW &#x56FD;   W  &#x56FD;<br>WWWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:8em;">WW &#x56FD;   W<br>&#x56FD;&#xFEFF;WWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">W.  W.  W &#x56FD;<br>WWWWWWWWWWW</dd>
+<dd style="inline-size:8em;">WW &#x56FD; W &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWW</dd>
+<dd style="inline-size:11em;">WW &#x56FD;   W  &#x56FD;<br>&#x56FD;&#xFEFF;WWWWWWWWWW</dd>
+<dd style="inline-size:11em;">WW &#x56FD;   W  &#x56FD;<br>WWWWWWWWWWW</dd>
+<dd style="inline-size:11em;">WW &#x56FD;   W  &#x56FD;<br>WWWWWWWWWWW</dd>
+<dd style="inline-size:8em;">WW &#x56FD;   W<br>&#x56FD;&#xFEFF;WWWWWWW</dd>
+<dd style="inline-size:11em;">W.  W.  W &#x56FD;<br>WWWWWWWWWWW</dd>
 </dl>
 <script>
 if (window.testRunner) {
diff --git a/third_party/WebKit/LayoutTests/fast/text/justify-vertical.html b/third_party/WebKit/LayoutTests/fast/text/justify-vertical.html
index e229385..15e7380a 100644
--- a/third_party/WebKit/LayoutTests/fast/text/justify-vertical.html
+++ b/third_party/WebKit/LayoutTests/fast/text/justify-vertical.html
@@ -16,24 +16,24 @@
 </style>
 <dl>
 <dt>Ideographic, Kana, CJK symbols</dt>
-<dd style="-webkit-logical-width:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD;&#x56FD;&#xFEFF;WWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD;WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD; WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;W WWWWWWWWW</dd>
-<dd style="-webkit-logical-width:9em;">&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD;&#x56FD;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD;WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;&#x56FD; WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD;&#x56FD;&#x3042;&#xFF01;W WWWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#x20B9F;&#xFEFF;WWWWWWWW</dd>
 <dt>Ideographic, Kana, CJK symbols with spans</dt>
-<dd style="-webkit-logical-width:9em;">&#x56FD;<span>&#x56FD;</span>&#x3042;&#xFF01;&#x56FD;&#x56FD;&#xFEFF;WWWWWWWW</dd>
+<dd style="inline-size:9em;">&#x56FD;<span>&#x56FD;</span>&#x3042;&#xFF01;&#x56FD;&#x56FD;&#xFEFF;WWWWWWWW</dd>
 <dt>Ideographic, Latin, and spaces</dt>
-<dd style="-webkit-logical-width:8em;">WW&#x56FD;W&#x56FD;&#x56FD;&#xFEFF;WWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW&#x56FD; W &#x56FD;&#x56FD;&#xFEFF;WWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW&#x56FD; W &#x56FD;WWWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">WW&#x56FD; W &#x56FD; WWWWWWWWWWW</dd>
-<dd style="-webkit-logical-width:8em;">WW&#x56FD; W &#x56FD;&#xFEFF;WWWWWWW</dd>
-<dd style="-webkit-logical-width:11em;">W. W. <span>W</span>&#x56FD;WWWWWWWWWWW</dd>
+<dd style="inline-size:8em;">WW&#x56FD;W&#x56FD;&#x56FD;&#xFEFF;WWWWWWW</dd>
+<dd style="inline-size:11em;">WW&#x56FD; W &#x56FD;&#x56FD;&#xFEFF;WWWWWWWWWW</dd>
+<dd style="inline-size:11em;">WW&#x56FD; W &#x56FD;WWWWWWWWWWW</dd>
+<dd style="inline-size:11em;">WW&#x56FD; W &#x56FD; WWWWWWWWWWW</dd>
+<dd style="inline-size:8em;">WW&#x56FD; W &#x56FD;&#xFEFF;WWWWWWW</dd>
+<dd style="inline-size:11em;">W. W. <span>W</span>&#x56FD;WWWWWWWWWWW</dd>
 <!-- These are not supported yet
 <dt>Combining and IVS</dt>
-<dd style="-webkit-logical-width:5em;">&#x304B;&#x3099;&#x304B;&#x3099;&#x304B;&#x3099;&#x304B;&#x3099;&#xFEFF;WWWW</dd>
-<dd style="-webkit-logical-width:5em;">&#x8FBA;&#xE0102;&#x8FBA;&#xE0102;&#x8FBA;&#xE0102;&#x8FBA;&#xFEFF;WWWW</dd>
+<dd style="inline-size:5em;">&#x304B;&#x3099;&#x304B;&#x3099;&#x304B;&#x3099;&#x304B;&#x3099;&#xFEFF;WWWW</dd>
+<dd style="inline-size:5em;">&#x8FBA;&#xE0102;&#x8FBA;&#xE0102;&#x8FBA;&#xE0102;&#x8FBA;&#xFEFF;WWWW</dd>
 -->
 </dl>
 <script>
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/Kusa-Makura-background-canvas.html b/third_party/WebKit/LayoutTests/fast/writing-mode/Kusa-Makura-background-canvas.html
index 58ffed1..2093b45 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/Kusa-Makura-background-canvas.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/Kusa-Makura-background-canvas.html
@@ -14,7 +14,7 @@
 /*body { font-family: HiraMinProN-W3;}		 to test the font vs fallback. */
 #div, h1, h2 { line-height: 150%; }			/* this line should make no difference for the line height as the default is 150 for the Japanese fonts. */
 
-.jisage_2 { -webkit-margin-start: 2em;}
+.jisage_2 { margin-inline-start: 2em;}
 body { margin: 8% 10%;}
 #ruby { line-height: 100%; }				/* hack to keep lines in conssitent positions. Should not be necessary. */
 #rt { line-height: 50%; }					/* hack to keep lines in conssitent positions. Should not be necessary. */
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/block-level-images.html b/third_party/WebKit/LayoutTests/fast/writing-mode/block-level-images.html
index 0722944..5e994ba9 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/block-level-images.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/block-level-images.html
@@ -1,14 +1,14 @@
 <!doctype html>
 <div style="margin:2px; float:left; height:300px;border:2px solid black;-webkit-writing-mode:vertical-lr">
 <div style="width:25px;background-color:green"></div>
-<img style="display:block;height:50%; -webkit-border-before:2px solid maroon; -webkit-border-after:2px solid maroon; -webkit-border-start: 1px solid purple; -webkit-border-end:5px dashed grey" src="resources/circle.svg">
-<img style="-webkit-margin-start: 40px; display:block;height:50px" src="resources/oval.png">
+<img style="display:block;height:50%; border-block-start:2px solid maroon; border-block-end:2px solid maroon; border-inline-start: 1px solid purple; border-inline-end:5px dashed grey" src="resources/circle.svg">
+<img style="margin-inline-start: 40px; display:block;height:50px" src="resources/oval.png">
 <div style="width:25px;background-color:green"></div>
 </div>
 
 <div style="margin:2px; float:left; height:300px;border:2px solid black;-webkit-writing-mode:vertical-lr; direction:rtl">
 <div style="width:25px;background-color:green"></div>
-<img style="display:block;height:50%; -webkit-border-before:2px solid maroon; -webkit-border-after:2px solid maroon; -webkit-border-start: 1px solid purple; -webkit-border-end:5px dashed grey" src="resources/circle.svg">
-<img style="-webkit-margin-start: 40px; display:block;height:50px" src="resources/oval.png">
+<img style="display:block;height:50%; border-block-start:2px solid maroon; border-block-end:2px solid maroon; border-inline-start: 1px solid purple; border-inline-end:5px dashed grey" src="resources/circle.svg">
+<img style="margin-inline-start: 40px; display:block;height:50px" src="resources/oval.png">
 <div style="width:25px;background-color:green"></div>
 </div>
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/border-radius-clipping-vertical-lr.html b/third_party/WebKit/LayoutTests/fast/writing-mode/border-radius-clipping-vertical-lr.html
index 702bb42..23185bd 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/border-radius-clipping-vertical-lr.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/border-radius-clipping-vertical-lr.html
@@ -1,9 +1,9 @@
 <html style="-webkit-writing-mode: vertical-lr">
 <body>
-<div style="-webkit-logical-width: 600px; -webkit-margin-after:50px">
-    <span style="-webkit-padding-start: 15px; -webkit-padding-end:15px; -webkit-padding-before:5px; -webkit-padding-after: 5px; font-size: 48px; background-color:blue">
+<div style="inline-size: 600px; margin-block-end:50px">
+    <span style="padding-inline-start: 15px; padding-inline-end:15px; padding-block-start:5px; padding-block-end: 5px; font-size: 48px; background-color:blue">
     <span style="
-        -webkit-padding-start: 15px; -webkit-padding-end:15px;
+        padding-inline-start: 15px; padding-inline-end:15px;
         font-size: 48px;
         background-color: lightyellow;
         -webkit-border-radius: 10px;
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-lr.html b/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-lr.html
index cba3c29e..7c18a76 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-lr.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-lr.html
@@ -9,7 +9,7 @@
       display: inline-block;
     }
     span {
-      -webkit-padding-start: 15px; -webkit-padding-end:15px;
+      padding-inline-start: 15px; padding-inline-end:15px;
       font-size: 48pt;
       background-color: #DDD;
       line-height: 2em;
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-rl.html b/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-rl.html
index 85af6c5..f2b3ddb5 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-rl.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/border-styles-vertical-rl.html
@@ -9,7 +9,7 @@
       display: inline-block;
     }
     span {
-      -webkit-padding-start: 15px; -webkit-padding-end:15px;
+      padding-inline-start: 15px; padding-inline-end:15px;
       font-size: 48pt;
       background-color: #DDD;
       line-height: 2em;
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/fieldsets.html b/third_party/WebKit/LayoutTests/fast/writing-mode/fieldsets.html
index 83762bb..0ade129 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/fieldsets.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/fieldsets.html
@@ -3,16 +3,16 @@
 <style>
 fieldset
 {
-    -webkit-logical-width:200px;
-    -webkit-logical-height:100px;
+    inline-size:200px;
+    block-size:100px;
     margin:1em;
     display:inline-block;
 }
 
 div
 {
-    -webkit-logical-width:100px;
-    -webkit-logical-height:25px;
+    inline-size:100px;
+    block-size:25px;
     background-color:orange;
 }
 
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/flipped-blocks-text-map-local-to-container.html b/third_party/WebKit/LayoutTests/fast/writing-mode/flipped-blocks-text-map-local-to-container.html
index ea0686b..216a4927 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/flipped-blocks-text-map-local-to-container.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/flipped-blocks-text-map-local-to-container.html
@@ -1,5 +1,5 @@
 <meta name=viewport content="width=device-width">
-<div style="border: solid; writing-mode: vertical-rl; -webkit-logical-width: 200px; -webkit-logical-height: 400px;
+<div style="border: solid; writing-mode: vertical-rl; inline-size: 200px; block-size: 400px;
     font-size: 36px; line-height: 2;
 ">
     Lorem ipsum dolor <span id="target">sit</span> amet consectetur adipiscing elit.
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear-expected.html b/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear-expected.html
index 36518494..004d56e 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear-expected.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear-expected.html
@@ -1,13 +1,13 @@
 <style>
     div.test {
-        -webkit-logical-width: 50px;
-        -webkit-logical-height: 100px;
+        inline-size: 50px;
+        block-size: 100px;
         background-color: green;
     }
 
     div.green {
         background-color: blue;
-        -webkit-logical-height: 50px;
+        block-size: 50px;
     }
 
     div.lr {
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear.html b/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear.html
index 30a187e..020d552 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/logical-height-after-clear.html
@@ -1,14 +1,14 @@
 <style>
     div.test {
-        -webkit-logical-width: 50px;
-        -webkit-logical-height: 100px;
+        inline-size: 50px;
+        block-size: 100px;
         background-color: red;
     }
 
     div.float {
         float: left;
-        -webkit-logical-height: 50px;
-        -webkit-logical-width: 50px;
+        block-size: 50px;
+        inline-size: 50px;
         background-color: blue;
     }
 
@@ -18,7 +18,7 @@
 
     div.green {
         background-color: green;
-        -webkit-logical-height: 25px;
+        block-size: 25px;
     }
 
     div.lr {
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/orthogonal-inline-block.html b/third_party/WebKit/LayoutTests/fast/writing-mode/orthogonal-inline-block.html
index b823218..392b5c7 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/orthogonal-inline-block.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/orthogonal-inline-block.html
@@ -46,46 +46,46 @@
   generate('margin-top: 1em', top);
   generate('margin-right: 1em', right);
   generate('margin-bottom: 1em', bottom);
-  generate('-webkit-margin-before: 1em', right);
-  generate('-webkit-margin-after: 1em', left);
-  generate('-webkit-margin-start: 1em', top);
-  generate('-webkit-margin-end: 1em', bottom);
+  generate('margin-block-start: 1em', right);
+  generate('margin-block-end: 1em', left);
+  generate('margin-inline-start: 1em', top);
+  generate('margin-inline-end: 1em', bottom);
 
   generate('border-left: 1em transparent solid', left);
   generate('border-top: 1em transparent solid', top);
   generate('border-right: 1em transparent solid', right);
   generate('border-bottom: 1em transparent solid', bottom);
-  generate('-webkit-border-before: 1em transparent solid', right);
-  generate('-webkit-border-after: 1em transparent solid', left);
-  generate('-webkit-border-start: 1em transparent solid', top);
-  generate('-webkit-border-end: 1em transparent solid', bottom);
+  generate('border-block-start: 1em transparent solid', right);
+  generate('border-block-end: 1em transparent solid', left);
+  generate('border-inline-start: 1em transparent solid', top);
+  generate('border-inline-end: 1em transparent solid', bottom);
 
   generate('padding-left: 1em', left);
   generate('padding-top: 1em', top);
   generate('padding-right: 1em', right);
   generate('padding-bottom: 1em', bottom);
-  generate('-webkit-padding-before: 1em', right);
-  generate('-webkit-padding-after: 1em', left);
-  generate('-webkit-padding-start: 1em', top);
-  generate('-webkit-padding-end: 1em', bottom);
+  generate('padding-block-start: 1em', right);
+  generate('padding-block-end: 1em', left);
+  generate('padding-inline-start: 1em', top);
+  generate('padding-inline-end: 1em', bottom);
 
   generate('margin-left: 20%', left);
   generate('margin-top: 20%', top);
   generate('margin-right: 20%', right);
   generate('margin-bottom: 20%', bottom);
-  generate('-webkit-margin-before: 20%', right);
-  generate('-webkit-margin-after: 20%', left);
-  generate('-webkit-margin-start: 20%', top);
-  generate('-webkit-margin-end: 20%', bottom);
+  generate('margin-block-start: 20%', right);
+  generate('margin-block-end: 20%', left);
+  generate('margin-inline-start: 20%', top);
+  generate('margin-inline-end: 20%', bottom);
 
   generate('padding-left: 20%', left);
   generate('padding-top: 20%', top);
   generate('padding-right: 20%', right);
   generate('padding-bottom: 20%', bottom);
-  generate('-webkit-padding-before: 20%', right);
-  generate('-webkit-padding-after: 20%', left);
-  generate('-webkit-padding-start: 20%', top);
-  generate('-webkit-padding-end: 20%', bottom);
+  generate('padding-block-start: 20%', right);
+  generate('padding-block-end: 20%', left);
+  generate('padding-inline-start: 20%', top);
+  generate('padding-inline-end: 20%', bottom);
 }
 
 function generate(inline_block_style, expected_direction) {
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/root-lr-basic.html b/third_party/WebKit/LayoutTests/fast/writing-mode/root-lr-basic.html
index 85d14ec..45c08e10 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/root-lr-basic.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/root-lr-basic.html
@@ -1,2 +1,2 @@
 <!doctype html>
-<html style="overflow:hidden; -webkit-writing-mode:vertical-lr; -webkit-logical-height:100px; border:2px solid black"></html>
+<html style="overflow:hidden; -webkit-writing-mode:vertical-lr; block-size:100px; border:2px solid black"></html>
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-lr-replaced-selection.html b/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-lr-replaced-selection.html
index 16b7395..7a9b3e6 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-lr-replaced-selection.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-lr-replaced-selection.html
@@ -15,4 +15,4 @@
 </head>
 
 <body onload="selectStuff()" style="font-size:36px; -webkit-writing-mode:vertical-lr;-webkit-text-orientation:sideways-right;text-rendering:optimizeLegibility;">This is the first line of text.<br>
-<img style="-webkit-logical-width:20px;-webkit-logical-height:50px;background-color:green"> This is the second line of <img style="-webkit-logical-width:20px;-webkit-logical-height:100px;background-color:purple"> text.
\ No newline at end of file
+<img style="inline-size:20px;block-size:50px;background-color:green"> This is the second line of <img style="inline-size:20px;block-size:100px;background-color:purple"> text.
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-rl-replaced-selection.html b/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-rl-replaced-selection.html
index 32d6428..0b640140 100644
--- a/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-rl-replaced-selection.html
+++ b/third_party/WebKit/LayoutTests/fast/writing-mode/vertical-rl-replaced-selection.html
@@ -16,4 +16,4 @@
 </head>
 
 <body onload="selectStuff()" style="font-size:36px; -webkit-writing-mode:vertical-rl;-webkit-text-orientation:sideways-right;text-rendering:optimizeLegibility">This is the first line of text.<br>
-<img style="-webkit-logical-width:20px;-webkit-logical-height:50px;background-color:green"> This is the second line of <img style="-webkit-logical-width:20px;-webkit-logical-height:100px;background-color:purple"> text.
\ No newline at end of file
+<img style="inline-size:20px;block-size:50px;background-color:green"> This is the second line of <img style="inline-size:20px;block-size:100px;background-color:purple"> text.
\ No newline at end of file
diff --git a/third_party/WebKit/LayoutTests/http/tests/csspaint/invalidation-background-image.html b/third_party/WebKit/LayoutTests/http/tests/csspaint/invalidation-background-image.html
index 86143c8b..6b3e32a 100644
--- a/third_party/WebKit/LayoutTests/http/tests/csspaint/invalidation-background-image.html
+++ b/third_party/WebKit/LayoutTests/http/tests/csspaint/invalidation-background-image.html
@@ -46,6 +46,6 @@
     { property: '-webkit-border-radius', invalidationProperty: 'border-radius', prevValue: '', value: '25px' },
     { property: 'border-radius', invalidationProperty: '-webkit-border-radius', prevValue: '25px', value: '' },
     { property: '-webkit-border-radius', prevValue: '100px', value: 'calc(50px + 50px)', noInvalidation: true },
-    { property: '-webkit-margin-start', invalidationProperty: 'margin-left', prevValue: 'calc(50px + 50px)', value: '100px', noInvalidation: true },
+    { property: 'margin-inline-start', invalidationProperty: 'margin-left', prevValue: 'calc(50px + 50px)', value: '100px', noInvalidation: true },
 ]);
 </script>
diff --git a/third_party/WebKit/LayoutTests/http/tests/csspaint/resources/test-runner-invalidation-logging.js b/third_party/WebKit/LayoutTests/http/tests/csspaint/resources/test-runner-invalidation-logging.js
index f8ae42d7..cea64138 100644
--- a/third_party/WebKit/LayoutTests/http/tests/csspaint/resources/test-runner-invalidation-logging.js
+++ b/third_party/WebKit/LayoutTests/http/tests/csspaint/resources/test-runner-invalidation-logging.js
@@ -10,7 +10,7 @@
 // testRunnerInvalidationLogging('background-image', [
 //      { property: 'max-height', value: '100px' },
 //      { property: 'color', prevValue: '#00F', value: 'blue', noInvalidation: true },
-//      { property: '-webkit-margin-start', invalidationProperty: 'margin-left', prevValue: 'calc(50px + 50px)', value: '100px', noInvalidation: true }
+//      { property: 'margin-inline-start', invalidationProperty: 'margin-left', prevValue: 'calc(50px + 50px)', value: '100px', noInvalidation: true }
 // ]);
 
 function testRunnerInvalidationLogging(imageType, tests) {
diff --git a/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/test-helpers.js b/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/test-helpers.js
index ecc94b7..240db09 100644
--- a/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/test-helpers.js
+++ b/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/test-helpers.js
@@ -210,3 +210,15 @@
                           'password2' + suffix, 'cookie2');
       });
 }
+
+function wait_for_port_message(port, handler) {
+  return new Promise((resolve, reject) => {
+    port.onmessage = (e) => {
+      try {
+        resolve(handler(e));
+      } catch (e) {
+        reject(e);
+      }
+    };
+  });
+}
diff --git a/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/update-no-controllee-worker.js b/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/update-no-controllee-worker.js
new file mode 100644
index 0000000..0eeb286
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/http/tests/serviceworker/resources/update-no-controllee-worker.js
@@ -0,0 +1,32 @@
+const rejectAfter = (timeout) => {
+  return new Promise((resolve, reject) => {
+    setTimeout(() => reject(), timeout);
+  });
+}
+
+const startUpdate = () => {
+  let p = [];
+  for (let i = 0; i < 10; i++) {
+    p.push(self.registration.update());
+  }
+  return Promise.all(p);
+};
+
+const update = () => {
+  // update() rejects in one of these cases:
+  //   1. at least one update() rejects, or
+  //   2. at least one update() does not resolve after 15 seconds.
+  return Promise.race([startUpdate(), rejectAfter(150000)]);
+};
+
+self.addEventListener('message', (e) => {
+  const port = e.data;
+
+  port.onmessage = (e) => {
+    update().then(() => {
+      port.postMessage('success');
+    }).catch((e) => {
+      port.postMessage('failure');
+    });
+  };
+});
diff --git a/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https-expected.txt b/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https-expected.txt
new file mode 100644
index 0000000..3e0ec4b
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https-expected.txt
@@ -0,0 +1,4 @@
+This is a testharness.js-based test.
+FAIL Verify multiple updates from service workers without controllees do not resolve immediately assert_equals: update should not have succeeded expected "failure" but got "success"
+Harness: the test ran to completion.
+
diff --git a/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https.html b/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https.html
new file mode 100644
index 0000000..40740c6
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/http/tests/serviceworker/update-no-controllee.https.html
@@ -0,0 +1,45 @@
+<!DOCTYPE html>
+<!-- This is not a WPT test because it tests non-specified Chrome-specific behavior. It tests that self-updating service workers eventually fail to update. -->
+<title>Service Worker: update no controllee</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src="resources/test-helpers.js"></script>
+<body>
+<script>
+promise_test(async (t) => {
+  // Set up ServiceWorker and wait until it's |activated|.
+  const url = 'resources/update-no-controllee-worker.js';
+  const scope = 'resources/blank.html';
+  const registration = await service_worker_unregister_and_register(t, url, scope);
+  await wait_for_state(t, registration.installing, 'activated');
+
+  const frame = await with_iframe(scope);
+
+  const w = frame.contentWindow;
+  const sw = w.navigator.serviceWorker;
+  assert_true(sw.controller instanceof w.ServiceWorker,
+              'controller should be a ServiceWorker object');
+
+  const channel = new MessageChannel();
+  sw.controller.postMessage(channel.port1, [channel.port1]);
+
+  // Remove frame so our worker has no controller.
+  frame.remove();
+  assert_equals(
+      sw.controller, null, 'disconnected frame should not be controlled');
+
+  // Send a message to the worker to start updating and wait for its response.
+  // We expect at least one of the updates, and therefore the complete proccess,
+  // to fail immediately because the delay is too long.
+  const result = wait_for_port_message(channel.port2, (e) => {
+    assert_equals(e.data, 'failure', 'update should not have succeeded');
+  });
+  channel.port2.postMessage('');
+  await result;
+
+  // Cleanup.
+  await registration.unregister();
+}, 'Verify multiple updates from service workers without controllees ' +
+       'do not resolve immediately');
+</script>
+</body>
diff --git a/third_party/WebKit/LayoutTests/http/tests/serviceworker/webexposed/global-interface-listing-service-worker-expected.txt b/third_party/WebKit/LayoutTests/http/tests/serviceworker/webexposed/global-interface-listing-service-worker-expected.txt
index 68c12d00..759ccea 100644
--- a/third_party/WebKit/LayoutTests/http/tests/serviceworker/webexposed/global-interface-listing-service-worker-expected.txt
+++ b/third_party/WebKit/LayoutTests/http/tests/serviceworker/webexposed/global-interface-listing-service-worker-expected.txt
@@ -64,11 +64,6 @@
     attribute @@toStringTag
     method constructor
     method updateUI
-interface BackgroundFetchedEvent : BackgroundFetchEvent
-    attribute @@toStringTag
-    getter fetches
-    method constructor
-    method updateUI
 interface BarcodeDetector
     attribute @@toStringTag
     method constructor
diff --git a/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/regression860637.html b/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/regression860637.html
new file mode 100644
index 0000000..b0c9722b
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/regression860637.html
@@ -0,0 +1,9 @@
+<!DOCTYPE html>
+<script src="../../../resources/testharness.js"></script>
+<script src="../../../resources/testharnessreport.js"></script>
+<script src="wasm_response_apis.js"></script>
+<script src="../wasm/resources/wasm-constants.js"></script>
+<script src="../wasm/resources/wasm-module-builder.js"></script>
+<script>
+  promise_test(TestRegression837417, "Regression test");
+</script>
diff --git a/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/wasm_response_apis.js b/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/wasm_response_apis.js
index 4ee7579..316f37e 100644
--- a/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/wasm_response_apis.js
+++ b/third_party/WebKit/LayoutTests/http/tests/wasm_streaming/wasm_response_apis.js
@@ -255,3 +255,12 @@
   worker.addEventListener('message', e => resolve(e.data));
   return promise.then(exists => assert_true(exists));
 }
+
+function TestRegression837417() {
+  let old_then = WebAssembly.Module.prototype.then;
+  WebAssembly.Module.prototype.then = resolve => resolve(String.fromCharCode(
+      null, 0x41, 0x41, 0x41, 0x41, 0x41, 0x41, 0x41, 0x41));
+
+  return WebAssembly.instantiateStreaming(fetch(incrementer_url))
+    .then(pair => assert_equals(5, pair.instance.exports.increment(4)));
+}
diff --git a/third_party/WebKit/LayoutTests/images/mask-box-image-crash.html b/third_party/WebKit/LayoutTests/images/mask-box-image-crash.html
index 35ced77..b491549 100644
--- a/third_party/WebKit/LayoutTests/images/mask-box-image-crash.html
+++ b/third_party/WebKit/LayoutTests/images/mask-box-image-crash.html
@@ -2,7 +2,7 @@
 if (window.testRunner)
     testRunner.dumpAsTextWithPixelResults();
 </script>
-<fieldset style="-webkit-margin-end: 99999px;
+<fieldset style="margin-inline-end: 99999px;
                  -webkit-mask-box-image: url('\
                  IAAABCAQAAAADLLz1FAAAACXZwQWcAAABCAAAAQgCVDTvpAAAAFElEQVQoz2P4DwUHGEZZo6xBwQI\
                  ANWY/j0PofaAAAAAASUVORK5CYII=') 10000% 10000 9999999999%;">
diff --git a/third_party/WebKit/LayoutTests/paint/transparency/compositing-alpha-fold-crash.html b/third_party/WebKit/LayoutTests/paint/transparency/compositing-alpha-fold-crash.html
index 7aae271..479e8e5 100644
--- a/third_party/WebKit/LayoutTests/paint/transparency/compositing-alpha-fold-crash.html
+++ b/third_party/WebKit/LayoutTests/paint/transparency/compositing-alpha-fold-crash.html
@@ -3,6 +3,6 @@
  <body style="opacity:0.5">
 <div style=" -webkit-mask-box-image-source: url();" >
 </div>
-<div style="box-shadow: 1px 1px black; -webkit-padding-before: 1px; isolation: isolate">
+<div style="box-shadow: 1px 1px black; padding-block-start: 1px; isolation: isolate">
 <div style="isolation: isolate"></div>
 </body>
diff --git a/third_party/WebKit/LayoutTests/scrollbars/scrollbar-part-created-with-no-parent-crash.html b/third_party/WebKit/LayoutTests/scrollbars/scrollbar-part-created-with-no-parent-crash.html
index 8ba9295d..8b34409 100644
--- a/third_party/WebKit/LayoutTests/scrollbars/scrollbar-part-created-with-no-parent-crash.html
+++ b/third_party/WebKit/LayoutTests/scrollbars/scrollbar-part-created-with-no-parent-crash.html
@@ -6,7 +6,7 @@
     margin: 0;
 }
 ::-webkit-scrollbar {
-    -webkit-logical-height: 65536;
+    block-size: 65536;
     -webkit-border-image: url(does_not_exist) 0 2 0 2;
 }
 
diff --git a/third_party/WebKit/LayoutTests/scrollbars/scrollbar-percent-padding-crash.html b/third_party/WebKit/LayoutTests/scrollbars/scrollbar-percent-padding-crash.html
index 808b25be..ef2c2789 100644
--- a/third_party/WebKit/LayoutTests/scrollbars/scrollbar-percent-padding-crash.html
+++ b/third_party/WebKit/LayoutTests/scrollbars/scrollbar-percent-padding-crash.html
@@ -3,7 +3,7 @@
 <head>
 <style>
 ::-webkit-scrollbar {
- -webkit-padding-start: 1%; background: #666 -webkit-gradient(linear, left top, right top, from(rgba(255,255,255,0.5)), color-stop(0.5, rgba(255,255,255,0.1)), color-stop(0.5, rgba(0,0,0,0)), to(rgba(0,0,0,0.01)));
+ padding-inline-start: 1%; background: #666 -webkit-gradient(linear, left top, right top, from(rgba(255,255,255,0.5)), color-stop(0.5, rgba(255,255,255,0.1)), color-stop(0.5, rgba(0,0,0,0)), to(rgba(0,0,0,0.01)));
 }
 </style>
 <script>
diff --git a/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical-expected.html b/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical-expected.html
index adde32e..39f6897 100644
--- a/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical-expected.html
+++ b/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical-expected.html
@@ -2,8 +2,8 @@
 <style>
     div.container {
         -webkit-writing-mode: vertical-rl;
-        -webkit-logical-width: 80px;
-        -webkit-logical-height: 200px;
+        inline-size: 80px;
+        block-size: 200px;
         outline: solid blue;
     }
 </style>
diff --git a/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical.html b/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical.html
index 77205ec..fa52028a 100644
--- a/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical.html
+++ b/third_party/WebKit/LayoutTests/svg/as-image/svg-intrinsic-size-rectangular-vertical.html
@@ -2,8 +2,8 @@
 <style>
     div.container {
         -webkit-writing-mode: vertical-rl;
-        -webkit-logical-width: 80px;
-        -webkit-logical-height: 200px;
+        inline-size: 80px;
+        block-size: 200px;
         outline: solid blue;
     }
 
diff --git a/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change-expected.html b/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change-expected.html
new file mode 100644
index 0000000..cfc2657
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change-expected.html
@@ -0,0 +1,5 @@
+<!DOCTYPE html>
+<script src="../../resources/run-after-layout-and-paint.js"></script>
+<svg style="width: 400px; height: 400px">
+  <image xlink:href="../custom/resources/green-rect.svg" width="100" height="100" style="transform: translate(1px, 1px);"></image>
+</svg>
diff --git a/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change.html b/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change.html
new file mode 100644
index 0000000..83eb337
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/svg/transforms/image-cull-rect-transform-change.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<script src="../../resources/run-after-layout-and-paint.js"></script>
+<svg style="width: 400px; height: 400px">
+  <image id="image" xlink:href="../custom/resources/green-rect.svg" width="100" height="100"></image>
+</svg>
+<script>
+image.style.transform = "translate(-300px, -300px)";
+if (window.testRunner)
+  testRunner.waitUntilDone();
+onload = function() {
+  runAfterLayoutAndPaint(() => {
+    image.style.transform = "translate(1px, 1px)";
+    window.testRunner.notifyDone();
+  });
+}
+</script>
diff --git a/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change-expected.html b/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change-expected.html
new file mode 100644
index 0000000..397ddf7
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change-expected.html
@@ -0,0 +1,5 @@
+<!DOCTYPE html>
+<script src="../../resources/run-after-layout-and-paint.js"></script>
+<svg style="width: 400px; height: 400px">
+  <rect width="100" height="100" fill="green" style="transform: translate(1px, 1px);"></rect>
+</svg>
diff --git a/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change.html b/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change.html
new file mode 100644
index 0000000..0b2eb79
--- /dev/null
+++ b/third_party/WebKit/LayoutTests/svg/transforms/rect-cull-rect-transform-change.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<script src="../../resources/run-after-layout-and-paint.js"></script>
+<svg style="width: 400px; height: 400px">
+  <rect id="rect" width="100" height="100" fill="green"></rect>
+</svg>
+<script>
+rect.style.transform = "translate(-300px, -300px)";
+if (window.testRunner)
+  testRunner.waitUntilDone();
+runAfterLayoutAndPaint(() => {
+  rect.style.transform = "translate(1px, 1px)";
+  window.testRunner.notifyDone();
+});
+</script>
diff --git a/third_party/WebKit/LayoutTests/touchadjustment/small-target-test.html b/third_party/WebKit/LayoutTests/touchadjustment/small-target-test.html
index 77ae0f8..2b96b28d 100644
--- a/third_party/WebKit/LayoutTests/touchadjustment/small-target-test.html
+++ b/third_party/WebKit/LayoutTests/touchadjustment/small-target-test.html
@@ -7,7 +7,7 @@
 
 <style>
   body {
-    -webkit-margin-start: 20px;
+    margin-inline-start: 20px;
     margin-top: 20px;
   }
   .control-pair {
diff --git a/third_party/WebKit/LayoutTests/xr/resources/xr-device-mocking.js b/third_party/WebKit/LayoutTests/xr/resources/xr-device-mocking.js
index 88b5719d..22058646 100644
--- a/third_party/WebKit/LayoutTests/xr/resources/xr-device-mocking.js
+++ b/third_party/WebKit/LayoutTests/xr/resources/xr-device-mocking.js
@@ -296,7 +296,13 @@
     this.displayClient_ = new device.mojom.VRDisplayClientPtr();
     this.displayInfo_ = displayInfo;
     this.service_ = service;
-    this.presentation_provider_ = new MockVRPresentationProvider();
+    this.presentation_provider_ = new MockXRPresentationProvider();
+
+    this.pose_ = null;
+    this.next_frame_id_ = 0;
+
+    this.input_sources_ = [];
+    this.next_input_source_index_ = 1;
 
     if (service.client_) {
       this.notifyClientOfDisplay();
@@ -306,28 +312,39 @@
   requestSession(sessionOptions, was_activation) {
     return this.supportsSession(sessionOptions).then((result) => {
       // The JavaScript bindings convert c_style_names to camelCase names.
-      var options = new device.mojom.VRDisplayFrameTransportOptions();
+      var options = new device.mojom.XRPresentationTransportOptions();
       options.transportMethod =
-          device.mojom.VRDisplayFrameTransportMethod.SUBMIT_AS_MAILBOX_HOLDER;
+          device.mojom.XRPresentationTransportMethod.SUBMIT_AS_MAILBOX_HOLDER;
       options.waitForTransferNotification = true;
       options.waitForRenderNotification = true;
 
-      let connection;
+      let submit_frame_sink;
       if (result.supportsSession) {
-        connection = {
+        submit_frame_sink = {
           clientRequest: this.presentation_provider_.getClientRequest(),
           provider: this.presentation_provider_.bindProvider(sessionOptions),
           transportOptions: options
         };
 
-        let magicWindowPtr = new device.mojom.VRMagicWindowProviderPtr();
-        let magicWindowRequest = mojo.makeRequest(magicWindowPtr);
-        let magicWindowBinding = new mojo.Binding(
-            device.mojom.VRMagicWindowProvider, this, magicWindowRequest);
+
+        let dataProviderPtr = new device.mojom.XRFrameDataProviderPtr();
+        let dataProviderRequest = mojo.makeRequest(dataProviderPtr);
+        let dataProviderBinding = new mojo.Binding(
+            device.mojom.XRFrameDataProvider, this, dataProviderRequest);
+
+        let enviromentProviderPtr =
+            new device.mojom.XREnviromentIntegrationProviderPtr();
+        let enviromentProviderRequest = mojo.makeRequest(enviromentProviderPtr);
+        let enviromentProviderBinding = new mojo.Binding(
+            device.mojom.XREnviromentIntegrationProvider, this,
+            enviromentProviderRequest);
 
         return Promise.resolve({
-          session:
-              {connection: connection, magicWindowProvider: magicWindowPtr}
+          session: {
+            submitFrameSink: submit_frame_sink,
+            dataProvider: dataProviderPtr,
+            enviromentProvider: enviromentProviderPtr
+          }
         });
       } else {
         return Promise.resolve({session: null});
@@ -344,10 +361,10 @@
 
   setPose(pose) {
     if (pose == null) {
-      this.presentation_provider_.pose_ = null;
+      this.pose_ = null;
     } else {
-      this.presentation_provider_.initPose();
-      this.presentation_provider_.fillPose(pose);
+      this.initPose();
+      this.fillPose(pose);
     }
   }
 
@@ -370,16 +387,33 @@
   }
 
   getFrameData() {
+    if (this.pose_) {
+      this.pose_.poseIndex++;
+
+      let input_states = [];
+      for (let i = 0; i < this.input_sources_.length; ++i) {
+        input_states.push(this.input_sources_[i].getInputSourceState());
+      }
+
+      this.pose_.inputState = input_states;
+    }
+
+    // Convert current document time to monotonic time.
+    var now = window.performance.now() / 1000.0;
+    var diff = now - internals.monotonicTimeToZeroBasedDocumentTime(now);
+    now += diff;
+    now *= 1000000;
+
     return Promise.resolve({
       frameData: {
-        pose: this.presentation_provider_.pose_,
+        pose: this.pose_,
         bufferHolder: null,
         bufferSize: {},
-        timeDelta: [],
-        projectionMatrix: [1, 0, 0, 0,
-                            0, 1, 0, 0,
-                            0, 0, 1, 0,
-                            0, 0, 0, 1]
+        timeDelta: {
+          microseconds: now,
+        },
+        frameId: this.next_frame_id_++,
+        projectionMatrix: [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]
       }
     });
   }
@@ -425,95 +459,6 @@
         displayPtr, clientRequest, this.displayInfo_);
   }
 
-  addInputSource(input_source) {
-    this.presentation_provider_.OnInputSourceAdded(input_source);
-  }
-
-  removeInputSource(input_source) {
-    this.presentation_provider_.OnInputSourceRemoved(input_source);
-  }
-}
-
-class MockVRPresentationProvider {
-  constructor() {
-    this.binding_ =
-        new mojo.Binding(device.mojom.VRPresentationProviderPtr, this);
-    this.pose_ = null;
-    this.next_frame_id_ = 0;
-    this.submit_frame_count_ = 0;
-    this.missing_frame_count_ = 0;
-
-    this.input_sources_ = [];
-    this.next_input_source_index_ = 1;
-  }
-
-  bindProvider(request) {
-    let providerPtr = new device.mojom.VRPresentationProviderPtr();
-    let providerRequest = mojo.makeRequest(providerPtr);
-
-    this.binding_.close();
-
-    this.binding_ = new mojo.Binding(
-        device.mojom.VRPresentationProvider, this, providerRequest);
-
-    return providerPtr;
-  }
-
-  getClientRequest() {
-    this.submitFrameClient_ = new device.mojom.VRSubmitFrameClientPtr();
-    return mojo.makeRequest(this.submitFrameClient_);
-  }
-
-
-  submitFrameMissing(frameId, mailboxHolder, timeWaited) {
-    this.missing_frame_count_++;
-  }
-
-  submitFrame(frameId, mailboxHolder, timeWaited) {
-    this.submit_frame_count_++;
-
-    // Trigger the submit completion callbacks here. WARNING: The
-    // Javascript-based mojo mocks are *not* re-entrant. It's OK to
-    // wait for these notifications on the next frame, but waiting
-    // within the current frame would never finish since the incoming
-    // calls would be queued until the current execution context finishes.
-    this.submitFrameClient_.onSubmitFrameTransferred(true);
-    this.submitFrameClient_.onSubmitFrameRendered();
-  }
-
-  getFrameData() {
-    if (this.pose_) {
-      this.pose_.poseIndex++;
-
-      let input_states = [];
-      for (let i = 0; i < this.input_sources_.length; ++i) {
-        input_states.push(this.input_sources_[i].getInputSourceState());
-      }
-
-      this.pose_.inputState = input_states;
-    }
-
-    // Convert current document time to monotonic time.
-    var now = window.performance.now() / 1000.0;
-    var diff = now - internals.monotonicTimeToZeroBasedDocumentTime(now);
-    now += diff;
-    now *= 1000000;
-
-    return Promise.resolve({
-      frameData: {
-        pose: this.pose_,
-        timeDelta: {
-          microseconds: now,
-        },
-        frameId: this.next_frame_id_++,
-        projectionMatrix : [1, 0, 0, 0,
-                            0, 1, 0, 0,
-                            0, 0, 1, 0,
-                            0, 0, 0, 1]
-      }
-    });
-  }
-
   initPose() {
     this.pose_ = {
       orientation: null,
@@ -536,14 +481,14 @@
     }
   }
 
-  OnInputSourceAdded(input_source) {
+  addInputSource(input_source) {
     let index = this.next_input_source_index_;
     input_source.source_id_ = index;
     this.next_input_source_index_++;
     this.input_sources_.push(input_source);
   }
 
-  OnInputSourceRemoved(input_source) {
+  removeInputSource(input_source) {
     for (let i = 0; i < this.input_sources_.length; ++i) {
       if (input_source.source_id_ == this.input_sources_[i].source_id_) {
         this.input_sources_.splice(i, 1);
@@ -553,6 +498,49 @@
   }
 }
 
+class MockXRPresentationProvider {
+  constructor() {
+    this.binding_ =
+        new mojo.Binding(device.mojom.XRPresentationProviderPtr, this);
+    this.submit_frame_count_ = 0;
+    this.missing_frame_count_ = 0;
+  }
+
+  bindProvider(request) {
+    let providerPtr = new device.mojom.XRPresentationProviderPtr();
+    let providerRequest = mojo.makeRequest(providerPtr);
+
+    this.binding_.close();
+
+    this.binding_ = new mojo.Binding(
+        device.mojom.XRPresentationProvider, this, providerRequest);
+
+    return providerPtr;
+  }
+
+  getClientRequest() {
+    this.submitFrameClient_ = new device.mojom.XRPresentationClientPtr();
+    return mojo.makeRequest(this.submitFrameClient_);
+  }
+
+
+  submitFrameMissing(frameId, mailboxHolder, timeWaited) {
+    this.missing_frame_count_++;
+  }
+
+  submitFrame(frameId, mailboxHolder, timeWaited) {
+    this.submit_frame_count_++;
+
+    // Trigger the submit completion callbacks here. WARNING: The
+    // Javascript-based mojo mocks are *not* re-entrant. It's OK to
+    // wait for these notifications on the next frame, but waiting
+    // within the current frame would never finish since the incoming
+    // calls would be queued until the current execution context finishes.
+    this.submitFrameClient_.onSubmitFrameTransferred(true);
+    this.submitFrameClient_.onSubmitFrameRendered();
+  }
+}
+
 class MockVRService {
   constructor() {
     this.bindingSet_ = new mojo.BindingSet(device.mojom.VRService);
diff --git a/third_party/abseil-cpp/BUILD.gn b/third_party/abseil-cpp/BUILD.gn
index 6005148..4d9f46c4 100644
--- a/third_party/abseil-cpp/BUILD.gn
+++ b/third_party/abseil-cpp/BUILD.gn
@@ -19,28 +19,7 @@
 }
 
 config("absl_include_config") {
-  # Using -isystem (with clang and GCC) and -imsvc (with clang-cl) instead of
-  # include_dirs (-I), so we don't need to suppress warnings coming from
-  # Abseil. Doing so would mask warnings in our own code.
-  if (is_win) {
-    if (is_clang) {
-      # clang-cl:
-      cflags = [
-        "-imsvc",
-        rebase_path(".", root_build_dir),
-      ]
-    } else {
-      # MSVC doesn't have -isystem, in that case we fallback to include_dirs and
-      # we use the warning suppression flags defined in :absl_default_cflags_cc.
-      include_dirs = [ "." ]
-    }
-  } else {
-    # GCC or clang:
-    cflags = [
-      "-isystem",
-      rebase_path(".", root_build_dir),
-    ]
-  }
+  include_dirs = [ "." ]
 }
 
 config("absl_define_config") {
diff --git a/third_party/abseil-cpp/absl/time/BUILD.gn b/third_party/abseil-cpp/absl/time/BUILD.gn
index 7f412bcc..24de31c 100644
--- a/third_party/abseil-cpp/absl/time/BUILD.gn
+++ b/third_party/abseil-cpp/absl/time/BUILD.gn
@@ -14,11 +14,21 @@
   visibility = [ "*" ]
 }
 
+config("suppress_unguarded_availability") {
+  # TODO(bugs.webrtc.org/9557): Remove -Wno-unguarded-availability when
+  # abseil will support Xcode 9.0+ (it currently supports Xcode 7.3.1+
+  # which doesn't have -Wunguarded-availability and __builtin_available).
+  cflags = [
+    "-Wno-unguarded-availability",
+  ]
+}
+
 source_set("time") {
   configs -= [ "//build/config/compiler:chromium_code" ]
   configs += [
     "//build/config/compiler:no_chromium_code",
     "//third_party/abseil-cpp:absl_default_cflags_cc",
+    ":suppress_unguarded_availability",
   ]
   public_configs = [ "//third_party/abseil-cpp:absl_include_config" ]
   sources = [
diff --git a/third_party/blink/public/BUILD.gn b/third_party/blink/public/BUILD.gn
index c6c030a0..4ae8004 100644
--- a/third_party/blink/public/BUILD.gn
+++ b/third_party/blink/public/BUILD.gn
@@ -319,7 +319,6 @@
     "platform/web_native_scroll_behavior.h",
     "platform/web_network_state_notifier.h",
     "platform/web_platform_event_listener.h",
-    "platform/web_platform_event_type.h",
     "platform/web_point.h",
     "platform/web_pointer_event.h",
     "platform/web_pointer_properties.h",
diff --git a/third_party/blink/public/platform/modules/background_fetch/background_fetch.mojom b/third_party/blink/public/platform/modules/background_fetch/background_fetch.mojom
index af3a5239..fb00f88 100644
--- a/third_party/blink/public/platform/modules/background_fetch/background_fetch.mojom
+++ b/third_party/blink/public/platform/modules/background_fetch/background_fetch.mojom
@@ -73,7 +73,8 @@
   UpdateUI(int64 service_worker_registration_id,
            string developer_id,
            string unique_id,
-           string title)
+           string? title,
+           skia.mojom.Bitmap? icon)
       => (BackgroundFetchError error);
 
   // Aborts the Background Fetch registration identified by |unique_id| and
diff --git a/third_party/blink/public/platform/platform.h b/third_party/blink/public/platform/platform.h
index ff0d114..3c22845 100644
--- a/third_party/blink/public/platform/platform.h
+++ b/third_party/blink/public/platform/platform.h
@@ -55,7 +55,6 @@
 #include "third_party/blink/public/platform/web_gamepad_listener.h"
 #include "third_party/blink/public/platform/web_gesture_device.h"
 #include "third_party/blink/public/platform/web_localized_string.h"
-#include "third_party/blink/public/platform/web_platform_event_type.h"
 #include "third_party/blink/public/platform/web_rtc_api_name.h"
 #include "third_party/blink/public/platform/web_size.h"
 #include "third_party/blink/public/platform/web_speech_synthesizer.h"
diff --git a/third_party/blink/public/platform/web_platform_event_type.h b/third_party/blink/public/platform/web_platform_event_type.h
deleted file mode 100644
index 91c1e2f5..0000000
--- a/third_party/blink/public/platform/web_platform_event_type.h
+++ /dev/null
@@ -1,19 +0,0 @@
-// Copyright 2014 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef THIRD_PARTY_BLINK_PUBLIC_PLATFORM_WEB_PLATFORM_EVENT_TYPE_H_
-#define THIRD_PARTY_BLINK_PUBLIC_PLATFORM_WEB_PLATFORM_EVENT_TYPE_H_
-
-namespace blink {
-
-enum WebPlatformEventType {
-  kWebPlatformEventTypeDeviceMotion,
-  kWebPlatformEventTypeDeviceOrientation,
-  kWebPlatformEventTypeDeviceOrientationAbsolute,
-  kWebPlatformEventTypeGamepad,
-};
-
-}  // namespace blink
-
-#endif  // THIRD_PARTY_BLINK_PUBLIC_PLATFORM_WEB_PLATFORM_EVENT_TYPE_H_
diff --git a/third_party/blink/public/platform/web_runtime_features.h b/third_party/blink/public/platform/web_runtime_features.h
index 895bd6cb..ea32357e 100644
--- a/third_party/blink/public/platform/web_runtime_features.h
+++ b/third_party/blink/public/platform/web_runtime_features.h
@@ -191,6 +191,7 @@
   BLINK_PLATFORM_EXPORT static void EnableV8ContextSnapshot(bool);
   BLINK_PLATFORM_EXPORT static void EnableAutomationControlled(bool);
   BLINK_PLATFORM_EXPORT static void EnableWorkStealingInScriptRunner(bool);
+  BLINK_PLATFORM_EXPORT static void EnableScheduledScriptStreaming(bool);
   BLINK_PLATFORM_EXPORT static void EnableStopInBackground(bool);
   BLINK_PLATFORM_EXPORT static void EnableStopNonTimersInBackground(bool);
   BLINK_PLATFORM_EXPORT static void EnablePWAFullCodeCache(bool);
diff --git a/third_party/blink/renderer/bindings/core/v8/script_streamer.cc b/third_party/blink/renderer/bindings/core/v8/script_streamer.cc
index 70f3cbd..9e56a7092 100644
--- a/third_party/blink/renderer/bindings/core/v8/script_streamer.cc
+++ b/third_party/blink/renderer/bindings/core/v8/script_streamer.cc
@@ -13,12 +13,15 @@
 #include "third_party/blink/renderer/core/dom/element.h"
 #include "third_party/blink/renderer/core/frame/settings.h"
 #include "third_party/blink/renderer/core/html/parser/text_resource_decoder.h"
+#include "third_party/blink/renderer/core/inspector/inspector_trace_events.h"
 #include "third_party/blink/renderer/core/script/classic_pending_script.h"
 #include "third_party/blink/renderer/platform/cross_thread_functional.h"
 #include "third_party/blink/renderer/platform/histogram.h"
 #include "third_party/blink/renderer/platform/instrumentation/tracing/trace_event.h"
 #include "third_party/blink/renderer/platform/loader/fetch/cached_metadata.h"
 #include "third_party/blink/renderer/platform/loader/fetch/resource.h"
+#include "third_party/blink/renderer/platform/runtime_enabled_features.h"
+#include "third_party/blink/renderer/platform/scheduler/public/background_scheduler.h"
 #include "third_party/blink/renderer/platform/scheduler/public/thread_scheduler.h"
 #include "third_party/blink/renderer/platform/shared_buffer.h"
 #include "third_party/blink/renderer/platform/wtf/deque.h"
@@ -392,6 +395,19 @@
   streaming_suppressed_ = true;
 }
 
+static void RunScriptStreamingTask(
+    std::unique_ptr<v8::ScriptCompiler::ScriptStreamingTask> task,
+    ScriptStreamer* streamer) {
+  TRACE_EVENT1(
+      "v8,devtools.timeline", "v8.parseOnBackground", "data",
+      InspectorParseScriptEvent::Data(streamer->ScriptResourceIdentifier(),
+                                      streamer->ScriptURLString()));
+  // Running the task can and will block: SourceStream::GetSomeData will get
+  // called and it will block and wait for data from the network.
+  task->Run();
+  streamer->StreamingCompleteOnBackgroundThread();
+}
+
 void ScriptStreamer::NotifyAppendData(ScriptResource* resource) {
   DCHECK(IsMainThread());
   if (streaming_suppressed_)
@@ -436,11 +452,12 @@
       }
     }
 
-    if (ScriptStreamerThread::Shared()->IsRunningTask()) {
-      // At the moment we only have one thread for running the tasks. A
-      // new task shouldn't be queued before the running task completes,
-      // because the running task can block and wait for data from the
-      // network.
+    if (!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled() &&
+        ScriptStreamerThread::Shared()->IsRunningTask()) {
+      // If scheduled script streaming is disabled, we only have one thread for
+      // running the tasks. A new task shouldn't be queued before the running
+      // task completes, because the running task can block and wait for data
+      // from the network.
       SuppressStreaming();
       RecordNotStreamingReasonHistogram(script_type_, kThreadBusy);
       RecordStartedStreamingHistogram(script_type_, 0);
@@ -476,10 +493,24 @@
       return;
     }
 
-    ScriptStreamerThread::Shared()->PostTask(
-        CrossThreadBind(&ScriptStreamerThread::RunScriptStreamingTask,
-                        WTF::Passed(std::move(script_streaming_task)),
-                        WrapCrossThreadPersistent(this)));
+    if (RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled()) {
+      // Script streaming tasks are high priority, as they can block the
+      // parser, and they can (and probably will) block during their own
+      // execution as they wait for more input.
+      //
+      // TODO(leszeks): Decrease the priority of these tasks where possible.
+      BackgroundScheduler::PostOnBackgroundThreadWithTraits(
+          FROM_HERE, {base::TaskPriority::USER_BLOCKING, base::MayBlock()},
+          CrossThreadBind(RunScriptStreamingTask,
+                          WTF::Passed(std::move(script_streaming_task)),
+                          WrapCrossThreadPersistent(this)));
+    } else {
+      ScriptStreamerThread::Shared()->PostTask(
+          CrossThreadBind(&ScriptStreamerThread::RunScriptStreamingTask,
+                          WTF::Passed(std::move(script_streaming_task)),
+                          WrapCrossThreadPersistent(this)));
+    }
+
     RecordStartedStreamingHistogram(script_type_, 1);
   }
   if (stream_)
diff --git a/third_party/blink/renderer/bindings/core/v8/script_streamer_test.cc b/third_party/blink/renderer/bindings/core/v8/script_streamer_test.cc
index 0ea5e53..6111fec 100644
--- a/third_party/blink/renderer/bindings/core/v8/script_streamer_test.cc
+++ b/third_party/blink/renderer/bindings/core/v8/script_streamer_test.cc
@@ -108,8 +108,10 @@
   }
 
   void ProcessTasksUntilStreamingComplete() {
-    while (ScriptStreamerThread::Shared()->IsRunningTask()) {
-      test::RunPendingTasks();
+    if (!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled()) {
+      while (ScriptStreamerThread::Shared()->IsRunningTask()) {
+        test::RunPendingTasks();
+      }
     }
     // Once more, because the "streaming complete" notification might only
     // now be in the task queue.
diff --git a/third_party/blink/renderer/bindings/core/v8/script_streamer_thread.cc b/third_party/blink/renderer/bindings/core/v8/script_streamer_thread.cc
index a197c9d..9051edde 100644
--- a/third_party/blink/renderer/bindings/core/v8/script_streamer_thread.cc
+++ b/third_party/blink/renderer/bindings/core/v8/script_streamer_thread.cc
@@ -11,6 +11,7 @@
 #include "third_party/blink/renderer/bindings/core/v8/script_streamer.h"
 #include "third_party/blink/renderer/core/inspector/inspector_trace_events.h"
 #include "third_party/blink/renderer/platform/instrumentation/tracing/trace_event.h"
+#include "third_party/blink/renderer/platform/runtime_enabled_features.h"
 #include "third_party/blink/renderer/platform/web_task_runner.h"
 
 namespace blink {
@@ -22,19 +23,24 @@
 static Mutex* g_mutex = nullptr;
 
 void ScriptStreamerThread::Init() {
-  DCHECK(!g_shared_thread);
-  DCHECK(IsMainThread());
-  // This is called in the main thread before any tasks are created, so no
-  // locking is needed.
-  g_mutex = new Mutex();
-  g_shared_thread = new ScriptStreamerThread();
+  // Only enabled when ScheduledScriptStreaming is disabled.
+  if (!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled()) {
+    DCHECK(!g_shared_thread);
+    DCHECK(IsMainThread());
+    // This is called in the main thread before any tasks are created, so no
+    // locking is needed.
+    g_mutex = new Mutex();
+    g_shared_thread = new ScriptStreamerThread();
+  }
 }
 
 ScriptStreamerThread* ScriptStreamerThread::Shared() {
+  DCHECK(!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled());
   return g_shared_thread;
 }
 
 void ScriptStreamerThread::PostTask(CrossThreadClosure task) {
+  DCHECK(!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled());
   DCHECK(IsMainThread());
   MutexLocker locker(mutex_);
   DCHECK(!running_task_);
@@ -60,6 +66,7 @@
 void ScriptStreamerThread::RunScriptStreamingTask(
     std::unique_ptr<v8::ScriptCompiler::ScriptStreamingTask> task,
     ScriptStreamer* streamer) {
+  DCHECK(!RuntimeEnabledFeatures::ScheduledScriptStreamingEnabled());
   TRACE_EVENT1(
       "v8,devtools.timeline", "v8.parseOnBackground", "data",
       InspectorParseScriptEvent::Data(streamer->ScriptResourceIdentifier(),
diff --git a/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.cc b/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.cc
index 292660ef..ec89538 100644
--- a/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.cc
+++ b/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.cc
@@ -581,8 +581,8 @@
                                                       size_t size,
                                                       size_t* actual_size) {
   *actual_size = WTF::Partitions::BufferActualSize(size);
-  return WTF::Partitions::BufferRealloc(old_buffer, *actual_size,
-                                        "SerializedScriptValue buffer");
+  return WTF::Partitions::BufferTryRealloc(old_buffer, *actual_size,
+                                           "SerializedScriptValue buffer");
 }
 
 void V8ScriptValueSerializer::FreeBufferMemory(void* buffer) {
diff --git a/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.h b/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.h
index 7d74e07..7d8031f 100644
--- a/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.h
+++ b/third_party/blink/renderer/bindings/core/v8/serialization/v8_script_value_serializer.h
@@ -93,6 +93,9 @@
   v8::Maybe<uint32_t> GetWasmModuleTransferId(
       v8::Isolate*,
       v8::Local<v8::WasmCompiledModule>) override;
+  // Reallocates memory at |ptr| to the new size and returns the new pointer or
+  // nullptr on failure. |actual_size| will hold the actual size of allocation
+  // requested.
   void* ReallocateBufferMemory(void* old_buffer,
                                size_t,
                                size_t* actual_size) override;
diff --git a/third_party/blink/renderer/bindings/core/v8/v8_wasm_response_extensions.cc b/third_party/blink/renderer/bindings/core/v8/v8_wasm_response_extensions.cc
index 26154ed..fd56d312 100644
--- a/third_party/blink/renderer/bindings/core/v8/v8_wasm_response_extensions.cc
+++ b/third_party/blink/renderer/bindings/core/v8/v8_wasm_response_extensions.cc
@@ -20,6 +20,103 @@
 
 namespace {
 
+// The |FetchDataLoader| for streaming compilation of WebAssembly code. The
+// received bytes get forwarded to the V8 API class |WasmStreaming|.
+class FetchDataLoaderForWasmStreaming final : public FetchDataLoader,
+                                              public BytesConsumer::Client {
+  USING_GARBAGE_COLLECTED_MIXIN(FetchDataLoaderForWasmStreaming);
+
+ public:
+  FetchDataLoaderForWasmStreaming(ScriptState* script_state,
+                                  std::shared_ptr<v8::WasmStreaming> streaming)
+      : streaming_(std::move(streaming)), script_state_(script_state) {}
+
+  void Start(BytesConsumer* consumer,
+             FetchDataLoader::Client* client) override {
+    DCHECK(!consumer_);
+    DCHECK(!client_);
+    client_ = client;
+    consumer_ = consumer;
+    consumer_->SetClient(this);
+    OnStateChange();
+  }
+
+  void OnStateChange() override {
+    while (true) {
+      // |buffer| is owned by |consumer_|.
+      const char* buffer = nullptr;
+      size_t available = 0;
+      BytesConsumer::Result result = consumer_->BeginRead(&buffer, &available);
+
+      if (result == BytesConsumer::Result::kShouldWait)
+        return;
+      if (result == BytesConsumer::Result::kOk) {
+        if (available > 0) {
+          DCHECK_NE(buffer, nullptr);
+          streaming_->OnBytesReceived(reinterpret_cast<const uint8_t*>(buffer),
+                                      available);
+        }
+        result = consumer_->EndRead(available);
+      }
+      switch (result) {
+        case BytesConsumer::Result::kShouldWait:
+          NOTREACHED();
+          return;
+        case BytesConsumer::Result::kOk: {
+          break;
+        }
+        case BytesConsumer::Result::kDone: {
+          streaming_->Finish();
+          client_->DidFetchDataLoadedCustomFormat();
+          return;
+        }
+        case BytesConsumer::Result::kError: {
+          return AbortCompilation();
+        }
+      }
+    }
+  }
+
+  String DebugName() const override { return "FetchDataLoaderForWasmModule"; }
+
+  void Cancel() override {
+    consumer_->Cancel();
+    return AbortCompilation();
+  }
+
+  void Trace(blink::Visitor* visitor) override {
+    visitor->Trace(consumer_);
+    visitor->Trace(client_);
+    visitor->Trace(script_state_);
+    FetchDataLoader::Trace(visitor);
+    BytesConsumer::Client::Trace(visitor);
+  }
+
+ private:
+  // TODO(ahaas): replace with spec-ed error types, once spec clarifies
+  // what they are.
+  void AbortCompilation() {
+    if (script_state_->ContextIsValid()) {
+      ScriptState::Scope scope(script_state_);
+      streaming_->Abort(V8ThrowException::CreateTypeError(
+          script_state_->GetIsolate(), "Could not download wasm module"));
+    } else {
+      // We are not allowed to execute a script, which indicates that we should
+      // not reject the promise of the streaming compilation. By passing no
+      // abort reason, we indicate the V8 side that the promise should not get
+      // rejected.
+      streaming_->Abort(v8::Local<v8::Value>());
+    }
+  }
+  Member<BytesConsumer> consumer_;
+  Member<FetchDataLoader::Client> client_;
+  std::shared_ptr<v8::WasmStreaming> streaming_;
+  const Member<ScriptState> script_state_;
+};
+
+// TODO(ahaas): Remove |FetchDataLoaderAsWasmModule| once the
+// |SetWasmCompileStreamingCallback| API is successfully replaced by the
+// |SetWasmStreamingCallback| API.
 class FetchDataLoaderAsWasmModule final : public FetchDataLoader,
                                           public BytesConsumer::Client {
   USING_GARBAGE_COLLECTED_MIXIN(FetchDataLoaderAsWasmModule);
@@ -42,7 +139,7 @@
 
   void OnStateChange() override {
     while (true) {
-      // {buffer} is owned by {m_consumer}.
+      // |buffer| is owned by |consumer_|.
       const char* buffer = nullptr;
       size_t available = 0;
       BytesConsumer::Result result = consumer_->BeginRead(&buffer, &available);
@@ -96,8 +193,8 @@
   // TODO(mtrofin): replace with spec-ed error types, once spec clarifies
   // what they are.
   void AbortCompilation() {
-    ScriptState::Scope scope(script_state_);
-    if (!ExecutionContext::From(script_state_)->IsContextDestroyed()) {
+    if (script_state_->ContextIsValid()) {
+      ScriptState::Scope scope(script_state_);
       builder_.Abort(V8ThrowException::CreateTypeError(
           script_state_->GetIsolate(), "Could not download wasm module"));
     } else {
@@ -135,6 +232,97 @@
   }
 };
 
+// ExceptionToAbortStreamingScope converts a possible exception to an abort
+// message for WasmStreaming instead of throwing the exception.
+//
+// All exceptions which happen in the setup of WebAssembly streaming compilation
+// have to be passed as an abort message to V8 so that V8 can reject the promise
+// associated to the streaming compilation.
+class ExceptionToAbortStreamingScope {
+  STACK_ALLOCATED();
+  WTF_MAKE_NONCOPYABLE(ExceptionToAbortStreamingScope);
+
+ public:
+  ExceptionToAbortStreamingScope(std::shared_ptr<v8::WasmStreaming> streaming,
+                                 ExceptionState& exception_state)
+      : streaming_(streaming), exception_state_(exception_state) {}
+
+  ~ExceptionToAbortStreamingScope() {
+    if (!exception_state_.HadException())
+      return;
+
+    streaming_->Abort(exception_state_.GetException());
+    exception_state_.ClearException();
+  }
+
+ private:
+  std::shared_ptr<v8::WasmStreaming> streaming_;
+  ExceptionState& exception_state_;
+};
+
+void StreamFromResponseCallback(
+    const v8::FunctionCallbackInfo<v8::Value>& args) {
+  ExceptionState exception_state(args.GetIsolate(),
+                                 ExceptionState::kExecutionContext,
+                                 "WebAssembly", "compile");
+  std::shared_ptr<v8::WasmStreaming> streaming =
+      v8::WasmStreaming::Unpack(args.GetIsolate(), args.Data());
+  ExceptionToAbortStreamingScope exception_scope(streaming, exception_state);
+
+  ScriptState* script_state = ScriptState::ForCurrentRealm(args);
+  if (!script_state->ContextIsValid()) {
+    // We do not have an execution context, we just abort streaming compilation
+    // immediately without error.
+    streaming->Abort(v8::Local<v8::Value>());
+    return;
+  }
+
+  Response* response =
+      V8Response::ToImplWithTypeCheck(args.GetIsolate(), args[0]);
+  if (!response) {
+    exception_state.ThrowTypeError(
+        "An argument must be provided, which must be a "
+        "Response or Promise<Response> object");
+    return;
+  }
+
+  if (!response->ok()) {
+    exception_state.ThrowTypeError("HTTP status code is not ok");
+    return;
+  }
+
+  if (response->MimeType() != "application/wasm") {
+    exception_state.ThrowTypeError(
+        "Incorrect response MIME type. Expected 'application/wasm'.");
+    return;
+  }
+
+  Body::BodyLocked body_locked = response->IsBodyLocked(exception_state);
+  if (body_locked == Body::BodyLocked::kBroken)
+    return;
+
+  if (body_locked == Body::BodyLocked::kLocked ||
+      response->IsBodyUsed(exception_state) == Body::BodyUsed::kUsed) {
+    DCHECK(!exception_state.HadException());
+    exception_state.ThrowTypeError(
+        "Cannot compile WebAssembly.Module from an already read Response");
+    return;
+  }
+
+  if (exception_state.HadException())
+    return;
+
+  if (!response->BodyBuffer()) {
+    exception_state.ThrowTypeError("Response object has a null body.");
+    return;
+  }
+
+  FetchDataLoaderForWasmStreaming* loader =
+      new FetchDataLoaderForWasmStreaming(script_state, streaming);
+  response->BodyBuffer()->StartLoading(loader, new WasmDataLoaderClient(),
+                                       exception_state);
+}
+
 // This callback may be entered as a promise is resolved, or directly
 // from the overload callback.
 // See
@@ -147,7 +335,7 @@
   ExceptionToRejectPromiseScope reject_promise_scope(args, exception_state);
 
   ScriptState* script_state = ScriptState::ForCurrentRealm(args);
-  if (!ExecutionContext::From(script_state)) {
+  if (!script_state->ContextIsValid()) {
     V8SetReturnValue(args, ScriptPromise().V8Value());
     return;
   }
@@ -172,13 +360,18 @@
     return;
   }
 
-  if (response->IsBodyLocked(exception_state) == Body::BodyLocked::kLocked ||
+  Body::BodyLocked body_locked = response->IsBodyLocked(exception_state);
+  if (body_locked == Body::BodyLocked::kBroken)
+    return;
+
+  if (body_locked == Body::BodyLocked::kLocked ||
       response->IsBodyUsed(exception_state) == Body::BodyUsed::kUsed) {
     DCHECK(!exception_state.HadException());
     exception_state.ThrowTypeError(
         "Cannot compile WebAssembly.Module from an already read Response");
     return;
   }
+
   if (exception_state.HadException())
     return;
 
@@ -234,6 +427,7 @@
 void WasmResponseExtensions::Initialize(v8::Isolate* isolate) {
   if (RuntimeEnabledFeatures::WebAssemblyStreamingEnabled()) {
     isolate->SetWasmCompileStreamingCallback(WasmCompileStreamingImpl);
+    isolate->SetWasmStreamingCallback(StreamFromResponseCallback);
   }
 }
 
diff --git a/third_party/blink/renderer/bindings/modules/BUILD.gn b/third_party/blink/renderer/bindings/modules/BUILD.gn
index 1fe835b..01b6956d 100644
--- a/third_party/blink/renderer/bindings/modules/BUILD.gn
+++ b/third_party/blink/renderer/bindings/modules/BUILD.gn
@@ -22,7 +22,6 @@
     "//third_party/blink/renderer/modules/background_fetch/background_fetch_click_event.idl",
     "//third_party/blink/renderer/modules/background_fetch/background_fetch_event.idl",
     "//third_party/blink/renderer/modules/background_fetch/background_fetch_fail_event.idl",
-    "//third_party/blink/renderer/modules/background_fetch/background_fetched_event.idl",
     "//third_party/blink/renderer/modules/background_sync/sync_event.idl",
     "//third_party/blink/renderer/modules/cookie_store/cookie_change_event.idl",
     "//third_party/blink/renderer/modules/cookie_store/extendable_cookie_change_event.idl",
diff --git a/third_party/blink/renderer/core/html/canvas/html_canvas_element.cc b/third_party/blink/renderer/core/html/canvas/html_canvas_element.cc
index 71b6440..4f46e9bf 100644
--- a/third_party/blink/renderer/core/html/canvas/html_canvas_element.cc
+++ b/third_party/blink/renderer/core/html/canvas/html_canvas_element.cc
@@ -415,9 +415,13 @@
 
     if (!LowLatencyEnabled())
       canvas2d_bridge_->FinalizeFrame();
+  }
 
-    if (LowLatencyEnabled() && !dirty_rect_.IsEmpty() &&
-        GetOrCreateCanvasResourceProvider(kPreferAcceleration)) {
+  if (LowLatencyEnabled() && !dirty_rect_.IsEmpty()) {
+    AccelerationHint hint =
+        Is2d() ? kPreferNoAcceleration : kPreferAcceleration;
+    if (GetOrCreateCanvasResourceProvider(hint)) {
+      ResourceProvider()->TryEnableSingleBuffering();
       // Push a frame
       base::TimeTicks start_time = WTF::CurrentTimeTicks();
       scoped_refptr<CanvasResource> canvas_resource =
@@ -1138,9 +1142,8 @@
 scoped_refptr<Image> HTMLCanvasElement::CopiedImage(
     SourceDrawingBuffer source_buffer,
     AccelerationHint hint) {
-  if (SurfaceLayerBridge()) {
+  if (PlaceholderFrame())
     return PlaceholderFrame()->Bitmap();
-  }
 
   if (!IsPaintable())
     return nullptr;
diff --git a/third_party/blink/renderer/core/paint/BUILD.gn b/third_party/blink/renderer/core/paint/BUILD.gn
index f4738106..b7663d0d 100644
--- a/third_party/blink/renderer/core/paint/BUILD.gn
+++ b/third_party/blink/renderer/core/paint/BUILD.gn
@@ -216,6 +216,8 @@
     "svg_inline_text_box_painter.h",
     "svg_mask_painter.cc",
     "svg_mask_painter.h",
+    "svg_model_object_painter.cc",
+    "svg_model_object_painter.h",
     "svg_paint_context.cc",
     "svg_paint_context.h",
     "svg_root_inline_box_painter.cc",
diff --git a/third_party/blink/renderer/core/paint/paint_layer_painter.cc b/third_party/blink/renderer/core/paint/paint_layer_painter.cc
index 8760ba53..dcf00696 100644
--- a/third_party/blink/renderer/core/paint/paint_layer_painter.cc
+++ b/third_party/blink/renderer/core/paint/paint_layer_painter.cc
@@ -817,9 +817,12 @@
     PaintLayerFlags paint_flags,
     const PaintLayerFragment& fragment,
     const LayoutSize& subpixel_accumulation) {
-  // Now do a paint with the root layer shifted to be us. The cull rect
-  // is infinite to simplify invalidation of clip property node changes across
-  // transform boundaries.
+  // Now do a paint with the root layer shifted to be us.
+
+  // We do not apply cull rect optimizations across transforms for two reasons:
+  //   1) Performance: We can optimize transform changes by not repainting.
+  //   2) Complexity: Difficulty updating clips when ancestor transforms change.
+  // For these reasons, we use an infinite dirty rect here.
   PaintLayerPaintingInfo new_paint_info(
       &paint_layer_, LayoutRect(LayoutRect::InfiniteIntRect()),
       painting_info.GetGlobalPaintFlags(), subpixel_accumulation);
diff --git a/third_party/blink/renderer/core/paint/svg_container_painter.cc b/third_party/blink/renderer/core/paint/svg_container_painter.cc
index 13e54af..58e7b810 100644
--- a/third_party/blink/renderer/core/paint/svg_container_painter.cc
+++ b/third_party/blink/renderer/core/paint/svg_container_painter.cc
@@ -13,6 +13,7 @@
 #include "third_party/blink/renderer/core/paint/object_painter.h"
 #include "third_party/blink/renderer/core/paint/paint_info.h"
 #include "third_party/blink/renderer/core/paint/svg_foreign_object_painter.h"
+#include "third_party/blink/renderer/core/paint/svg_model_object_painter.h"
 #include "third_party/blink/renderer/core/paint/svg_paint_context.h"
 #include "third_party/blink/renderer/core/svg/svg_svg_element.h"
 
@@ -24,11 +25,6 @@
       !layout_svg_container_.SelfWillPaint())
     return;
 
-  FloatRect bounding_box =
-      layout_svg_container_.VisualRectInLocalSVGCoordinates();
-  // LayoutSVGHiddenContainer's visual rect is always empty but we need to
-  // paint its descendants.
-
   // Spec: An empty viewBox on the <svg> element disables rendering.
   DCHECK(layout_svg_container_.GetElement());
   if (IsSVGSVGElement(*layout_svg_container_.GetElement()) &&
@@ -37,17 +33,19 @@
 
   PaintInfo paint_info_before_filtering(paint_info);
 
-  // Content underneath transforms applies an infinite cull rect. This is
-  // to simplify invalidation of clip property node changes across transform
-  // boundaries.
+  if (SVGModelObjectPainter(layout_svg_container_)
+          .CullRectSkipsPainting(paint_info_before_filtering)) {
+    return;
+  }
+
+  // We do not apply cull rect optimizations across transforms for two reasons:
+  //   1) Performance: We can optimize transform changes by not repainting.
+  //   2) Complexity: Difficulty updating clips when ancestor transforms change.
+  // This is why we use an infinite cull rect if there is a transform. Non-svg
+  // content, does this in PaintLayerPainter::PaintSingleFragment.
   if (layout_svg_container_.StyleRef().HasTransform()) {
     paint_info_before_filtering.ApplyInfiniteCullRect();
   } else {
-    if (!layout_svg_container_.IsSVGHiddenContainer() &&
-        !paint_info.GetCullRect().IntersectsCullRect(
-            layout_svg_container_.LocalToSVGParentTransform(), bounding_box))
-      return;
-
     paint_info_before_filtering.UpdateCullRect(
         layout_svg_container_.LocalToSVGParentTransform());
   }
@@ -94,20 +92,14 @@
     }
   }
 
-  if (paint_info_before_filtering.phase != PaintPhase::kForeground)
-    return;
+  SVGModelObjectPainter(layout_svg_container_)
+      .PaintOutline(paint_info_before_filtering);
 
-  if (layout_svg_container_.Style()->OutlineWidth() &&
-      layout_svg_container_.Style()->Visibility() == EVisibility::kVisible) {
-    PaintInfo outline_paint_info(paint_info_before_filtering);
-    outline_paint_info.phase = PaintPhase::kSelfOutlineOnly;
-    ObjectPainter(layout_svg_container_)
-        .PaintOutline(outline_paint_info, LayoutPoint(bounding_box.Location()));
-  }
-
-  if (paint_info_before_filtering.IsPrinting())
+  if (paint_info_before_filtering.IsPrinting() &&
+      paint_info_before_filtering.phase == PaintPhase::kForeground) {
     ObjectPainter(layout_svg_container_)
         .AddPDFURLRectIfNeeded(paint_info_before_filtering, LayoutPoint());
+  }
 }
 
 }  // namespace blink
diff --git a/third_party/blink/renderer/core/paint/svg_image_painter.cc b/third_party/blink/renderer/core/paint/svg_image_painter.cc
index 977d09f..93d3611f 100644
--- a/third_party/blink/renderer/core/paint/svg_image_painter.cc
+++ b/third_party/blink/renderer/core/paint/svg_image_painter.cc
@@ -4,20 +4,15 @@
 
 #include "third_party/blink/renderer/core/paint/svg_image_painter.h"
 
-#include "third_party/blink/renderer/core/frame/local_frame.h"
-#include "third_party/blink/renderer/core/frame/local_frame_view.h"
 #include "third_party/blink/renderer/core/layout/layout_image_resource.h"
 #include "third_party/blink/renderer/core/layout/svg/layout_svg_image.h"
-#include "third_party/blink/renderer/core/page/chrome_client.h"
-#include "third_party/blink/renderer/core/page/page.h"
-#include "third_party/blink/renderer/core/paint/object_painter.h"
 #include "third_party/blink/renderer/core/paint/paint_info.h"
+#include "third_party/blink/renderer/core/paint/svg_model_object_painter.h"
 #include "third_party/blink/renderer/core/paint/svg_paint_context.h"
 #include "third_party/blink/renderer/core/svg/graphics/svg_image.h"
 #include "third_party/blink/renderer/core/svg/svg_image_element.h"
 #include "third_party/blink/renderer/platform/graphics/graphics_context.h"
 #include "third_party/blink/renderer/platform/graphics/paint/drawing_recorder.h"
-#include "third_party/blink/renderer/platform/graphics/paint/paint_record.h"
 #include "third_party/blink/renderer/platform/graphics/scoped_interpolation_quality.h"
 
 namespace blink {
@@ -28,13 +23,14 @@
       !layout_svg_image_.ImageResource()->HasImage())
     return;
 
-  FloatRect bounding_box = layout_svg_image_.VisualRectInLocalSVGCoordinates();
-  if (!paint_info.GetCullRect().IntersectsCullRect(
-          layout_svg_image_.LocalToSVGParentTransform(), bounding_box))
-    return;
-
   PaintInfo paint_info_before_filtering(paint_info);
-  // Images cannot have children so do not call updateCullRect.
+
+  if (SVGModelObjectPainter(layout_svg_image_)
+          .CullRectSkipsPainting(paint_info_before_filtering)) {
+    return;
+  }
+  // Images cannot have children so do not call UpdateCullRect.
+
   SVGTransformContext transform_context(
       paint_info_before_filtering, layout_svg_image_,
       layout_svg_image_.LocalToSVGParentTransform());
@@ -52,12 +48,8 @@
     }
   }
 
-  if (layout_svg_image_.Style()->OutlineWidth()) {
-    PaintInfo outline_paint_info(paint_info_before_filtering);
-    outline_paint_info.phase = PaintPhase::kSelfOutlineOnly;
-    ObjectPainter(layout_svg_image_)
-        .PaintOutline(outline_paint_info, LayoutPoint(bounding_box.Location()));
-  }
+  SVGModelObjectPainter(layout_svg_image_)
+      .PaintOutline(paint_info_before_filtering);
 }
 
 void SVGImagePainter::PaintForeground(const PaintInfo& paint_info) {
diff --git a/third_party/blink/renderer/core/paint/svg_model_object_painter.cc b/third_party/blink/renderer/core/paint/svg_model_object_painter.cc
new file mode 100644
index 0000000..956f01c
--- /dev/null
+++ b/third_party/blink/renderer/core/paint/svg_model_object_painter.cc
@@ -0,0 +1,46 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "third_party/blink/renderer/core/paint/svg_model_object_painter.h"
+
+#include "third_party/blink/renderer/core/layout/svg/layout_svg_model_object.h"
+#include "third_party/blink/renderer/core/paint/object_painter.h"
+#include "third_party/blink/renderer/core/paint/paint_info.h"
+
+namespace blink {
+
+bool SVGModelObjectPainter::CullRectSkipsPainting(const PaintInfo& paint_info) {
+  // We do not apply cull rect optimizations across transforms for two reasons:
+  //   1) Performance: We can optimize transform changes by not repainting.
+  //   2) Complexity: Difficulty updating clips when ancestor transforms change.
+  // For these reasons, we do not cull painting if there is a transform.
+  if (layout_svg_model_object_.StyleRef().HasTransform())
+    return false;
+
+  // LayoutSVGHiddenContainer's visual rect is always empty but we need to
+  // paint its descendants so we cannot skip painting.
+  if (layout_svg_model_object_.IsSVGHiddenContainer())
+    return false;
+
+  return !paint_info.GetCullRect().IntersectsCullRect(
+      layout_svg_model_object_.LocalToSVGParentTransform(),
+      layout_svg_model_object_.VisualRectInLocalSVGCoordinates());
+}
+
+void SVGModelObjectPainter::PaintOutline(const PaintInfo& paint_info) {
+  if (paint_info.phase != PaintPhase::kForeground)
+    return;
+  if (layout_svg_model_object_.Style()->Visibility() != EVisibility::kVisible)
+    return;
+  if (!layout_svg_model_object_.Style()->OutlineWidth())
+    return;
+
+  PaintInfo outline_paint_info(paint_info);
+  outline_paint_info.phase = PaintPhase::kSelfOutlineOnly;
+  auto visual_rect = layout_svg_model_object_.VisualRectInLocalSVGCoordinates();
+  ObjectPainter(layout_svg_model_object_)
+      .PaintOutline(outline_paint_info, LayoutPoint(visual_rect.Location()));
+}
+
+}  // namespace blink
diff --git a/third_party/blink/renderer/core/paint/svg_model_object_painter.h b/third_party/blink/renderer/core/paint/svg_model_object_painter.h
new file mode 100644
index 0000000..6f3df6d
--- /dev/null
+++ b/third_party/blink/renderer/core/paint/svg_model_object_painter.h
@@ -0,0 +1,35 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef THIRD_PARTY_BLINK_RENDERER_CORE_PAINT_SVG_MODEL_OBJECT_PAINTER_H_
+#define THIRD_PARTY_BLINK_RENDERER_CORE_PAINT_SVG_MODEL_OBJECT_PAINTER_H_
+
+#include "third_party/blink/renderer/platform/wtf/allocator.h"
+
+namespace blink {
+
+struct PaintInfo;
+class LayoutSVGModelObject;
+
+class SVGModelObjectPainter {
+  STACK_ALLOCATED();
+
+ public:
+  SVGModelObjectPainter(const LayoutSVGModelObject& layout_svg_model_object)
+      : layout_svg_model_object_(layout_svg_model_object) {}
+
+  // If the object is outside the cull rect, painting can be skipped in most
+  // cases. An important exception is when there is a transform style: see the
+  // comment in the implementation.
+  bool CullRectSkipsPainting(const PaintInfo&);
+
+  void PaintOutline(const PaintInfo&);
+
+ private:
+  const LayoutSVGModelObject& layout_svg_model_object_;
+};
+
+}  // namespace blink
+
+#endif  // THIRD_PARTY_BLINK_RENDERER_CORE_PAINT_SVG_MODEL_OBJECT_PAINTER_H_
diff --git a/third_party/blink/renderer/core/paint/svg_shape_painter.cc b/third_party/blink/renderer/core/paint/svg_shape_painter.cc
index ea42ac8b..2c8ed98 100644
--- a/third_party/blink/renderer/core/paint/svg_shape_painter.cc
+++ b/third_party/blink/renderer/core/paint/svg_shape_painter.cc
@@ -11,9 +11,9 @@
 #include "third_party/blink/renderer/core/layout/svg/svg_marker_data.h"
 #include "third_party/blink/renderer/core/layout/svg/svg_resources.h"
 #include "third_party/blink/renderer/core/layout/svg/svg_resources_cache.h"
-#include "third_party/blink/renderer/core/paint/object_painter.h"
 #include "third_party/blink/renderer/core/paint/paint_info.h"
 #include "third_party/blink/renderer/core/paint/svg_container_painter.h"
+#include "third_party/blink/renderer/core/paint/svg_model_object_painter.h"
 #include "third_party/blink/renderer/core/paint/svg_paint_context.h"
 #include "third_party/blink/renderer/platform/graphics/graphics_context_state_saver.h"
 #include "third_party/blink/renderer/platform/graphics/paint/drawing_recorder.h"
@@ -46,13 +46,14 @@
       layout_svg_shape_.IsShapeEmpty())
     return;
 
-  FloatRect bounding_box = layout_svg_shape_.VisualRectInLocalSVGCoordinates();
-  if (!paint_info.GetCullRect().IntersectsCullRect(
-          layout_svg_shape_.LocalSVGTransform(), bounding_box))
-    return;
-
   PaintInfo paint_info_before_filtering(paint_info);
-  // Shapes cannot have children so do not call updateCullRect.
+
+  if (SVGModelObjectPainter(layout_svg_shape_)
+          .CullRectSkipsPainting(paint_info_before_filtering)) {
+    return;
+  }
+  // Shapes cannot have children so do not call UpdateCullRect.
+
   SVGTransformContext transform_context(paint_info_before_filtering,
                                         layout_svg_shape_,
                                         layout_svg_shape_.LocalSVGTransform());
@@ -124,7 +125,8 @@
             }
             break;
           case PT_MARKERS:
-            PaintMarkers(paint_context.GetPaintInfo(), bounding_box);
+            PaintMarkers(paint_context.GetPaintInfo(),
+                         layout_svg_shape_.VisualRectInLocalSVGCoordinates());
             break;
           default:
             NOTREACHED();
@@ -134,12 +136,8 @@
     }
   }
 
-  if (layout_svg_shape_.Style()->OutlineWidth()) {
-    PaintInfo outline_paint_info(paint_info_before_filtering);
-    outline_paint_info.phase = PaintPhase::kSelfOutlineOnly;
-    ObjectPainter(layout_svg_shape_)
-        .PaintOutline(outline_paint_info, LayoutPoint(bounding_box.Location()));
-  }
+  SVGModelObjectPainter(layout_svg_shape_)
+      .PaintOutline(paint_info_before_filtering);
 }
 
 class PathWithTemporaryWindingRule {
diff --git a/third_party/blink/renderer/devtools/front_end/profiler/HeapSnapshotView.js b/third_party/blink/renderer/devtools/front_end/profiler/HeapSnapshotView.js
index 57bf5fb7..b020a21 100644
--- a/third_party/blink/renderer/devtools/front_end/profiler/HeapSnapshotView.js
+++ b/third_party/blink/renderer/devtools/front_end/profiler/HeapSnapshotView.js
@@ -209,6 +209,9 @@
    */
   async linkifyObject(snapshotObjectId) {
     const heapProfilerModel = this._profile.heapProfilerModel();
+    // heapProfilerModel is null if snapshot was loaded from file
+    if (!heapProfilerModel)
+      return null;
     const remoteObject = await heapProfilerModel.objectForSnapshotObjectId(String(snapshotObjectId), 'link');
     if (!remoteObject || remoteObject.type !== 'function')
       return null;
diff --git a/third_party/blink/renderer/modules/background_fetch/BUILD.gn b/third_party/blink/renderer/modules/background_fetch/BUILD.gn
index 831d97f..0cd04774 100644
--- a/third_party/blink/renderer/modules/background_fetch/BUILD.gn
+++ b/third_party/blink/renderer/modules/background_fetch/BUILD.gn
@@ -32,8 +32,6 @@
     "background_fetch_type_converters.h",
     "background_fetch_update_event.cc",
     "background_fetch_update_event.h",
-    "background_fetched_event.cc",
-    "background_fetched_event.h",
     "service_worker_global_scope_background_fetch.h",
     "service_worker_registration_background_fetch.cc",
     "service_worker_registration_background_fetch.h",
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.cc b/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.cc
index 2ed7682..ca9a66d 100644
--- a/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.cc
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.cc
@@ -69,10 +69,17 @@
 void BackgroundFetchBridge::UpdateUI(const String& developer_id,
                                      const String& unique_id,
                                      const String& title,
+                                     const SkBitmap& icon,
                                      UpdateUICallback callback) {
+  if (title.IsNull() && icon.isNull()) {
+    std::move(callback).Run(
+        mojom::blink::BackgroundFetchError::INVALID_ARGUMENT);
+    return;
+  }
+
   GetService()->UpdateUI(
       GetSupplementable()->WebRegistration()->RegistrationId(), developer_id,
-      unique_id, title, std::move(callback));
+      unique_id, title, icon, std::move(callback));
 }
 
 void BackgroundFetchBridge::GetRegistration(const String& developer_id,
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h b/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h
index 652f8ff..8e48fa0 100644
--- a/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h
@@ -60,11 +60,12 @@
   void GetIconDisplaySize(GetIconDisplaySizeCallback callback);
 
   // Updates the user interface for the Background Fetch identified by
-  // |unique_id| with the updated |title|. Will invoke the |callback| when the
-  // interface has been requested to update.
+  // |unique_id| with the updated |title| or |icon|. Will invoke the |callback|
+  // when the interface has been requested to update.
   void UpdateUI(const String& developer_id,
                 const String& unique_id,
                 const String& title,
+                const SkBitmap& icon,
                 UpdateUICallback callback);
 
   // Aborts the active Background Fetch for |unique_id|. Will invoke the
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.cc b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.cc
index c1aeb79..8f57469d7 100644
--- a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.cc
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.cc
@@ -7,11 +7,13 @@
 #include "third_party/blink/public/platform/modules/background_fetch/web_background_fetch_settled_fetch.h"
 #include "third_party/blink/renderer/bindings/core/v8/script_promise_resolver.h"
 #include "third_party/blink/renderer/core/dom/dom_exception.h"
+#include "third_party/blink/renderer/core/execution_context/execution_context.h"
 #include "third_party/blink/renderer/core/fetch/request.h"
 #include "third_party/blink/renderer/core/fetch/response.h"
 #include "third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h"
+#include "third_party/blink/renderer/modules/background_fetch/background_fetch_icon_loader.h"
 #include "third_party/blink/renderer/modules/background_fetch/background_fetch_settled_fetch.h"
-#include "third_party/blink/renderer/modules/background_fetch/background_fetched_event_init.h"
+#include "third_party/blink/renderer/modules/background_fetch/background_fetch_update_ui_options.h"
 #include "third_party/blink/renderer/modules/event_modules_names.h"
 #include "third_party/blink/renderer/platform/bindings/script_state.h"
 
@@ -30,17 +32,20 @@
     WaitUntilObserver* observer,
     ServiceWorkerRegistration* registration)
     : BackgroundFetchSettledEvent(type, initializer, unique_id, observer),
-      registration_(registration) {}
+      registration_(registration),
+      loader_(new BackgroundFetchIconLoader) {}
 
 BackgroundFetchUpdateEvent::~BackgroundFetchUpdateEvent() = default;
 
 void BackgroundFetchUpdateEvent::Trace(blink::Visitor* visitor) {
   visitor->Trace(registration_);
+  visitor->Trace(loader_);
   BackgroundFetchSettledEvent::Trace(visitor);
 }
 
-ScriptPromise BackgroundFetchUpdateEvent::updateUI(ScriptState* script_state,
-                                                   const String& title) {
+ScriptPromise BackgroundFetchUpdateEvent::updateUI(
+    ScriptState* script_state,
+    const BackgroundFetchUpdateUIOptions& ui_options) {
   if (!registration_) {
     // Return a Promise that will never settle when a developer calls this
     // method on a BackgroundFetchedEvent instance they created themselves.
@@ -48,17 +53,36 @@
   }
   DCHECK(!unique_id_.IsEmpty());
 
+  if (!ui_options.hasTitle() && ui_options.icons().IsEmpty()) {
+    // Nothing to update, just return a resolved promise.
+    ScriptPromise::CastUndefined(script_state);
+  }
+
   ScriptPromiseResolver* resolver = ScriptPromiseResolver::Create(script_state);
   ScriptPromise promise = resolver->Promise();
 
-  BackgroundFetchBridge::From(registration_)
-      ->UpdateUI(id(), unique_id_, title,
-                 WTF::Bind(&BackgroundFetchUpdateEvent::DidUpdateUI,
-                           WrapPersistent(this), WrapPersistent(resolver)));
+  if (ui_options.icons().IsEmpty()) {
+    DidGetIcon(resolver, ui_options.title(), SkBitmap());
+  } else {
+    loader_->Start(
+        BackgroundFetchBridge::From(registration_),
+        ExecutionContext::From(script_state), ui_options.icons(),
+        WTF::Bind(&BackgroundFetchUpdateEvent::DidGetIcon, WrapPersistent(this),
+                  WrapPersistent(resolver), ui_options.title()));
+  }
 
   return promise;
 }
 
+void BackgroundFetchUpdateEvent::DidGetIcon(ScriptPromiseResolver* resolver,
+                                            const String& title,
+                                            const SkBitmap& icon) {
+  BackgroundFetchBridge::From(registration_)
+      ->UpdateUI(id(), unique_id_, title, icon,
+                 WTF::Bind(&BackgroundFetchUpdateEvent::DidUpdateUI,
+                           WrapPersistent(this), WrapPersistent(resolver)));
+}
+
 void BackgroundFetchUpdateEvent::DidUpdateUI(
     ScriptPromiseResolver* resolver,
     mojom::blink::BackgroundFetchError error) {
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.h b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.h
index 5683234..c060dcff 100644
--- a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.h
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.h
@@ -11,6 +11,9 @@
 
 namespace blink {
 
+class BackgroundFetchIconLoader;
+class BackgroundFetchUpdateUIOptions;
+
 // Event for interacting with fetch requests that have completed.
 class MODULES_EXPORT BackgroundFetchUpdateEvent final
     : public BackgroundFetchSettledEvent {
@@ -37,7 +40,8 @@
   ~BackgroundFetchUpdateEvent() override;
 
   // Web Exposed method defined in the IDL file.
-  ScriptPromise updateUI(ScriptState* script_state, const String& title);
+  ScriptPromise updateUI(ScriptState* script_state,
+                         const BackgroundFetchUpdateUIOptions& ui_options);
 
   void Trace(blink::Visitor* visitor) override;
 
@@ -53,10 +57,15 @@
                              WaitUntilObserver* observer,
                              ServiceWorkerRegistration* registration);
 
+  void DidGetIcon(ScriptPromiseResolver* resolver,
+                  const String& title,
+                  const SkBitmap& icon);
+
   void DidUpdateUI(ScriptPromiseResolver* resolver,
                    mojom::blink::BackgroundFetchError error);
 
   Member<ServiceWorkerRegistration> registration_;
+  Member<BackgroundFetchIconLoader> loader_;
 };
 
 }  // namespace blink
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.idl b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.idl
index c9c7948..f4faa5c 100644
--- a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.idl
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_event.idl
@@ -10,5 +10,5 @@
     RuntimeEnabled=BackgroundFetch
 ]
 interface BackgroundFetchUpdateEvent : BackgroundFetchSettledEvent {
-    [CallWith=ScriptState] Promise<void> updateUI(DOMString title);
+    [CallWith=ScriptState] Promise<void> updateUI(BackgroundFetchUpdateUIOptions title);
 };
\ No newline at end of file
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetch_update_ui_options.idl b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_ui_options.idl
new file mode 100644
index 0000000..549c1b70
--- /dev/null
+++ b/third_party/blink/renderer/modules/background_fetch/background_fetch_update_ui_options.idl
@@ -0,0 +1,13 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+// https://wicg.github.io/background-fetch/#backgroundfetchupdateevent
+
+// TODO(crbug.com/865063): Fix names after the standard is updated.
+// Spec Bug: https://github.com/WICG/background-fetch/issues/86
+
+dictionary BackgroundFetchUpdateUIOptions {
+    DOMString? title;
+    sequence<ImageResource> icons = [];
+};
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.cc b/third_party/blink/renderer/modules/background_fetch/background_fetched_event.cc
deleted file mode 100644
index 5416bd2..0000000
--- a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.cc
+++ /dev/null
@@ -1,107 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "third_party/blink/renderer/modules/background_fetch/background_fetched_event.h"
-
-#include "third_party/blink/public/platform/modules/background_fetch/web_background_fetch_settled_fetch.h"
-#include "third_party/blink/renderer/bindings/core/v8/script_promise_resolver.h"
-#include "third_party/blink/renderer/core/dom/dom_exception.h"
-#include "third_party/blink/renderer/core/fetch/request.h"
-#include "third_party/blink/renderer/core/fetch/response.h"
-#include "third_party/blink/renderer/modules/background_fetch/background_fetch_bridge.h"
-#include "third_party/blink/renderer/modules/background_fetch/background_fetch_settled_fetch.h"
-#include "third_party/blink/renderer/modules/background_fetch/background_fetched_event_init.h"
-#include "third_party/blink/renderer/modules/event_modules_names.h"
-#include "third_party/blink/renderer/platform/bindings/script_state.h"
-
-namespace blink {
-
-BackgroundFetchedEvent::BackgroundFetchedEvent(
-    const AtomicString& type,
-    const BackgroundFetchedEventInit& initializer)
-    : BackgroundFetchEvent(type, initializer, nullptr /* observer */),
-      fetches_(initializer.fetches()) {}
-
-BackgroundFetchedEvent::BackgroundFetchedEvent(
-    const AtomicString& type,
-    const BackgroundFetchedEventInit& initializer,
-    const String& unique_id,
-    const WebVector<WebBackgroundFetchSettledFetch>& fetches,
-    ScriptState* script_state,
-    WaitUntilObserver* observer,
-    ServiceWorkerRegistration* registration)
-    : BackgroundFetchEvent(type, initializer, observer),
-      unique_id_(unique_id),
-      registration_(registration) {
-  fetches_.ReserveInitialCapacity(fetches.size());
-  for (const WebBackgroundFetchSettledFetch& fetch : fetches) {
-    auto* settled_fetch = BackgroundFetchSettledFetch::Create(
-        Request::Create(script_state, fetch.request),
-        Response::Create(script_state, fetch.response));
-
-    fetches_.push_back(settled_fetch);
-  }
-}
-
-BackgroundFetchedEvent::~BackgroundFetchedEvent() = default;
-
-HeapVector<Member<BackgroundFetchSettledFetch>>
-BackgroundFetchedEvent::fetches() const {
-  return fetches_;
-}
-
-ScriptPromise BackgroundFetchedEvent::updateUI(ScriptState* script_state,
-                                               const String& title) {
-  if (!registration_) {
-    // Return a Promise that will never settle when a developer calls this
-    // method on a BackgroundFetchedEvent instance they created themselves.
-    return ScriptPromise();
-  }
-  DCHECK(!unique_id_.IsEmpty());
-
-  ScriptPromiseResolver* resolver = ScriptPromiseResolver::Create(script_state);
-  ScriptPromise promise = resolver->Promise();
-
-  BackgroundFetchBridge::From(registration_)
-      ->UpdateUI(id(), unique_id_, title,
-                 WTF::Bind(&BackgroundFetchedEvent::DidUpdateUI,
-                           WrapPersistent(this), WrapPersistent(resolver)));
-
-  return promise;
-}
-
-void BackgroundFetchedEvent::DidUpdateUI(
-    ScriptPromiseResolver* resolver,
-    mojom::blink::BackgroundFetchError error) {
-  switch (error) {
-    case mojom::blink::BackgroundFetchError::NONE:
-    case mojom::blink::BackgroundFetchError::INVALID_ID:
-      resolver->Resolve();
-      return;
-    case mojom::blink::BackgroundFetchError::STORAGE_ERROR:
-      resolver->Reject(
-          DOMException::Create(DOMExceptionCode::kAbortError,
-                               "Failed to update UI due to I/O error."));
-      return;
-    case mojom::blink::BackgroundFetchError::SERVICE_WORKER_UNAVAILABLE:
-    case mojom::blink::BackgroundFetchError::DUPLICATED_DEVELOPER_ID:
-    case mojom::blink::BackgroundFetchError::INVALID_ARGUMENT:
-      // Not applicable for this callback.
-      break;
-  }
-
-  NOTREACHED();
-}
-
-const AtomicString& BackgroundFetchedEvent::InterfaceName() const {
-  return EventNames::BackgroundFetchedEvent;
-}
-
-void BackgroundFetchedEvent::Trace(blink::Visitor* visitor) {
-  visitor->Trace(fetches_);
-  visitor->Trace(registration_);
-  BackgroundFetchEvent::Trace(visitor);
-}
-
-}  // namespace blink
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.h b/third_party/blink/renderer/modules/background_fetch/background_fetched_event.h
deleted file mode 100644
index d86e7392..0000000
--- a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.h
+++ /dev/null
@@ -1,87 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef THIRD_PARTY_BLINK_RENDERER_MODULES_BACKGROUND_FETCH_BACKGROUND_FETCHED_EVENT_H_
-#define THIRD_PARTY_BLINK_RENDERER_MODULES_BACKGROUND_FETCH_BACKGROUND_FETCHED_EVENT_H_
-
-#include "third_party/blink/public/platform/modules/background_fetch/background_fetch.mojom-blink.h"
-#include "third_party/blink/public/platform/web_vector.h"
-#include "third_party/blink/renderer/bindings/core/v8/script_promise.h"
-#include "third_party/blink/renderer/modules/background_fetch/background_fetch_event.h"
-#include "third_party/blink/renderer/modules/modules_export.h"
-#include "third_party/blink/renderer/platform/heap/handle.h"
-#include "third_party/blink/renderer/platform/wtf/text/atomic_string.h"
-
-namespace blink {
-
-class BackgroundFetchSettledFetch;
-class BackgroundFetchedEventInit;
-class ScriptPromiseResolver;
-class ScriptState;
-class ServiceWorkerRegistration;
-struct WebBackgroundFetchSettledFetch;
-
-class MODULES_EXPORT BackgroundFetchedEvent final
-    : public BackgroundFetchEvent {
-  DEFINE_WRAPPERTYPEINFO();
-
- public:
-  static BackgroundFetchedEvent* Create(
-      const AtomicString& type,
-      const BackgroundFetchedEventInit& initializer) {
-    return new BackgroundFetchedEvent(type, initializer);
-  }
-
-  static BackgroundFetchedEvent* Create(
-      const AtomicString& type,
-      const BackgroundFetchedEventInit& initializer,
-      const String& unique_id,
-      const WebVector<WebBackgroundFetchSettledFetch>& fetches,
-      ScriptState* script_state,
-      WaitUntilObserver* observer,
-      ServiceWorkerRegistration* registration) {
-    return new BackgroundFetchedEvent(type, initializer, unique_id, fetches,
-                                      script_state, observer, registration);
-  }
-
-  ~BackgroundFetchedEvent() override;
-
-  // Web Exposed attribute defined in the IDL file.
-  HeapVector<Member<BackgroundFetchSettledFetch>> fetches() const;
-
-  // Web Exposed method defined in the IDL file.
-  ScriptPromise updateUI(ScriptState* script_state, const String& title);
-
-  // ExtendableEvent interface.
-  const AtomicString& InterfaceName() const override;
-
-  void Trace(blink::Visitor* visitor) override;
-
- private:
-  BackgroundFetchedEvent(const AtomicString& type,
-                         const BackgroundFetchedEventInit& initializer);
-  BackgroundFetchedEvent(
-      const AtomicString& type,
-      const BackgroundFetchedEventInit& initializer,
-      const String& unique_id,
-      const WebVector<WebBackgroundFetchSettledFetch>& fetches,
-      ScriptState* script_state,
-      WaitUntilObserver* observer,
-      ServiceWorkerRegistration* registration);
-
-  void DidUpdateUI(ScriptPromiseResolver* resolver,
-                   mojom::blink::BackgroundFetchError error);
-
-  // Globally unique ID for the registration, generated in content/. Used to
-  // distinguish registrations in case a developer re-uses |developer_id_|s. Not
-  // exposed to JavaScript.
-  String unique_id_;
-
-  HeapVector<Member<BackgroundFetchSettledFetch>> fetches_;
-  Member<ServiceWorkerRegistration> registration_;
-};
-
-}  // namespace blink
-
-#endif  // THIRD_PARTY_BLINK_RENDERER_MODULES_BACKGROUND_FETCH_BACKGROUND_FETCHED_EVENT_H_
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.idl b/third_party/blink/renderer/modules/background_fetch/background_fetched_event.idl
deleted file mode 100644
index 2226821..0000000
--- a/third_party/blink/renderer/modules/background_fetch/background_fetched_event.idl
+++ /dev/null
@@ -1,15 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-// https://wicg.github.io/background-fetch/#background-fetch-end-event
-
-[
-    Constructor(DOMString type, BackgroundFetchedEventInit init),
-    Exposed=ServiceWorker,
-    RuntimeEnabled=BackgroundFetch
-] interface BackgroundFetchedEvent : BackgroundFetchEvent {
-    readonly attribute FrozenArray<BackgroundFetchSettledFetch> fetches;
-
-    [CallWith=ScriptState] Promise<void> updateUI(DOMString title);
-};
diff --git a/third_party/blink/renderer/modules/background_fetch/background_fetched_event_init.idl b/third_party/blink/renderer/modules/background_fetch/background_fetched_event_init.idl
deleted file mode 100644
index 68d6f684..0000000
--- a/third_party/blink/renderer/modules/background_fetch/background_fetched_event_init.idl
+++ /dev/null
@@ -1,9 +0,0 @@
-// Copyright 2017 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-// https://wicg.github.io/background-fetch/#background-fetch-end-event
-
-dictionary BackgroundFetchedEventInit : BackgroundFetchEventInit {
-    required sequence<BackgroundFetchSettledFetch> fetches;
-};
diff --git a/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d_test.cc b/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d_test.cc
index 5c679f9..208ff8c 100644
--- a/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d_test.cc
+++ b/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d_test.cc
@@ -120,16 +120,15 @@
     return CanvasElement().GetGPUMemoryUsage();
   }
   void DrawSomething() {
-    Canvas2DLayerBridge* bridge = CanvasElement().GetCanvas2DLayerBridge();
-    bridge->DidDraw(FloatRect(0, 0, 1, 1));
-    bridge->FinalizeFrame();
+    CanvasElement().DidDraw();
+    CanvasElement().FinalizeFrame();
     // Grabbing an image forces a flush
-    bridge->NewImageSnapshot(kPreferAcceleration);
+    CanvasElement().CopiedImage(kBackBuffer, kPreferAcceleration);
   }
 
-  void CreateContext(OpacityMode,
-                     String color_space = String(),
-                     LinearPixelMathState = kLinearPixelMathDisabled);
+  enum LatencyMode { kNormalLatency, kLowLatency };
+
+  void CreateContext(OpacityMode, LatencyMode = kNormalLatency);
   ScriptState* GetScriptState() {
     return ToScriptStateForMainWorld(canvas_element_->GetFrame());
   }
@@ -185,19 +184,12 @@
       opaque_bitmap_(IntSize(10, 10), kOpaqueBitmap),
       alpha_bitmap_(IntSize(10, 10), kTransparentBitmap) {}
 
-void CanvasRenderingContext2DTest::CreateContext(
-    OpacityMode opacity_mode,
-    String color_space,
-    LinearPixelMathState LinearPixelMath_state) {
+void CanvasRenderingContext2DTest::CreateContext(OpacityMode opacity_mode,
+                                                 LatencyMode latency_mode) {
   String canvas_type("2d");
   CanvasContextCreationAttributesCore attributes;
   attributes.alpha = opacity_mode == kNonOpaque;
-  if (!color_space.IsEmpty()) {
-    attributes.color_space = color_space;
-    if (LinearPixelMath_state == kLinearPixelMathEnabled) {
-      attributes.pixel_format = "float16";
-    }
-  }
+  attributes.low_latency = latency_mode == kLowLatency;
   canvas_element_->GetCanvasRenderingContext(canvas_type, attributes);
 }
 
@@ -1209,4 +1201,22 @@
   EXPECT_FALSE(layer->NeedsCompositingInputsUpdate());
 }
 
+TEST_F(CanvasRenderingContext2DTest, LowLatencyIsSingleBuffered) {
+  CreateContext(kNonOpaque, kLowLatency);
+  // No need to set-up the layer bridge when testing low latency mode.
+  DrawSomething();
+  auto frame1_resource =
+      CanvasElement()
+          .GetOrCreateCanvasResourceProvider(kPreferNoAcceleration)
+          ->ProduceFrame();
+  EXPECT_TRUE(!!frame1_resource);
+  DrawSomething();
+  auto frame2_resource =
+      CanvasElement()
+          .GetOrCreateCanvasResourceProvider(kPreferNoAcceleration)
+          ->ProduceFrame();
+  EXPECT_TRUE(!!frame2_resource);
+  EXPECT_EQ(frame1_resource.get(), frame2_resource.get());
+}
+
 }  // namespace blink
diff --git a/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.cc b/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.cc
index 8a15c24..01f17d13 100644
--- a/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.cc
+++ b/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.cc
@@ -88,10 +88,4 @@
   return last_device_orientation_data_.Get();
 }
 
-WebPlatformEventType DeviceOrientationDispatcher::GetWebPlatformEventType()
-    const {
-  return (absolute_) ? kWebPlatformEventTypeDeviceOrientationAbsolute
-                     : kWebPlatformEventTypeDeviceOrientation;
-}
-
 }  // namespace blink
diff --git a/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.h b/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.h
index 62a9453..b8e25cb0 100644
--- a/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.h
+++ b/third_party/blink/renderer/modules/device_orientation/device_orientation_dispatcher.h
@@ -33,7 +33,6 @@
 
 #include "base/memory/scoped_refptr.h"
 #include "third_party/blink/public/platform/modules/device_orientation/web_device_orientation_listener.h"
-#include "third_party/blink/public/platform/web_platform_event_type.h"
 #include "third_party/blink/renderer/core/frame/platform_event_dispatcher.h"
 #include "third_party/blink/renderer/platform/heap/handle.h"
 
@@ -74,8 +73,6 @@
   void StartListening(LocalFrame* frame) override;
   void StopListening() override;
 
-  WebPlatformEventType GetWebPlatformEventType() const;
-
   const bool absolute_;
   Member<DeviceOrientationData> last_device_orientation_data_;
   std::unique_ptr<DeviceOrientationEventPump> event_pump_;
diff --git a/third_party/blink/renderer/modules/modules_idl_files.gni b/third_party/blink/renderer/modules/modules_idl_files.gni
index c5efe86..6ff80c5 100644
--- a/third_party/blink/renderer/modules/modules_idl_files.gni
+++ b/third_party/blink/renderer/modules/modules_idl_files.gni
@@ -74,7 +74,6 @@
           "background_fetch/background_fetch_settled_fetch.idl",
           "background_fetch/background_fetch_settled_fetches.idl",
           "background_fetch/background_fetch_update_event.idl",
-          "background_fetch/background_fetched_event.idl",
           "background_sync/sync_event.idl",
           "background_sync/sync_manager.idl",
           "battery/battery_manager.idl",
@@ -469,7 +468,7 @@
           "background_fetch/background_fetch_fail_event_init.idl",
           "background_fetch/background_fetch_options.idl",
           "background_fetch/background_fetch_settled_event_init.idl",
-          "background_fetch/background_fetched_event_init.idl",
+          "background_fetch/background_fetch_update_ui_options.idl",
           "background_sync/sync_event_init.idl",
           "bluetooth/bluetooth_le_scan_filter_init.idl",
           "bluetooth/request_device_options.idl",
diff --git a/third_party/blink/renderer/modules/vr/vr_display.cc b/third_party/blink/renderer/modules/vr/vr_display.cc
index a221211..d49ebfc 100644
--- a/third_party/blink/renderer/modules/vr/vr_display.cc
+++ b/third_party/blink/renderer/modules/vr/vr_display.cc
@@ -244,7 +244,7 @@
 
     pending_magic_window_vsync_ = false;
     pending_presenting_vsync_ = true;
-    vr_presentation_provider_->GetFrameData(
+    vr_presentation_data_provider_->GetFrameData(
         WTF::Bind(&VRDisplay::OnPresentingVSync, WrapWeakPersistent(this)));
 
     DVLOG(2) << __FUNCTION__ << " done: pending_presenting_vsync_="
@@ -502,17 +502,26 @@
 void VRDisplay::OnRequestSessionReturned(
     device::mojom::blink::XRSessionPtr session) {
   pending_present_request_ = false;
-  if (session && session->connection) {
-    vr_presentation_provider_.Bind(std::move(session->connection->provider));
+  if (session) {
+    DCHECK(session->submit_frame_sink);
+    vr_presentation_data_provider_.Bind(std::move(session->data_provider));
+    // The presentation provider error handler can trigger if a device is
+    // disconnected from the system. This can happen if, for example, an HMD is
+    // unplugged.
+    vr_presentation_data_provider_.set_connection_error_handler(
+        WTF::Bind(&VRDisplay::OnPresentationProviderConnectionError,
+                  WrapWeakPersistent(this)));
+    vr_presentation_provider_.Bind(
+        std::move(session->submit_frame_sink->provider));
     vr_presentation_provider_.set_connection_error_handler(
         WTF::Bind(&VRDisplay::OnPresentationProviderConnectionError,
                   WrapWeakPersistent(this)));
 
     frame_transport_ = new XRFrameTransport();
     frame_transport_->BindSubmitFrameClient(
-        std::move(session->connection->client_request));
+        std::move(session->submit_frame_sink->client_request));
     frame_transport_->SetTransportOptions(
-        std::move(session->connection->transport_options));
+        std::move(session->submit_frame_sink->transport_options));
 
     this->BeginPresent();
   } else {
@@ -529,11 +538,11 @@
 
 void VRDisplay::OnMagicWindowRequestReturned(
     device::mojom::blink::XRSessionPtr session) {
-  if (!session || !session->magic_window_provider) {
-    // System does not support any kind of magic window.
+  if (!session) {
+    // System does not support any kind of session.
     return;
   }
-  magic_window_provider_.Bind(std::move(session->magic_window_provider));
+  magic_window_provider_.Bind(std::move(session->data_provider));
   RequestVSync();
 }
 
@@ -611,7 +620,7 @@
   }
   is_presenting_ = true;
   // Call RequestVSync to switch from the (internal) document rAF to the
-  // VrPresentationProvider GetFrameData rate.
+  // XRPresentationProvider GetFrameData rate.
   RequestVSync();
   ReportPresentationResult(PresentationResult::kSuccess);
 
@@ -1038,6 +1047,7 @@
            << " pending_magic_window_vsync_=" << pending_magic_window_vsync_
            << " pending_presenting_vsync_=" << pending_presenting_vsync_;
   vr_presentation_provider_.reset();
+  vr_presentation_data_provider_.reset();
   if (is_presenting_) {
     ForceExitPresent();
   }
diff --git a/third_party/blink/renderer/modules/vr/vr_display.h b/third_party/blink/renderer/modules/vr/vr_display.h
index bfd14698..4c6da38 100644
--- a/third_party/blink/renderer/modules/vr/vr_display.h
+++ b/third_party/blink/renderer/modules/vr/vr_display.h
@@ -217,14 +217,15 @@
   bool did_log_getFrameData_ = false;
   bool did_log_requestPresent_ = false;
 
-  device::mojom::blink::VRMagicWindowProviderPtr magic_window_provider_;
+  device::mojom::blink::XRFrameDataProviderPtr magic_window_provider_;
 
   device::mojom::blink::VRDisplayHostPtr display_;
 
   bool present_image_needs_copy_ = false;
 
   mojo::Binding<device::mojom::blink::VRDisplayClient> display_client_binding_;
-  device::mojom::blink::VRPresentationProviderPtr vr_presentation_provider_;
+  device::mojom::blink::XRFrameDataProviderPtr vr_presentation_data_provider_;
+  device::mojom::blink::XRPresentationProviderPtr vr_presentation_provider_;
 
   HeapDeque<Member<ScriptPromiseResolver>> pending_present_resolvers_;
 };
diff --git a/third_party/blink/renderer/modules/xr/xr_device.cc b/third_party/blink/renderer/modules/xr/xr_device.cc
index 7925e02..edd484b 100644
--- a/third_party/blink/renderer/modules/xr/xr_device.cc
+++ b/third_party/blink/renderer/modules/xr/xr_device.cc
@@ -199,9 +199,6 @@
     return;
   }
 
-  if (session_ptr->magic_window_provider)
-    magic_window_provider_.Bind(std::move(session_ptr->magic_window_provider));
-
   XRPresentationContext* output_context = nullptr;
   if (options.hasOutputContext()) {
     output_context = options.outputContext();
@@ -218,8 +215,10 @@
   sessions_.insert(session);
 
   if (options.immersive()) {
-    frameProvider()->BeginImmersiveSession(session,
-                                           std::move(session_ptr->connection));
+    frameProvider()->BeginImmersiveSession(session, std::move(session_ptr));
+  } else {
+    magic_window_provider_.Bind(std::move(session_ptr->data_provider));
+    enviroment_provider_.Bind(std::move(session_ptr->enviroment_provider));
   }
 
   resolver->Resolve(session);
diff --git a/third_party/blink/renderer/modules/xr/xr_device.h b/third_party/blink/renderer/modules/xr/xr_device.h
index f4db8e21..2e83621 100644
--- a/third_party/blink/renderer/modules/xr/xr_device.h
+++ b/third_party/blink/renderer/modules/xr/xr_device.h
@@ -54,10 +54,14 @@
   const device::mojom::blink::VRDisplayHostPtr& xrDisplayHostPtr() const {
     return display_;
   }
-  const device::mojom::blink::VRMagicWindowProviderPtr&
-  xrMagicWindowProviderPtr() const {
+  const device::mojom::blink::XRFrameDataProviderPtr& xrMagicWindowProviderPtr()
+      const {
     return magic_window_provider_;
   }
+  const device::mojom::blink::XREnviromentIntegrationProviderPtr&
+  xrEnviromentProviderPtr() const {
+    return enviroment_provider_;
+  }
   const device::mojom::blink::VRDisplayInfoPtr& xrDisplayInfoPtr() const {
     return display_info_;
   }
@@ -107,7 +111,8 @@
   // Indicates whether we've already logged a request for an immersive session.
   bool did_log_request_immersive_session_ = false;
 
-  device::mojom::blink::VRMagicWindowProviderPtr magic_window_provider_;
+  device::mojom::blink::XRFrameDataProviderPtr magic_window_provider_;
+  device::mojom::blink::XREnviromentIntegrationProviderPtr enviroment_provider_;
   device::mojom::blink::VRDisplayHostPtr display_;
   device::mojom::blink::VRDisplayInfoPtr display_info_;
   unsigned int display_info_id_ = 0;
diff --git a/third_party/blink/renderer/modules/xr/xr_frame_provider.cc b/third_party/blink/renderer/modules/xr/xr_frame_provider.cc
index 8a5a4e9..79b42f7 100644
--- a/third_party/blink/renderer/modules/xr/xr_frame_provider.cc
+++ b/third_party/blink/renderer/modules/xr/xr_frame_provider.cc
@@ -92,25 +92,29 @@
 
 void XRFrameProvider::BeginImmersiveSession(
     XRSession* session,
-    device::mojom::blink::XRPresentationConnectionPtr connection) {
+    device::mojom::blink::XRSessionPtr session_ptr) {
   // Make sure the session is indeed an immersive one.
   DCHECK(session && session->immersive());
 
   // Ensure we can only have one immersive session at a time.
   DCHECK(!immersive_session_);
-  DCHECK(connection);
+  DCHECK(session_ptr->data_provider);
+  DCHECK(session_ptr->submit_frame_sink);
 
   immersive_session_ = session;
 
-  presentation_provider_.Bind(std::move(connection->provider));
+  immersive_data_provider_.Bind(std::move(session_ptr->data_provider));
+
+  presentation_provider_.Bind(
+      std::move(session_ptr->submit_frame_sink->provider));
   presentation_provider_.set_connection_error_handler(
       WTF::Bind(&XRFrameProvider::OnPresentationProviderConnectionError,
                 WrapWeakPersistent(this)));
 
   frame_transport_->BindSubmitFrameClient(
-      std::move(connection->client_request));
+      std::move(session_ptr->submit_frame_sink->client_request));
   frame_transport_->SetTransportOptions(
-      std::move(connection->transport_options));
+      std::move(session_ptr->submit_frame_sink->transport_options));
   frame_transport_->PresentChange();
 }
 
@@ -131,6 +135,7 @@
 
 void XRFrameProvider::OnPresentationProviderConnectionError() {
   presentation_provider_.reset();
+  immersive_data_provider_.reset();
   if (vsync_connection_failed_)
     return;
   immersive_session_->ForceEnd();
@@ -147,10 +152,8 @@
   immersive_session_ = nullptr;
   pending_immersive_vsync_ = false;
   frame_id_ = -1;
-
-  if (presentation_provider_.is_bound()) {
-    presentation_provider_.reset();
-  }
+  presentation_provider_.reset();
+  immersive_data_provider_.reset();
 
   frame_transport_ = new XRFrameTransport();
 
@@ -200,7 +203,7 @@
 
   pending_immersive_vsync_ = true;
 
-  presentation_provider_->GetFrameData(WTF::Bind(
+  immersive_data_provider_->GetFrameData(WTF::Bind(
       &XRFrameProvider::OnImmersiveFrameData, WrapWeakPersistent(this)));
 }
 
@@ -527,6 +530,7 @@
 
 void XRFrameProvider::Dispose() {
   presentation_provider_.reset();
+  immersive_data_provider_.reset();
   // TODO(bajones): Do something for outstanding frame requests?
 }
 
diff --git a/third_party/blink/renderer/modules/xr/xr_frame_provider.h b/third_party/blink/renderer/modules/xr/xr_frame_provider.h
index 159a4d55..c483e03 100644
--- a/third_party/blink/renderer/modules/xr/xr_frame_provider.h
+++ b/third_party/blink/renderer/modules/xr/xr_frame_provider.h
@@ -27,11 +27,10 @@
   explicit XRFrameProvider(XRDevice*);
 
   XRSession* immersive_session() const { return immersive_session_; }
-  device::mojom::blink::VRSubmitFrameClientPtr GetSubmitFrameClient();
+  device::mojom::blink::XRPresentationClientPtr GetSubmitFrameClient();
 
-  void BeginImmersiveSession(
-      XRSession* session,
-      device::mojom::blink::XRPresentationConnectionPtr connection);
+  void BeginImmersiveSession(XRSession* session,
+                             device::mojom::blink::XRSessionPtr session_ptr);
   void OnImmersiveSessionEnded();
 
   void RequestFrame(XRSession*);
@@ -66,8 +65,8 @@
   // Non-immersive Sessions which have requested a frame update.
   HeapVector<Member<XRSession>> requesting_sessions_;
 
-  device::mojom::blink::VRPresentationProviderPtr presentation_provider_;
-  device::mojom::blink::VRMagicWindowProviderPtr magic_window_provider_;
+  device::mojom::blink::XRPresentationProviderPtr presentation_provider_;
+  device::mojom::blink::XRFrameDataProviderPtr immersive_data_provider_;
   device::mojom::blink::VRPosePtr frame_pose_;
 
   // This frame ID is XR-specific and is used to track when frames arrive at the
diff --git a/third_party/blink/renderer/modules/xr/xr_session.cc b/third_party/blink/renderer/modules/xr/xr_session.cc
index d838ef8..b3ca9f0 100644
--- a/third_party/blink/renderer/modules/xr/xr_session.cc
+++ b/third_party/blink/renderer/modules/xr/xr_session.cc
@@ -312,7 +312,7 @@
 
   // TODO(https://crbug.com/845520): Promise should be rejected if session
   // is deleted.
-  device_->xrMagicWindowProviderPtr()->RequestHitTest(
+  device_->xrEnviromentProviderPtr()->RequestHitTest(
       std::move(ray),
       WTF::Bind(&XRSession::OnHitTestResults, WrapWeakPersistent(this),
                 WrapPersistent(resolver)));
@@ -550,8 +550,8 @@
     DVLOG(2) << __FUNCTION__ << ": got angle=" << output_angle;
   }
 
-  if (device_->xrMagicWindowProviderPtr()) {
-    device_->xrMagicWindowProviderPtr()->UpdateSessionGeometry(
+  if (device_->xrEnviromentProviderPtr()) {
+    device_->xrEnviromentProviderPtr()->UpdateSessionGeometry(
         IntSize(output_width_, output_height_),
         display::Display::DegreesToRotation(output_angle));
   }
diff --git a/third_party/blink/renderer/platform/exported/web_runtime_features.cc b/third_party/blink/renderer/platform/exported/web_runtime_features.cc
index e3383f4..a176ed33 100644
--- a/third_party/blink/renderer/platform/exported/web_runtime_features.cc
+++ b/third_party/blink/renderer/platform/exported/web_runtime_features.cc
@@ -524,6 +524,10 @@
   RuntimeEnabledFeatures::SetWorkStealingInScriptRunnerEnabled(enable);
 }
 
+void WebRuntimeFeatures::EnableScheduledScriptStreaming(bool enable) {
+  RuntimeEnabledFeatures::SetScheduledScriptStreamingEnabled(enable);
+}
+
 void WebRuntimeFeatures::EnableStopInBackground(bool enable) {
   RuntimeEnabledFeatures::SetStopInBackgroundEnabled(enable);
 }
diff --git a/third_party/blink/renderer/platform/graphics/canvas_resource_provider.cc b/third_party/blink/renderer/platform/graphics/canvas_resource_provider.cc
index 4685fcd2..8d951cd 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_resource_provider.cc
+++ b/third_party/blink/renderer/platform/graphics/canvas_resource_provider.cc
@@ -152,6 +152,7 @@
 
   ~CanvasResourceProviderTextureGpuMemoryBuffer() override = default;
   bool SupportsDirectCompositing() const override { return true; }
+  bool SupportsSingleBuffering() const override { return true; }
 
  private:
   scoped_refptr<CanvasResource> CreateResource() final {
@@ -263,6 +264,7 @@
 
   ~CanvasResourceProviderRamGpuMemoryBuffer() override = default;
   bool SupportsDirectCompositing() const override { return true; }
+  bool SupportsSingleBuffering() const override { return true; }
 
  private:
   scoped_refptr<CanvasResource> CreateResource() final {
@@ -318,6 +320,7 @@
   }
   ~CanvasResourceProviderSharedBitmap() override = default;
   bool SupportsDirectCompositing() const override { return true; }
+  bool SupportsSingleBuffering() const override { return true; }
 
  private:
   scoped_refptr<CanvasResource> CreateResource() final {
@@ -684,6 +687,7 @@
   canvas_image_provider_.reset();
   xform_canvas_ = nullptr;
   surface_ = nullptr;
+  single_buffer_ = nullptr;
 }
 
 uint32_t CanvasResourceProvider::ContentUniqueID() const {
@@ -721,6 +725,11 @@
 }
 
 scoped_refptr<CanvasResource> CanvasResourceProvider::NewOrRecycledResource() {
+  if (IsSingleBuffered()) {
+    if (!single_buffer_)
+      single_buffer_ = CreateResource();
+    return single_buffer_;
+  }
   if (recycled_resources_.size()) {
     scoped_refptr<CanvasResource> resource =
         std::move(recycled_resources_.back());
@@ -730,4 +739,11 @@
   return CreateResource();
 }
 
+void CanvasResourceProvider::TryEnableSingleBuffering() {
+  if (IsSingleBuffered() || !SupportsSingleBuffering())
+    return;
+  SetResourceRecyclingEnabled(false);
+  is_single_buffered_ = true;
+}
+
 }  // namespace blink
diff --git a/third_party/blink/renderer/platform/graphics/canvas_resource_provider.h b/third_party/blink/renderer/platform/graphics/canvas_resource_provider.h
index 661aa5d..12866715 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_resource_provider.h
+++ b/third_party/blink/renderer/platform/graphics/canvas_resource_provider.h
@@ -100,11 +100,26 @@
   virtual bool IsValid() const = 0;
   virtual bool IsAccelerated() const = 0;
   virtual bool SupportsDirectCompositing() const = 0;
+  virtual bool SupportsSingleBuffering() const { return false; }
   uint32_t ContentUniqueID() const;
   CanvasResourceDispatcher* ResourceDispatcher() {
     return resource_dispatcher_.get();
   }
 
+  // Indicates that the compositing path is single buffered, meaning that
+  // ProduceFrame() return a reference to the same resource each time, which
+  // implies that Producing an animation frame may overwrite the resource used
+  // by the previous frame. This results in graphics updates skipping the
+  // queue, thus reducing latency, but with the possible side effects of
+  // tearring (in cases where the resource is scanned out directly) and
+  // irregular frame rate.
+  bool IsSingleBuffered() { return is_single_buffered_; }
+
+  // Attempt to enable single buffering mode on this resource provider.  May
+  // fail if the CanvasResourcePRovider subclass does not support this mode of
+  // operation.
+  void TryEnableSingleBuffering();
+
   void RecycleResource(scoped_refptr<CanvasResource>);
   void SetResourceRecyclingEnabled(bool);
   void ClearRecycledResources();
@@ -189,6 +204,9 @@
   WTF::Vector<scoped_refptr<CanvasResource>> recycled_resources_;
   bool resource_recycling_enabled_ = true;
 
+  bool is_single_buffered_ = false;
+  scoped_refptr<CanvasResource> single_buffer_;
+
   base::WeakPtrFactory<CanvasResourceProvider> weak_ptr_factory_;
 
   DISALLOW_COPY_AND_ASSIGN(CanvasResourceProvider);
diff --git a/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.cc b/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.cc
index 2db3ef5..a0ace7b 100644
--- a/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.cc
+++ b/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.cc
@@ -30,24 +30,24 @@
 }
 
 void XRFrameTransport::SetTransportOptions(
-    device::mojom::blink::VRDisplayFrameTransportOptionsPtr transport_options) {
+    device::mojom::blink::XRPresentationTransportOptionsPtr transport_options) {
   transport_options_ = std::move(transport_options);
 }
 
 void XRFrameTransport::BindSubmitFrameClient(
-    device::mojom::blink::VRSubmitFrameClientRequest request) {
+    device::mojom::blink::XRPresentationClientRequest request) {
   submit_frame_client_binding_.Close();
   submit_frame_client_binding_.Bind(std::move(request));
 }
 
 bool XRFrameTransport::DrawingIntoSharedBuffer() {
   switch (transport_options_->transport_method) {
-    case device::mojom::blink::VRDisplayFrameTransportMethod::
+    case device::mojom::blink::XRPresentationTransportMethod::
         SUBMIT_AS_TEXTURE_HANDLE:
-    case device::mojom::blink::VRDisplayFrameTransportMethod::
+    case device::mojom::blink::XRPresentationTransportMethod::
         SUBMIT_AS_MAILBOX_HOLDER:
       return false;
-    case device::mojom::blink::VRDisplayFrameTransportMethod::
+    case device::mojom::blink::XRPresentationTransportMethod::
         DRAW_INTO_TEXTURE_MAILBOX:
       return true;
     default:
@@ -85,7 +85,7 @@
 }
 
 void XRFrameTransport::FrameSubmitMissing(
-    device::mojom::blink::VRPresentationProvider* vr_presentation_provider,
+    device::mojom::blink::XRPresentationProvider* vr_presentation_provider,
     gpu::gles2::GLES2Interface* gl,
     int16_t vr_frame_id) {
   TRACE_EVENT0("gpu", __FUNCTION__);
@@ -95,7 +95,7 @@
 }
 
 void XRFrameTransport::FrameSubmit(
-    device::mojom::blink::VRPresentationProvider* vr_presentation_provider,
+    device::mojom::blink::XRPresentationProvider* vr_presentation_provider,
     gpu::gles2::GLES2Interface* gl,
     DrawingBuffer::Client* drawing_buffer_client,
     scoped_refptr<Image> image_ref,
@@ -105,7 +105,7 @@
   DCHECK(transport_options_);
 
   if (transport_options_->transport_method ==
-      device::mojom::blink::VRDisplayFrameTransportMethod::
+      device::mojom::blink::XRPresentationTransportMethod::
           SUBMIT_AS_TEXTURE_HANDLE) {
 #if defined(OS_WIN)
     // Currently, we assume that this transport needs a copy.
@@ -143,7 +143,7 @@
     NOTIMPLEMENTED();
 #endif
   } else if (transport_options_->transport_method ==
-             device::mojom::blink::VRDisplayFrameTransportMethod::
+             device::mojom::blink::XRPresentationTransportMethod::
                  SUBMIT_AS_MAILBOX_HOLDER) {
     // Currently, this transport assumes we don't need to make a separate copy
     // of the canvas content.
@@ -188,7 +188,7 @@
         frame_wait_time_);
     TRACE_EVENT_END0("gpu", "XRFrameTransport::SubmitFrame");
   } else if (transport_options_->transport_method ==
-             device::mojom::blink::VRDisplayFrameTransportMethod::
+             device::mojom::blink::XRPresentationTransportMethod::
                  DRAW_INTO_TEXTURE_MAILBOX) {
     TRACE_EVENT0("gpu", "XRFrameTransport::SubmitFrameDrawnIntoTexture");
     gpu::SyncToken sync_token;
diff --git a/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.h b/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.h
index 1af00bce..57dc839 100644
--- a/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.h
+++ b/third_party/blink/renderer/platform/graphics/gpu/xr_frame_transport.h
@@ -30,25 +30,25 @@
 
 class PLATFORM_EXPORT XRFrameTransport final
     : public GarbageCollectedFinalized<XRFrameTransport>,
-      public device::mojom::blink::VRSubmitFrameClient {
+      public device::mojom::blink::XRPresentationClient {
  public:
   explicit XRFrameTransport();
   ~XRFrameTransport() override;
 
   void BindSubmitFrameClient(
-      device::mojom::blink::VRSubmitFrameClientRequest request);
+      device::mojom::blink::XRPresentationClientRequest request);
 
   void PresentChange();
 
   void SetTransportOptions(
-      device::mojom::blink::VRDisplayFrameTransportOptionsPtr);
+      device::mojom::blink::XRPresentationTransportOptionsPtr);
 
   bool DrawingIntoSharedBuffer();
 
   // Call before finalizing the frame's image snapshot.
   void FramePreImage(gpu::gles2::GLES2Interface*);
 
-  void FrameSubmit(device::mojom::blink::VRPresentationProvider*,
+  void FrameSubmit(device::mojom::blink::XRPresentationProvider*,
                    gpu::gles2::GLES2Interface*,
                    DrawingBuffer::Client*,
                    scoped_refptr<Image> image_ref,
@@ -56,7 +56,7 @@
                    int16_t vr_frame_id,
                    bool needs_copy);
 
-  void FrameSubmitMissing(device::mojom::blink::VRPresentationProvider*,
+  void FrameSubmitMissing(device::mojom::blink::XRPresentationProvider*,
                           gpu::gles2::GLES2Interface*,
                           int16_t vr_frame_id);
 
@@ -68,12 +68,12 @@
   WTF::TimeDelta WaitForGpuFenceReceived();
   void CallPreviousFrameCallback();
 
-  // VRSubmitFrameClient
+  // XRPresentationClient
   void OnSubmitFrameTransferred(bool success) override;
   void OnSubmitFrameRendered() override;
   void OnSubmitFrameGpuFence(const gfx::GpuFenceHandle&) override;
 
-  mojo::Binding<device::mojom::blink::VRSubmitFrameClient>
+  mojo::Binding<device::mojom::blink::XRPresentationClient>
       submit_frame_client_binding_;
 
   // Used to keep the image alive until the next frame if using
@@ -91,7 +91,7 @@
   bool waiting_for_previous_frame_fence_ = false;
   std::unique_ptr<gfx::GpuFence> previous_frame_fence_;
 
-  device::mojom::blink::VRDisplayFrameTransportOptionsPtr transport_options_;
+  device::mojom::blink::XRPresentationTransportOptionsPtr transport_options_;
 
   std::unique_ptr<GpuMemoryBufferImageCopy> frame_copier_;
 };
diff --git a/third_party/blink/renderer/platform/heap/gc_info.cc b/third_party/blink/renderer/platform/heap/gc_info.cc
index 1033c65..6c515bc 100644
--- a/third_party/blink/renderer/platform/heap/gc_info.cc
+++ b/third_party/blink/renderer/platform/heap/gc_info.cc
@@ -49,7 +49,7 @@
 }
 
 void GCInfoTable::EnsureGCInfoIndex(const GCInfo* gc_info,
-                                    size_t* gc_info_index_slot) {
+                                    uint32_t* gc_info_index_slot) {
   DCHECK(gc_info);
   DCHECK(gc_info_index_slot);
 
@@ -64,14 +64,13 @@
   if (*gc_info_index_slot)
     return;
 
-  int index = ++current_index_;
-  size_t gc_info_index = static_cast<size_t>(index);
+  uint32_t gc_info_index = ++current_index_;
   CHECK(gc_info_index < GCInfoTable::kMaxIndex);
   if (current_index_ >= limit_)
     Resize();
 
   table_[gc_info_index] = gc_info;
-  ReleaseStore(reinterpret_cast<int*>(gc_info_index_slot), index);
+  ReleaseStore(gc_info_index_slot, gc_info_index);
 }
 
 void GCInfoTable::Resize() {
diff --git a/third_party/blink/renderer/platform/heap/gc_info.h b/third_party/blink/renderer/platform/heap/gc_info.h
index b6e7e42..a5c138e 100644
--- a/third_party/blink/renderer/platform/heap/gc_info.h
+++ b/third_party/blink/renderer/platform/heap/gc_info.h
@@ -44,7 +44,7 @@
 };
 
 #if DCHECK_IS_ON()
-PLATFORM_EXPORT void AssertObjectHasGCInfo(const void*, size_t gc_info_index);
+PLATFORM_EXPORT void AssertObjectHasGCInfo(const void*, uint32_t gc_info_index);
 #endif
 
 class PLATFORM_EXPORT GCInfoTable {
@@ -56,14 +56,14 @@
   // of the Oilpan GC Clang plugin, there appear to be at most about 6,000
   // types. Thus 14 bits should be more than twice as many bits as we will ever
   // need.
-  static constexpr size_t kMaxIndex = 1 << 14;
+  static constexpr uint32_t kMaxIndex = 1 << 14;
 
   // Sets up a singleton table that can be acquired using Get().
   static void CreateGlobalTable();
 
   static GCInfoTable& Get() { return *global_table_; }
 
-  inline const GCInfo* GCInfoFromIndex(size_t index) {
+  inline const GCInfo* GCInfoFromIndex(uint32_t index) {
     DCHECK_GE(index, 1u);
     DCHECK(index < kMaxIndex);
     DCHECK(table_);
@@ -72,9 +72,9 @@
     return info;
   }
 
-  void EnsureGCInfoIndex(const GCInfo*, size_t*);
+  void EnsureGCInfoIndex(const GCInfo*, uint32_t*);
 
-  size_t GcInfoIndex() { return current_index_; }
+  uint32_t GcInfoIndex() { return current_index_; }
 
  private:
   FRIEND_TEST_ALL_PREFIXES(GCInfoTest, InitialEmpty);
@@ -95,10 +95,10 @@
 
   // GCInfo indices start from 1 for heap objects, with 0 being treated
   // specially as the index for freelist entries and large heap objects.
-  size_t current_index_ = 0;
+  uint32_t current_index_ = 0;
 
   // The limit (exclusive) of the currently allocated table.
-  size_t limit_ = 0;
+  uint32_t limit_ = 0;
 
   Mutex table_mutex_;
 };
@@ -108,14 +108,14 @@
 template <typename T>
 struct GCInfoAtBaseType {
   STATIC_ONLY(GCInfoAtBaseType);
-  static size_t Index() {
+  static uint32_t Index() {
     static_assert(sizeof(T), "T must be fully defined");
     static const GCInfo kGcInfo = {
         TraceTrait<T>::Trace,          FinalizerTrait<T>::Finalize,
         NameTrait<T>::GetName,         FinalizerTrait<T>::kNonTrivialFinalizer,
         std::is_polymorphic<T>::value,
     };
-    static size_t gc_info_index = 0;
+    static uint32_t gc_info_index = 0;
     if (!AcquireLoad(&gc_info_index))
       GCInfoTable::Get().EnsureGCInfoIndex(&kGcInfo, &gc_info_index);
     DCHECK_GE(gc_info_index, 1u);
@@ -144,7 +144,7 @@
 template <typename T>
 struct GCInfoTrait {
   STATIC_ONLY(GCInfoTrait);
-  static size_t Index() {
+  static uint32_t Index() {
     return GCInfoAtBaseType<typename GetGarbageCollectedType<T>::type>::Index();
   }
 };
diff --git a/third_party/blink/renderer/platform/heap/gc_info_test.cc b/third_party/blink/renderer/platform/heap/gc_info_test.cc
index d651be0..6447fc31 100644
--- a/third_party/blink/renderer/platform/heap/gc_info_test.cc
+++ b/third_party/blink/renderer/platform/heap/gc_info_test.cc
@@ -16,8 +16,8 @@
 TEST(GCInfoTest, ResizeToMaxIndex) {
   GCInfoTable table;
   GCInfo info = {nullptr, nullptr, nullptr, false, false};
-  size_t slot = 0;
-  for (size_t i = 0; i < (GCInfoTable::kMaxIndex - 1); i++) {
+  uint32_t slot = 0;
+  for (uint32_t i = 0; i < (GCInfoTable::kMaxIndex - 1); i++) {
     slot = 0;
     table.EnsureGCInfoIndex(&info, &slot);
     EXPECT_LT(0u, slot);
diff --git a/third_party/blink/renderer/platform/heap/heap.cc b/third_party/blink/renderer/platform/heap/heap.cc
index 7688dfb5..cd4d676 100644
--- a/third_party/blink/renderer/platform/heap/heap.cc
+++ b/third_party/blink/renderer/platform/heap/heap.cc
@@ -430,8 +430,8 @@
   return arena_index_with_min_arena_age;
 }
 
-BaseArena* ThreadHeap::ExpandedVectorBackingArena(size_t gc_info_index) {
-  size_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
+BaseArena* ThreadHeap::ExpandedVectorBackingArena(uint32_t gc_info_index) {
+  uint32_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
   --likely_to_be_promptly_freed_[entry_index];
   int arena_index = vector_backing_arena_index_;
   arena_ages_[arena_index] = ++current_arena_ages_;
@@ -448,9 +448,9 @@
   }
 }
 
-void ThreadHeap::PromptlyFreed(size_t gc_info_index) {
+void ThreadHeap::PromptlyFreed(uint32_t gc_info_index) {
   DCHECK(thread_state_->CheckThread());
-  size_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
+  uint32_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
   // See the comment in vectorBackingArena() for why this is +3.
   likely_to_be_promptly_freed_[entry_index] += 3;
 }
@@ -550,7 +550,7 @@
   size_t total_dead_count = 0;
   size_t total_live_size = 0;
   size_t total_dead_size = 0;
-  for (size_t gc_info_index = 1;
+  for (uint32_t gc_info_index = 1;
        gc_info_index <= GCInfoTable::Get().GcInfoIndex(); ++gc_info_index) {
     total_live_count += info.live_count[gc_info_index];
     total_dead_count += info.dead_count[gc_info_index];
diff --git a/third_party/blink/renderer/platform/heap/heap.h b/third_party/blink/renderer/platform/heap/heap.h
index 1623c418..8f55a62a 100644
--- a/third_party/blink/renderer/platform/heap/heap.h
+++ b/third_party/blink/renderer/platform/heap/heap.h
@@ -280,7 +280,7 @@
   Address AllocateOnArenaIndex(ThreadState*,
                                size_t,
                                int arena_index,
-                               size_t gc_info_index,
+                               uint32_t gc_info_index,
                                const char* type_name);
   template <typename T>
   static Address Allocate(size_t, bool eagerly_sweep = false);
@@ -350,9 +350,9 @@
   //   (*) More than 33% of the same type of vectors have been promptly
   //       freed since the last GC.
   //
-  BaseArena* VectorBackingArena(size_t gc_info_index) {
+  BaseArena* VectorBackingArena(uint32_t gc_info_index) {
     DCHECK(thread_state_->CheckThread());
-    size_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
+    uint32_t entry_index = gc_info_index & kLikelyToBePromptlyFreedArrayMask;
     --likely_to_be_promptly_freed_[entry_index];
     int arena_index = vector_backing_arena_index_;
     // If likely_to_be_promptly_freed_[entryIndex] > 0, that means that
@@ -367,14 +367,14 @@
     DCHECK(IsVectorArenaIndex(arena_index));
     return arenas_[arena_index];
   }
-  BaseArena* ExpandedVectorBackingArena(size_t gc_info_index);
+  BaseArena* ExpandedVectorBackingArena(uint32_t gc_info_index);
   static bool IsVectorArenaIndex(int arena_index) {
     return BlinkGC::kVector1ArenaIndex <= arena_index &&
            arena_index <= BlinkGC::kVector4ArenaIndex;
   }
   static bool IsNormalArenaIndex(int);
   void AllocationPointAdjusted(int arena_index);
-  void PromptlyFreed(size_t gc_info_index);
+  void PromptlyFreed(uint32_t gc_info_index);
   void ClearArenaAges();
   int ArenaIndexOfVectorArenaLeastRecentlyExpanded(int begin_arena_index,
                                                    int end_arena_index);
@@ -602,7 +602,7 @@
 inline Address ThreadHeap::AllocateOnArenaIndex(ThreadState* state,
                                                 size_t size,
                                                 int arena_index,
-                                                size_t gc_info_index,
+                                                uint32_t gc_info_index,
                                                 const char* type_name) {
   DCHECK(state->IsAllocationAllowed());
   DCHECK_NE(arena_index, BlinkGC::kLargeObjectArenaIndex);
@@ -651,7 +651,7 @@
       arena_index = ArenaIndexForObjectSize(size);
   }
 
-  size_t gc_info_index = GCInfoTrait<T>::Index();
+  uint32_t gc_info_index = GCInfoTrait<T>::Index();
   // TODO(haraken): We don't support reallocate() for finalizable objects.
   DCHECK(!GCInfoTable::Get()
               .GCInfoFromIndex(previous_header->GcInfoIndex())
diff --git a/third_party/blink/renderer/platform/heap/heap_allocator.h b/third_party/blink/renderer/platform/heap/heap_allocator.h
index ab841704..e69c0df 100644
--- a/third_party/blink/renderer/platform/heap/heap_allocator.h
+++ b/third_party/blink/renderer/platform/heap/heap_allocator.h
@@ -72,7 +72,7 @@
     ThreadState* state =
         ThreadStateFor<ThreadingTrait<T>::kAffinity>::GetState();
     DCHECK(state->IsAllocationAllowed());
-    size_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
+    uint32_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
     NormalPageArena* arena = static_cast<NormalPageArena*>(
         state->Heap().VectorBackingArena(gc_info_index));
     return reinterpret_cast<T*>(arena->AllocateObject(
@@ -83,7 +83,7 @@
     ThreadState* state =
         ThreadStateFor<ThreadingTrait<T>::kAffinity>::GetState();
     DCHECK(state->IsAllocationAllowed());
-    size_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
+    uint32_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
     NormalPageArena* arena = static_cast<NormalPageArena*>(
         state->Heap().ExpandedVectorBackingArena(gc_info_index));
     return reinterpret_cast<T*>(arena->AllocateObject(
@@ -96,7 +96,7 @@
                                   size_t quantized_shrunk_size);
   template <typename T>
   static T* AllocateInlineVectorBacking(size_t size) {
-    size_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
+    uint32_t gc_info_index = GCInfoTrait<HeapVectorBacking<T>>::Index();
     ThreadState* state =
         ThreadStateFor<ThreadingTrait<T>::kAffinity>::GetState();
     const char* type_name = WTF_HEAP_PROFILER_TYPE_NAME(HeapVectorBacking<T>);
@@ -112,7 +112,7 @@
 
   template <typename T, typename HashTable>
   static T* AllocateHashTableBacking(size_t size) {
-    size_t gc_info_index =
+    uint32_t gc_info_index =
         GCInfoTrait<HeapHashTableBacking<HashTable>>::Index();
     ThreadState* state =
         ThreadStateFor<ThreadingTrait<T>::kAffinity>::GetState();
diff --git a/third_party/blink/renderer/platform/heap/heap_page.cc b/third_party/blink/renderer/platform/heap/heap_page.cc
index 47b4c3d..07ae942 100644
--- a/third_party/blink/renderer/platform/heap/heap_page.cc
+++ b/third_party/blink/renderer/platform/heap/heap_page.cc
@@ -1646,14 +1646,14 @@
       live_count++;
       live_size += header->size();
 
-      size_t gc_info_index = header->GcInfoIndex();
+      uint32_t gc_info_index = header->GcInfoIndex();
       info.live_count[gc_info_index]++;
       info.live_size[gc_info_index] += header->size();
     } else {
       dead_count++;
       dead_size += header->size();
 
-      size_t gc_info_index = header->GcInfoIndex();
+      uint32_t gc_info_index = header->GcInfoIndex();
       info.dead_count[gc_info_index]++;
       info.dead_size[gc_info_index] += header->size();
     }
@@ -1730,7 +1730,7 @@
   size_t live_count = 0;
   size_t dead_count = 0;
   HeapObjectHeader* header = ObjectHeader();
-  size_t gc_info_index = header->GcInfoIndex();
+  uint32_t gc_info_index = header->GcInfoIndex();
   size_t payload_size = header->PayloadSize();
   if (header->IsMarked()) {
     live_count = 1;
diff --git a/third_party/blink/renderer/platform/heap/heap_page.h b/third_party/blink/renderer/platform/heap/heap_page.h
index b6977736..2bc6d7c 100644
--- a/third_party/blink/renderer/platform/heap/heap_page.h
+++ b/third_party/blink/renderer/platform/heap/heap_page.h
@@ -194,7 +194,7 @@
 
   size_t size() const;
 
-  NO_SANITIZE_ADDRESS size_t GcInfoIndex() const {
+  NO_SANITIZE_ADDRESS uint32_t GcInfoIndex() const {
     return (encoded_ & kHeaderGCInfoIndexMask) >> kHeaderGCInfoIndexShift;
   }
 
diff --git a/third_party/blink/renderer/platform/runtime_enabled_features.json5 b/third_party/blink/renderer/platform/runtime_enabled_features.json5
index 5da92de..5a7e167 100644
--- a/third_party/blink/renderer/platform/runtime_enabled_features.json5
+++ b/third_party/blink/renderer/platform/runtime_enabled_features.json5
@@ -1118,6 +1118,9 @@
       name: "RTCUnifiedPlanByDefault",
     },
     {
+      name: "ScheduledScriptStreaming",
+    },
+    {
       name: "ScriptedSpeech",
       status: "stable",
     },
diff --git a/third_party/blink/renderer/platform/scheduler/DEPS b/third_party/blink/renderer/platform/scheduler/DEPS
index c8e9a1cd..749a0a1 100644
--- a/third_party/blink/renderer/platform/scheduler/DEPS
+++ b/third_party/blink/renderer/platform/scheduler/DEPS
@@ -39,6 +39,7 @@
   "+base/task/sequence_manager/sequence_manager.h",
   "+base/task/sequence_manager/task_queue.h",
   "+base/task/sequence_manager/time_domain.h",
+  "+base/task_scheduler/task_traits.h",
   "+base/threading/platform_thread.h",
   "+base/threading/sequenced_task_runner_handle.h",
   "+base/threading/thread.h",
diff --git a/third_party/blink/renderer/platform/scheduler/common/background_scheduler.cc b/third_party/blink/renderer/platform/scheduler/common/background_scheduler.cc
index 18d898e3..5337518 100644
--- a/third_party/blink/renderer/platform/scheduler/common/background_scheduler.cc
+++ b/third_party/blink/renderer/platform/scheduler/common/background_scheduler.cc
@@ -11,8 +11,16 @@
 
 void BackgroundScheduler::PostOnBackgroundThread(const base::Location& location,
                                                  CrossThreadClosure closure) {
-  base::PostTaskWithTraits(location,
-                           {base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
+  PostOnBackgroundThreadWithTraits(
+      location, {base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
+      std::move(closure));
+}
+
+void BackgroundScheduler::PostOnBackgroundThreadWithTraits(
+    const base::Location& location,
+    const base::TaskTraits& traits,
+    CrossThreadClosure closure) {
+  base::PostTaskWithTraits(location, traits,
                            ConvertToBaseCallback(std::move(closure)));
 }
 
diff --git a/third_party/blink/renderer/platform/scheduler/public/background_scheduler.h b/third_party/blink/renderer/platform/scheduler/public/background_scheduler.h
index ef80a0a..57bffd3 100644
--- a/third_party/blink/renderer/platform/scheduler/public/background_scheduler.h
+++ b/third_party/blink/renderer/platform/scheduler/public/background_scheduler.h
@@ -6,6 +6,8 @@
 #define THIRD_PARTY_BLINK_RENDERER_PLATFORM_SCHEDULER_PUBLIC_BACKGROUND_SCHEDULER_H_
 
 #include "base/location.h"
+#include "base/sequenced_task_runner.h"
+#include "base/task_scheduler/task_traits.h"
 #include "third_party/blink/renderer/platform/platform_export.h"
 #include "third_party/blink/renderer/platform/wtf/functional.h"
 
@@ -13,14 +15,22 @@
 
 namespace BackgroundScheduler {
 
-// This is a thin wrapper around base::TaskScheduler to accomodate
-// Blink's CrossThreadClosure, which only allows background tasks.
+// These are a thin wrapper around base::TaskScheduler to accomodate
+// Blink's CrossThreadClosure, which only allows background tasks
+// (i.e. tasks which are run off the main thread).
 //
 // Non-background tasks should be posted using another scheduler, e.g.
 // FrameShceduler.
 PLATFORM_EXPORT void PostOnBackgroundThread(const base::Location&,
                                             CrossThreadClosure);
 
+PLATFORM_EXPORT void PostOnBackgroundThreadWithTraits(const base::Location&,
+                                                      const base::TaskTraits&,
+                                                      CrossThreadClosure);
+
+// TODO(altimin): Expose CreateBackgroundTaskRunnerWithTraits when the
+// need arises.
+
 }  // namespace BackgroundScheduler
 
 }  // namespace blink
diff --git a/third_party/blink/renderer/platform/wtf/allocator/partitions.h b/third_party/blink/renderer/platform/wtf/allocator/partitions.h
index 079afd0..8ee8698e 100644
--- a/third_party/blink/renderer/platform/wtf/allocator/partitions.h
+++ b/third_party/blink/renderer/platform/wtf/allocator/partitions.h
@@ -101,6 +101,11 @@
                                            const char* type_name) {
     return BufferPartition()->Realloc(p, n, type_name);
   }
+  ALWAYS_INLINE static void* BufferTryRealloc(void* p,
+                                              size_t n,
+                                              const char* type_name) {
+    return BufferPartition()->TryRealloc(p, n, type_name);
+  }
   ALWAYS_INLINE static void BufferFree(void* p) { BufferPartition()->Free(p); }
   ALWAYS_INLINE static size_t BufferActualSize(size_t n) {
     return BufferPartition()->ActualSize(n);
diff --git a/third_party/blink/tools/audit_non_blink_usage.py b/third_party/blink/tools/audit_non_blink_usage.py
index 0b33cd7..aa4e0c1 100755
--- a/third_party/blink/tools/audit_non_blink_usage.py
+++ b/third_party/blink/tools/audit_non_blink_usage.py
@@ -118,6 +118,13 @@
             # Base atomic utilities
             'base::AtomicSequenceNumber',
 
+            # Task traits
+            'base::TaskTraits',
+            'base::MayBlock',
+            'base::TaskPriority',
+            'base::TaskShutdownBehavior',
+            'base::WithBaseSyncPrimitives',
+
             # Byte order
             'base::ByteSwap',
             'base::NetToHost(16|32|64)',
diff --git a/tools/binary_size/libsupersize/gsutil.py b/tools/binary_size/libsupersize/gsutil.py
new file mode 100644
index 0000000..855517b
--- /dev/null
+++ b/tools/binary_size/libsupersize/gsutil.py
@@ -0,0 +1,67 @@
+# Copyright 2018 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+"""Update the Google Cloud Storage bucket hosting the Super Size UI."""
+
+import argparse
+import os
+import subprocess
+import uuid
+
+
+GS_BUCKET = 'gs://chrome-supersize'
+
+
+def _SyncStatic():
+  """Upload static files from the static directory."""
+  static_files = os.path.join(os.path.dirname(__file__), 'static')
+  subprocess.check_call([
+    'gsutil.py', '-m', 'rsync', '-r', static_files, GS_BUCKET
+  ])
+
+
+def _SyncTemplates():
+  """Generate and upload the templates/sw.js file."""
+  template_file = os.path.join(os.path.dirname(__file__), 'templates', 'sw.js')
+  cache_hash = uuid.uuid4().hex
+
+  p = subprocess.Popen([
+    'gsutil.py', 'cp', '-p', '-', '%s/sw.js' % GS_BUCKET
+  ], stdin=subprocess.PIPE)
+  with open(template_file, 'r') as in_file:
+    p.communicate(in_file.read().replace('{{cache_hash}}', cache_hash))
+
+
+def _SetMetaAndPermissions():
+  # sw.js has the wrong type due to being created from a stream
+  subprocess.check_call([
+    'gsutil.py', 'setmeta', '-h', 'Content-Type:application/javascript',
+    '%s/sw.js' % GS_BUCKET
+  ])
+  subprocess.check_call([
+    'gsutil.py', '-m', 'setmeta', '-h', 'Content-Type:application/x-ndjson',
+    '%s/milestones/*.ndjson' % GS_BUCKET
+  ])
+
+  # All files in the root of the bucket are user readable
+  subprocess.check_call([
+    'gsutil.py', '-m', 'acl', 'ch', '-u', 'AllUsers:R', '%s/*' % GS_BUCKET
+  ])
+
+
+def main():
+  parser = argparse.ArgumentParser(description=__doc__)
+  parser.add_argument('--sync', action='store_true', required=True,
+                      help='Sync static and template files to GCS.')
+
+  args = parser.parse_args()
+
+  if args.sync:
+    _SyncStatic()
+    _SyncTemplates()
+    _SetMetaAndPermissions()
+
+
+if __name__ == '__main__':
+  main()
diff --git a/tools/binary_size/libsupersize/static/index.html b/tools/binary_size/libsupersize/static/index.html
index eb6187e..9e176e1 100644
--- a/tools/binary_size/libsupersize/static/index.html
+++ b/tools/binary_size/libsupersize/static/index.html
@@ -7,9 +7,10 @@
 -->
 
 <head>
-    <title>Binary Size Analysis</title>
+    <title>Super Size Tiger View</title>
     <script src="start-worker.js"></script>
     <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <meta name="theme-color" content="#4285f4">
     <link href="https://fonts.googleapis.com/css?family=Google+Sans:400,500|Roboto:400,500" rel="stylesheet">
     <style>
         body {
@@ -205,17 +206,23 @@
     </style>
     <link rel="stylesheet" href="options.css">
     <link rel="icon" href="favicon.ico" sizes="16x16 32x32 256x256" type="image/x-icon">
+    <link rel="manifest" href="manifest.json">
     <script defer src="shared.js"></script>
     <script defer src="state.js"></script>
     <script defer src="infocard-ui.js"></script>
     <script defer src="tree-ui.js"></script>
+    <script defer async>
+        if ('serviceWorker' in navigator) {
+            navigator.serviceWorker.register('sw.js');
+        }
+    </script>
 </head>
 
 <body>
     <div class="scrim toggle-options" hidden></div>
     <header class="appbar">
         <div class="appbar-inner">
-            <h1 class="headline">Binary Size Analysis</h1>
+            <h1 class="headline">Super Size Tiger View</h1>
             <input type="file" name="upload" id="upload" accept=".ndjson" >
             <label for="upload" class="text-button filled-button with-icon">
                 <svg class="icon" xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24" fill="#fff">
diff --git a/tools/binary_size/libsupersize/static/manifest.json b/tools/binary_size/libsupersize/static/manifest.json
new file mode 100644
index 0000000..355e862
--- /dev/null
+++ b/tools/binary_size/libsupersize/static/manifest.json
@@ -0,0 +1,22 @@
+{
+  "short_name": "Super Size",
+  "name": "Super Size Tiger View",
+  "description": "View interactive size breakdowns from Chrome",
+  "icons": [
+    {
+      "src": "favicon.ico",
+      "sizes": "16x16 32x32 256x256"
+    },
+    {
+      "src": "splash.png",
+      "type": "image/png",
+      "sizes": "512x512"
+    }
+  ],
+  "lang": "en",
+  "start_url": "index.html",
+  "background_color": "#fff",
+  "display": "standalone",
+  "scope": "./",
+  "theme_color": "#4285f4"
+}
diff --git a/tools/binary_size/libsupersize/static/options.css b/tools/binary_size/libsupersize/static/options.css
index f627327..2fa0185b 100644
--- a/tools/binary_size/libsupersize/static/options.css
+++ b/tools/binary_size/libsupersize/static/options.css
@@ -34,7 +34,7 @@
   position: fixed;
   right: 0;
   top: 0;
-  height: 100vh;
+  bottom: 0;
   width: 224px;
   background: #fffffff5;
   box-shadow: 0 1px 2px #3c40434d, 0 2px 6px 2px #3c404326;
@@ -306,15 +306,11 @@
   .symbols {
     padding: 0 16px;
   }
-  .appbar-progress {
-    margin: 0 -16px;
-    width: calc(100% + 32px);
-  }
 
-  .filled-button {
+  .text-button.with-icon {
     padding-right: 6px;
   }
-  .filled-button .label-text {
+  .text-button.with-icon .label-text {
     display: none;
   }
 }
diff --git a/tools/binary_size/libsupersize/static/splash.png b/tools/binary_size/libsupersize/static/splash.png
new file mode 100644
index 0000000..db8d5f8
--- /dev/null
+++ b/tools/binary_size/libsupersize/static/splash.png
Binary files differ
diff --git a/tools/binary_size/libsupersize/templates/sw.js b/tools/binary_size/libsupersize/templates/sw.js
new file mode 100644
index 0000000..6807eed
--- /dev/null
+++ b/tools/binary_size/libsupersize/templates/sw.js
@@ -0,0 +1,51 @@
+// Copyright 2018 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+// @ts-check
+'use strict';
+
+const cacheName = '{{cache_hash}}';
+const filesToCache = [
+  'favicon.ico',
+  'index.html',
+  'infocard-ui.js',
+  'infocard.css',
+  'manifest.json',
+  'options.css',
+  'shared.js',
+  'start-worker.js',
+  'state.js',
+  'tree-ui.js',
+  'tree-worker.js',
+];
+
+// On install, cache the items in the `filesToCache` list
+self.addEventListener('install', event => {
+  event.waitUntil(
+    caches.open(cacheName).then(cache => cache.addAll(filesToCache))
+  );
+});
+
+// On activate, remove any old caches
+self.addEventListener('activate', event => {
+  async function deleteOldCache(key) {
+    if (key !== cacheName) {
+      return caches.delete(key);
+    }
+  }
+
+  event.waitUntil(
+    caches.keys().then(keyList => Promise.all(keyList.map(deleteOldCache)))
+  );
+  return self.clients.claim();
+});
+
+// On fetch, return entries from the cache if possible
+self.addEventListener('fetch', event => {
+  event.respondWith(
+    caches
+      .match(event.request, {ignoreSearch: true})
+      .then(response => response || fetch(event.request))
+  );
+});
diff --git a/tools/cygprofile/phased_orderfile.py b/tools/cygprofile/phased_orderfile.py
index 73bb234..ac0d27fb 100755
--- a/tools/cygprofile/phased_orderfile.py
+++ b/tools/cygprofile/phased_orderfile.py
@@ -13,6 +13,13 @@
 orderfile with three parts: the code touched only in startup, the code
 touched only during interaction, and code common to the two phases. We refer to
 these parts as the orderfile phases.
+
+Example invocation, with PROFILE_DIR the location of the profile data pulled
+from a device and LIBTYPE either monochrome or chrome as appropriate.
+./tools/cygprofile/phased_orderfile.py \
+    --profile-directory=PROFILE_DIR \
+    --instrumented-build-dir=out-android/Orderfile/ \
+    --library-name=libLIBTYPE.so --offset-output-base=PROFILE_DIR/offset
 """
 
 import argparse
@@ -26,7 +33,7 @@
 
 
 # Files matched when using this script to analyze directly (see main()).
-PROFILE_GLOB = 'cygprofile-*.txt_*'
+PROFILE_GLOB = 'profile-hitmap-*.txt_*'
 
 
 OrderfilePhaseOffsets = collections.namedtuple(
@@ -45,6 +52,9 @@
   COMMON_STABILITY_THRESHOLD = 1.75
   INTERACTION_STABILITY_THRESHOLD = 2.5
 
+  # The process name of the browser as used in the profile dumps.
+  BROWSER = 'browser'
+
   def __init__(self, profiles, processor):
     """Intialize.
 
@@ -54,7 +64,11 @@
     """
     self._profiles = profiles
     self._processor = processor
+
+    # These members cache various computed values.
     self._phase_offsets = None
+    self._annotated_offsets = None
+    self._process_list = None
 
   def IsStableProfile(self):
     """Verify that the profiling has been stable.
@@ -65,7 +79,7 @@
       True if the profile was stable as described above.
     """
     (startup_stability, common_stability,
-     interaction_stability) = self.ComputeStability()
+     interaction_stability) = [s[0] for s in self.ComputeStability()]
 
     stable = True
     if startup_stability > self.STARTUP_STABILITY_THRESHOLD:
@@ -89,11 +103,46 @@
     they cover.
 
     Returns:
-      (float, float, float) A heuristic stability metric for startup, common and
-          interaction orderfile phases, respectively.
+      ((float, int), (float, int), (float, int)) A heuristic stability metric
+          for startup, common and interaction orderfile phases,
+          respectively. Each metric is a pair of the ratio of symbol sizes as
+          described above, and the size of the intersection.
+    """
+    (startup_intersection, startup_union,
+     common_intersection, common_union,
+     interaction_intersection, interaction_union,
+     _, _) = self.GetCombinedOffsets()
+    startup_intersection_size = self._processor.OffsetsPrimarySize(
+        startup_intersection)
+    common_intersection_size = self._processor.OffsetsPrimarySize(
+        common_intersection)
+    interaction_intersection_size = self._processor.OffsetsPrimarySize(
+        interaction_intersection)
+    startup_stability = self._SafeDiv(
+        self._processor.OffsetsPrimarySize(startup_union),
+        startup_intersection_size)
+    common_stability = self._SafeDiv(
+        self._processor.OffsetsPrimarySize(common_union),
+        common_intersection_size)
+    interaction_stability = self._SafeDiv(
+        self._processor.OffsetsPrimarySize(interaction_union),
+        interaction_intersection_size)
+    return ((startup_stability, startup_intersection_size),
+            (common_stability, common_intersection_size),
+            (interaction_stability, interaction_intersection_size))
+
+  def GetCombinedOffsets(self):
+    """Get offsets for the union and intersection of orderfile phases.
+
+    Returns:
+      ([int] * 8) For each of startup, common, interaction and all, respectively
+          the intersection and union offsets, in that order.
     """
     phase_offsets = self._GetOrderfilePhaseOffsets()
-    assert len(phase_offsets) > 1  # Otherwise the analysis is silly.
+    assert phase_offsets
+    if len(phase_offsets) == 1:
+      logging.error('Only one run set, the combined offset files will all be '
+                    'identical')
 
     startup_union = set(phase_offsets[0].startup)
     startup_intersection = set(phase_offsets[0].startup)
@@ -108,16 +157,172 @@
       common_intersection &= set(offsets.common)
       interaction_union |= set(offsets.interaction)
       interaction_intersection &= set(offsets.interaction)
-    startup_stability = self._SafeDiv(
-        self._processor.OffsetsPrimarySize(startup_union),
-        self._processor.OffsetsPrimarySize(startup_intersection))
-    common_stability = self._SafeDiv(
-        self._processor.OffsetsPrimarySize(common_union),
-        self._processor.OffsetsPrimarySize(common_intersection))
-    interaction_stability = self._SafeDiv(
-        self._processor.OffsetsPrimarySize(interaction_union),
-        self._processor.OffsetsPrimarySize(interaction_intersection))
-    return (startup_stability, common_stability, interaction_stability)
+    return (startup_intersection, startup_union,
+            common_intersection, common_union,
+            interaction_intersection, interaction_union,
+            (startup_union & common_union & interaction_union),
+            (startup_union | common_union | interaction_union))
+
+  def GetOffsetsForMemoryFootprint(self):
+    """Get offsets organized to minimize the memory footprint.
+
+    The startup, common and interaction offsets are computed for each
+    process. Any symbols used by one process in startup or interaction that are
+    used in a different phase by another process are moved to the common
+    section. This should minimize the memory footprint by keeping startup- or
+    interaction-only pages clean, at the possibly expense of startup time, as
+    more of the common section will need to be loaded. To mitigate that effect,
+    symbols moved from startup are placed at the beginning of the common
+    section, and those moved from interaction are placed at the end.
+
+    Browser startup symbols are placed at the beginning of the startup section
+    in the hope of working out with native library prefetching to minimize
+    startup time.
+
+    Returns:
+      OrdrerfilePhaseOffsets as described above.
+    """
+    startup = []
+    common_head = []
+    common = []
+    common_tail = []
+    interaction = []
+
+    process_offsets = {p: self._GetCombinedProcessOffsets(p)
+                       for p in self._GetProcessList()}
+    assert self.BROWSER in process_offsets.keys()
+
+    any_startup = set()
+    any_interaction = set()
+    any_common = set()
+    for offsets in process_offsets.itervalues():
+      any_startup |= set(offsets.startup)
+      any_interaction |= set(offsets.interaction)
+      any_common |= set(offsets.common)
+
+    already_added = set()
+    # This helper function splits |offsets|, adding to |alternate| all offsets
+    # that are in |interfering| or are already known to be common, and otherwise
+    # adding to |target|.
+    def add_process_offsets(offsets, interfering, target, alternate):
+      for o in offsets:
+        if o in already_added:
+          continue
+        if o in interfering or o in any_common:
+          alternate.append(o)
+        else:
+          target.append(o)
+        already_added.add(o)
+
+    # This helper updates |common| with new members of |offsets|.
+    def add_common_offsets(offsets):
+      for o in offsets:
+        if o not in already_added:
+          common.append(o)
+          already_added.add(o)
+
+    add_process_offsets(process_offsets[self.BROWSER].startup,
+                        any_interaction, startup, common_head)
+    add_process_offsets(process_offsets[self.BROWSER].interaction,
+                        any_startup, interaction, common_tail)
+    add_common_offsets(process_offsets[self.BROWSER].common)
+
+    for p in process_offsets:
+      if p == self.BROWSER:
+        continue
+      add_process_offsets(process_offsets[p].startup,
+                          any_interaction, startup, common_head)
+      add_process_offsets(process_offsets[p].interaction,
+                          any_startup, interaction, common_tail)
+      add_common_offsets(process_offsets[p].common)
+
+    return OrderfilePhaseOffsets(
+        startup=startup,
+        common=(common_head + common + common_tail),
+        interaction=interaction)
+
+  def GetOffsetsForStartup(self):
+    """Get offsets organized to minimize startup time.
+
+    The startup, common and interaction offsets are computed for each
+    process. Any symbol used by one process in interaction that appears in a
+    different phase in another process is moved to common, but any symbol that
+    appears in startup for *any* process stays in startup.
+
+    This should maximize startup performance at the expense of increasing the
+    memory footprint, as some startup symbols will not be able to page out.
+
+    The startup symbols in the browser process appear first in the hope of
+    working out with native library prefetching to minimize startup time.
+    """
+    startup = []
+    common = []
+    interaction = []
+    already_added = set()
+
+    process_offsets = {p: self._GetCombinedProcessOffsets(p)
+                       for p in self._GetProcessList()}
+    startup.extend(process_offsets[self.BROWSER].startup)
+    already_added |= set(process_offsets[self.BROWSER].startup)
+    common.extend(process_offsets[self.BROWSER].common)
+    already_added |= set(process_offsets[self.BROWSER].common)
+    interaction.extend(process_offsets[self.BROWSER].interaction)
+    already_added |= set(process_offsets[self.BROWSER].interaction)
+
+    for process, offsets in process_offsets.iteritems():
+      if process == self.BROWSER:
+        continue
+      startup.extend(o for o in offsets.startup
+                     if o not in already_added)
+      already_added |= set(offsets.startup)
+      common.extend(o for o in offsets.common
+                     if o not in already_added)
+      already_added |= set(offsets.common)
+      interaction.extend(o for o in offsets.interaction
+                     if o not in already_added)
+      already_added |= set(offsets.interaction)
+
+    return OrderfilePhaseOffsets(
+        startup=startup, common=common, interaction=interaction)
+
+  def _GetCombinedProcessOffsets(self, process):
+    """Combine offsets across runs for a particular process.
+
+    Args:
+      process (str) The process to combine.
+
+    Returns:
+      OrderfilePhaseOffsets, the startup, common and interaction offsets for the
+      process in question. The offsets are sorted arbitrarily.
+    """
+    (startup, common, interaction) = ([], [], [])
+    assert self._profiles.GetPhases() == set([0,1]), 'Unexpected phases'
+    for o in self._GetAnnotatedOffsets():
+      startup_count = o.Count(0, process)
+      interaction_count = o.Count(1, process)
+      if not startup_count and not interaction_count:
+        continue
+      if startup_count and interaction_count:
+        common.append(o.Offset())
+      elif startup_count:
+        startup.append(o.Offset())
+      else:
+        interaction.append(o.Offset())
+    return OrderfilePhaseOffsets(
+        startup=startup, common=common, interaction=interaction)
+
+  def _GetAnnotatedOffsets(self):
+    if self._annotated_offsets is None:
+      self._annotated_offsets = self._profiles.GetAnnotatedOffsets()
+      self._processor.TranslateAnnotatedSymbolOffsets(self._annotated_offsets)
+    return self._annotated_offsets
+
+  def _GetProcessList(self):
+    if self._process_list is None:
+      self._process_list = set()
+      for o in self._GetAnnotatedOffsets():
+        self._process_list.update(o.Processes())
+    return self._process_list
 
   def _GetOrderfilePhaseOffsets(self):
     """Compute the phase offsets for each run.
@@ -129,7 +334,8 @@
     if self._phase_offsets is not None:
       return self._phase_offsets
 
-    assert self._profiles.GetPhases() == set([0, 1]), 'Unexpected phases'
+    assert self._profiles.GetPhases() == set([0, 1]), (
+        'Unexpected phases {}'.format(self._profiles.GetPhases()))
     self._phase_offsets = []
     for first, second in zip(self._profiles.GetRunGroupOffsets(phase=0),
                              self._profiles.GetRunGroupOffsets(phase=1)):
@@ -175,10 +381,17 @@
                       help=('Directory containing profile runs. Files '
                             'matching {} are used.'.format(PROFILE_GLOB)))
   parser.add_argument('--instrumented-build-dir', type=str,
-                      help='Path to the instrumented build', required=True)
+                      help='Path to the instrumented build (eg, out/Orderfile)',
+                      required=True)
   parser.add_argument('--library-name', default='libchrome.so',
                       help=('Chrome shared library name (usually libchrome.so '
                             'or libmonochrome.so'))
+  parser.add_argument('--offset-output-base', default=None, type=str,
+                      help=('If present, a base name to output offsets to. '
+                            'No offsets are output if this is missing. The '
+                            'base name is suffixed with _for_memory and '
+                            '_for_startup, corresponding to the two sets of '
+                            'offsets produced.'))
   return parser
 
 
@@ -186,12 +399,25 @@
   logging.basicConfig(level=logging.INFO)
   parser = _CreateArgumentParser()
   args = parser.parse_args()
-  profiles = process_profiles.ProfileManager(
-      glob.glob(os.path.join(args.profile_directory, PROFILE_GLOB)))
+  profiles = process_profiles.ProfileManager(itertools.chain.from_iterable(
+      glob.glob(os.path.join(d, PROFILE_GLOB))
+      for d in args.profile_directory.split(',')))
   processor = process_profiles.SymbolOffsetProcessor(os.path.join(
       args.instrumented_build_dir, 'lib.unstripped', args.library_name))
   phaser = PhasedAnalyzer(profiles, processor)
-  print 'Stability: {:.2f} {:.2f} {:.2f}'.format(*phaser.ComputeStability())
+  stability = phaser.ComputeStability()
+  print 'Stability: {:.2} {:.2} {:.2}'.format(*[s[0] for s in stability])
+  print 'Sizes: {} {} {}'.format(*[s[1] for s in stability])
+  if args.offset_output_base is not None:
+    for name, offsets in zip(
+        ['_for_memory', '_for_startup'],
+        [phaser.GetOffsetsForMemoryFootprint(),
+         phaser.GetOffsetsForStartup()]):
+      with file(args.offset_output_base + name, 'w') as output:
+        output.write('\n'.join(
+            str(i) for i in (offsets.startup + offsets.common +
+                             offsets.interaction)))
+        output.write('\n')
 
 
 if __name__ == '__main__':
diff --git a/tools/cygprofile/phased_orderfile_unittest.py b/tools/cygprofile/phased_orderfile_unittest.py
index f2209bf..6941f7f 100755
--- a/tools/cygprofile/phased_orderfile_unittest.py
+++ b/tools/cygprofile/phased_orderfile_unittest.py
@@ -11,20 +11,47 @@
 import phased_orderfile
 import process_profiles
 
-from test_utils import (SimpleTestSymbol,
+from test_utils import (ProfileFile,
+                        SimpleTestSymbol,
                         TestSymbolOffsetProcessor,
                         TestProfileManager)
 
 
-class Mod10Processor(object):
+class Mod10Processor(process_profiles.SymbolOffsetProcessor):
   """A restricted mock for a SymbolOffsetProcessor.
 
-  This only implements GetReachedOffsetsFromDump, and works by mapping a dump
-  offset to offset - (offset % 10). If the dump offset is negative, it is marked
-  as not found.
+  This only implements {Translate,Get}ReacheOffsetsFromDump, and works by
+  mapping a dump offset to offset - (offset % 10). If the dump offset is
+  negative, it is marked as not found.
   """
-  def GetReachedOffsetsFromDump(self, dump):
-    return [x - (x % 10) for x in dump if x >= 0]
+  def __init__(self):
+    super(Mod10Processor, self).__init__(None)
+
+  def _TranslateReachedOffsetsFromDump(self, items, get, update):
+    for i in items:
+      x = get(i)
+      if x >= 0:
+        update(i, x - (x % 10))
+      else:
+        update(i, None)
+
+
+class IdentityProcessor(process_profiles.SymbolOffsetProcessor):
+  """A restricted mock for a SymbolOffsetProcessor.
+
+  This only implements {Translate,Get}ReachedOffsetsFromDump, and maps the dump
+  offset to itself. If the dump offset is negative, it is marked as not found.
+  """
+  def __init__(self):
+    super(IdentityProcessor, self).__init__(None)
+
+  def _TranslateReachedOffsetsFromDump(self, items, get, update):
+    for i in items:
+      x = get(i)
+      if x >= 0:
+        update(i, x)
+      else:
+        update(i, None)
 
 
 class PhasedOrderfileTestCase(unittest.TestCase):
@@ -32,11 +59,6 @@
   def setUp(self):
     self._file_counter = 0
 
-  def File(self, timestamp_sec, phase):
-    self._file_counter += 1
-    return 'file-{}-{}.txt_{}'.format(
-        self._file_counter, timestamp_sec * 1000 * 1000 * 1000, phase)
-
   def testProfileStability(self):
     symbols = [SimpleTestSymbol(str(i), i, 10)
                for i in xrange(20)]
@@ -46,7 +68,8 @@
         startup=s, common=c, interaction=i)
     phaser._phase_offsets = [opo(range(5), range(6, 10), range(11,15)),
                              opo(range(4), range(6, 10), range(18, 20))]
-    self.assertEquals((1.25, 1, None), phaser.ComputeStability())
+    self.assertEquals((1.25, 1, None),
+                      tuple(s[0] for s in phaser.ComputeStability()))
 
   def testIsStable(self):
     symbols = [SimpleTestSymbol(str(i), i, 10)
@@ -64,12 +87,12 @@
 
   def testGetOrderfilePhaseOffsets(self):
     mgr = TestProfileManager({
-        self.File(0, 0): [12, 21, -1, 33],
-        self.File(0, 1): [31, 49, 52],
-        self.File(100, 0): [113, 128],
-        self.File(200, 1): [132, 146],
-        self.File(300, 0): [19, 20, 32],
-        self.File(300, 1): [24, 39]})
+        ProfileFile(0, 0): [12, 21, -1, 33],
+        ProfileFile(0, 1): [31, 49, 52],
+        ProfileFile(100, 0): [113, 128],
+        ProfileFile(200, 1): [132, 146],
+        ProfileFile(300, 0): [19, 20, 32],
+        ProfileFile(300, 1): [24, 39]})
     phaser = phased_orderfile.PhasedAnalyzer(mgr, Mod10Processor())
     opo = lambda s, c, i: phased_orderfile.OrderfilePhaseOffsets(
         startup=s, common=c, interaction=i)
@@ -79,6 +102,47 @@
                           opo([10], [20, 30], [])],
                          phaser._GetOrderfilePhaseOffsets())
 
+  def testGetCombinedProcessOffsets(self):
+    mgr = TestProfileManager({
+        ProfileFile(40, 0, ''): [1, 2, 3],
+        ProfileFile(50, 1, ''): [3, 4, 5],
+        ProfileFile(51, 0, 'renderer'): [2, 3, 6],
+        ProfileFile(51, 1, 'gpu-process'): [6, 7],
+        ProfileFile(70, 0, ''): [2, 8, 9],
+        ProfileFile(70, 1, ''): [9]})
+    phaser = phased_orderfile.PhasedAnalyzer(mgr, IdentityProcessor())
+    offsets = phaser._GetCombinedProcessOffsets('browser')
+    self.assertListEqual([1, 2, 8], sorted(offsets.startup))
+    self.assertListEqual([4, 5], sorted(offsets.interaction))
+    self.assertListEqual([3, 9], sorted(offsets.common))
+
+    offsets = phaser._GetCombinedProcessOffsets('gpu-process')
+    self.assertListEqual([], sorted(offsets.startup))
+    self.assertListEqual([6, 7], sorted(offsets.interaction))
+    self.assertListEqual([], sorted(offsets.common))
+
+    self.assertListEqual(['browser', 'gpu-process', 'renderer'],
+                         sorted(phaser._GetProcessList()))
+
+  def testGetOffsetVariations(self):
+    mgr = TestProfileManager({
+        ProfileFile(40, 0, ''): [1, 2, 3],
+        ProfileFile(50, 1, ''): [3, 4, 5],
+        ProfileFile(51, 0, 'renderer'): [2, 3, 6],
+        ProfileFile(51, 1, 'gpu-process'): [6, 7],
+        ProfileFile(70, 0, ''): [2, 6, 8, 9],
+        ProfileFile(70, 1, ''): [9]})
+    phaser = phased_orderfile.PhasedAnalyzer(mgr, IdentityProcessor())
+    offsets = phaser.GetOffsetsForMemoryFootprint()
+    self.assertListEqual([1, 2, 8], offsets.startup)
+    self.assertListEqual([6, 3, 9], offsets.common)
+    self.assertListEqual([4, 5, 7], offsets.interaction)
+
+    offsets = phaser.GetOffsetsForStartup()
+    self.assertListEqual([1, 2, 6, 8], offsets.startup)
+    self.assertListEqual([3, 9], offsets.common)
+    self.assertListEqual([4, 5, 7], offsets.interaction)
+
 
 if __name__ == "__main__":
   unittest.main()
diff --git a/tools/cygprofile/process_profiles.py b/tools/cygprofile/process_profiles.py
index a436ead..916750f6 100755
--- a/tools/cygprofile/process_profiles.py
+++ b/tools/cygprofile/process_profiles.py
@@ -56,6 +56,7 @@
     self._name_to_symbol = None
     self._offset_to_primary = None
     self._offset_to_symbols = None
+    self._offset_to_symbol_info = None
 
   def SymbolInfos(self):
     """The symbols associated with this processor's binary.
@@ -152,24 +153,14 @@
     Returns:
       [int] Reached symbol offsets.
     """
-    dump_offset_to_symbol_info = self._GetDumpOffsetToSymbolInfo()
-    logging.info('Offset to Symbol size = %d', len(dump_offset_to_symbol_info))
-    assert max(dump) / 4 <= len(dump_offset_to_symbol_info)
-    already_seen = set()
     reached_offsets = []
-    reached_return_addresses_not_found = 0
-    for dump_offset in dump:
-      symbol_info = dump_offset_to_symbol_info[dump_offset / 4]
-      if symbol_info is None:
-        reached_return_addresses_not_found += 1
-        continue
-      if symbol_info.offset in already_seen:
-        continue
-      reached_offsets.append(symbol_info.offset)
-      already_seen.add(symbol_info.offset)
-    if reached_return_addresses_not_found:
-      logging.warning('%d return addresses don\'t map to any symbol',
-                      reached_return_addresses_not_found)
+    already_seen = set()
+    def update(_, symbol_offset):
+      if symbol_offset is None or symbol_offset in already_seen:
+        return
+      reached_offsets.append(symbol_offset)
+      already_seen.add(symbol_offset)
+    self._TranslateReachedOffsetsFromDump(dump, lambda x: x, update)
     return reached_offsets
 
   def MatchSymbolNames(self, symbol_names):
@@ -185,6 +176,52 @@
     matched_names = our_symbol_names.intersection(set(symbol_names))
     return [self.NameToSymbolMap()[n] for n in matched_names]
 
+  def TranslateAnnotatedSymbolOffsets(self, annotated_offsets):
+    """Merges offsets across run groups and translates to symbol offsets.
+
+    Like GetReachedOffsetsFromDump, but works with AnnotatedOffsets.
+
+    Args:
+      annotated_offsets (AnnotatedOffset iterable) List of annotated offsets,
+        eg from ProfileManager.GetAnnotatedOffsets(). This will be mutated to
+        translate raw offsets to symbol offsets.
+    """
+    self._TranslateReachedOffsetsFromDump(
+        annotated_offsets,
+        lambda o: o.Offset(),
+        lambda o, symbol_offset: o.SetOffset(symbol_offset))
+
+  def _TranslateReachedOffsetsFromDump(self, items, get, update):
+    """Translate raw binary offsets to symbol offsets.
+
+    See GetReachedOffsetsFromDump for details. This version calls
+    |get(i)| on each element |i| of |items|, then calls
+    |update(i, symbol_offset)| with the updated offset. If the offset is not
+    found, update will be called with None.
+
+    Args:
+      items: (iterable) Items containing offsets.
+      get: (lambda item) As described above.
+      update: (lambda item, int) As described above.
+    """
+    dump_offset_to_symbol_info = self._GetDumpOffsetToSymbolInfo()
+    logging.info('Offset to Symbol size = %d', len(dump_offset_to_symbol_info))
+    reached_return_addresses_not_found = 0
+    for i in items:
+      dump_offset = get(i)
+      idx = dump_offset / 4
+      assert idx < len(dump_offset_to_symbol_info), (
+          'Dump offset out of binary range')
+      symbol_info = dump_offset_to_symbol_info[idx]
+      if symbol_info is None:
+        reached_return_addresses_not_found += 1
+        update(i, None)
+      else:
+        update(i, symbol_info.offset)
+    if reached_return_addresses_not_found:
+      logging.warning('%d return addresses don\'t map to any symbol',
+                      reached_return_addresses_not_found)
+
   def _GetDumpOffsetToSymbolInfo(self):
     """Computes an array mapping each word in .text to a symbol.
 
@@ -192,15 +229,16 @@
       [symbol_extractor.SymbolInfo or None] For every 4 bytes of the .text
         section, maps it to a symbol, or None.
     """
-    min_offset = min(s.offset for s in self.SymbolInfos())
-    max_offset = max(s.offset + s.size for s in self.SymbolInfos())
-    text_length_words = (max_offset - min_offset) / 4
-    offset_to_symbol_info = [None for _ in xrange(text_length_words)]
-    for s in self.SymbolInfos():
-      offset = s.offset - min_offset
-      for i in range(offset / 4, (offset + s.size) / 4):
-        offset_to_symbol_info[i] = s
-    return offset_to_symbol_info
+    if self._offset_to_symbol_info is None:
+      min_offset = min(s.offset for s in self.SymbolInfos())
+      max_offset = max(s.offset + s.size for s in self.SymbolInfos())
+      text_length_words = (max_offset - min_offset) / 4
+      self._offset_to_symbol_info = [None for _ in xrange(text_length_words)]
+      for s in self.SymbolInfos():
+        offset = s.offset - min_offset
+        for i in range(offset / 4, (offset + s.size) / 4):
+          self._offset_to_symbol_info[i] = s
+    return self._offset_to_symbol_info
 
 
 class ProfileManager(object):
@@ -222,11 +260,11 @@
   example the dump for the startup could be phase 0 and then the steady-state
   would be labeled phase 1.
 
-  We assume the files are named like *-TIMESTAMP.SUFFIX_PHASE, where TIMESTAMP
-  is in nanoseconds, SUFFIX is string without dashes, PHASE is an integer
-  numbering the phases as 0, 1, 2..., and the only dot is the one between
-  TIMESTAMP and SUFFIX. Note that the current dump filename also includes a
-  process id which is currently unused.
+  We assume the files are named like
+  profile-hitmap-PROCESS-PID-TIMESTAMP.SUFFIX_PHASE, where PROCESS is a possibly
+  empty string, PID is the process id, TIMESTAMP is in nanoseconds, SUFFIX is
+  string without dashes, PHASE is an integer numbering the phases as 0, 1, 2...,
+  and the only dot is the one between TIMESTAMP and SUFFIX.
 
   This manager supports several configurations of dumps.
 
@@ -242,6 +280,44 @@
     time. This files can be grouped into run sets that are within 30 seconds of
     each other. Each run set is then grouped into phases as before.
   """
+  class AnnotatedOffset(object):
+    """Describes an offset with how it appeared in a profile set.
+
+    Each offset is annotated with the phase and process that it appeared in, and
+    can report how often it occurred in a specific phase and process.
+    """
+    def __init__(self, offset):
+      self._offset = offset
+      self._count = {}
+
+    def __str__(self):
+      return '{}: {}'.format(self._offset, self._count)
+
+    def __eq__(self, other):
+      if other is None:
+        return False
+      return (self._offset == other._offset and
+              self._count == other._count)
+
+    def Increment(self, phase, process):
+      key = (phase, process)
+      self._count[key] = self._count.setdefault(key, 0) + 1
+
+    def Count(self, phase, process):
+      return self._count.get((phase, process), 0)
+
+    def Processes(self):
+      return set(k[1] for k in self._count.iterkeys())
+
+    def Phases(self):
+      return set(k[0] for k in self._count.iterkeys())
+
+    def Offset(self):
+      return self._offset
+
+    def SetOffset(self, o):
+      self._offset = o
+
   class _RunGroup(object):
     RUN_GROUP_THRESHOLD_NS = 30e9
 
@@ -295,6 +371,22 @@
     return self._GetOffsetsForGroup(f for f in self._filenames
                                     if self._Phase(f) == phase)
 
+  def GetAnnotatedOffsets(self):
+    """Merges offsets across run groups and annotates each one.
+
+    Returns:
+      [AnnotatedOffset]
+    """
+    offset_map = {}  # offset int -> AnnotatedOffset
+    for g in self._GetRunGroups():
+      for f in g:
+        phase = self._Phase(f)
+        process = self._ProcessName(f)
+        for offset in self._ReadOffsets(f):
+          offset_map.setdefault(offset, self.AnnotatedOffset(offset)).Increment(
+              phase, process)
+    return offset_map.values()
+
   def GetRunGroupOffsets(self, phase=None):
     """Merges files from each run group and returns offset list for each.
 
@@ -323,10 +415,20 @@
     return [g.Filenames(phase) for g in self._run_groups]
 
   @classmethod
+  def _ProcessName(cls, filename):
+    # The filename starts with 'profile-hitmap-' and ends with
+    # '-PID-TIMESTAMP.text_X'. Anything in between is the process name. The
+    # browser has an empty process name, which is insterted here.
+    process_name_parts = os.path.basename(filename).split('-')[2:-2]
+    if not process_name_parts:
+      return 'browser'
+    return '-'.join(process_name_parts)
+
+  @classmethod
   def _Timestamp(cls, filename):
-      dash_index = filename.rindex('-')
-      dot_index = filename.rindex('.')
-      return int(filename[dash_index+1:dot_index])
+    dash_index = filename.rindex('-')
+    dot_index = filename.rindex('.')
+    return int(filename[dash_index+1:dot_index])
 
   @classmethod
   def _Phase(cls, filename):
@@ -347,6 +449,19 @@
         g.Add(f)
         self._run_groups.append(g)
 
+    # Some sanity checks on the run groups.
+    assert self._run_groups
+    if len(self._run_groups) < 5:
+      return  # Small runs have too much variance for testing.
+    sizes = map(lambda g: len(g.Filenames()), self._run_groups)
+    avg_size = sum(sizes) / len(self._run_groups)
+    num_outliers = len([s for s in sizes
+                        if s > 1.5 * avg_size or s < 0.75 * avg_size])
+    expected_outliers = 0.1 * len(self._run_groups)
+    assert num_outliers < expected_outliers, (
+        'Saw {} outliers instead of at most {} for average of {}'.format(
+            num_outliers, expected_outliers, avg_size))
+
 
 def GetReachedOffsetsFromDumpFiles(dump_filenames, library_filename):
   """Produces a list of symbol offsets reached by the dumps.
diff --git a/tools/cygprofile/process_profiles_unittest.py b/tools/cygprofile/process_profiles_unittest.py
index 77fc31c..c896a49 100755
--- a/tools/cygprofile/process_profiles_unittest.py
+++ b/tools/cygprofile/process_profiles_unittest.py
@@ -10,7 +10,8 @@
 
 import process_profiles
 
-from test_utils import (SimpleTestSymbol,
+from test_utils import (ProfileFile,
+                        SimpleTestSymbol,
                         TestSymbolOffsetProcessor,
                         TestProfileManager)
 
@@ -28,10 +29,10 @@
                          self.symbol_2, self.symbol_3]
     self._file_counter = 0
 
-  def File(self, timestamp_sec, phase):
-    self._file_counter += 1
-    return 'file-{}-{}.txt_{}'.format(
-        self._file_counter, timestamp_sec * 1000 * 1000 * 1000, phase)
+  def MakeAnnotatedOffset(self, offset, counts):
+    ao = process_profiles.ProfileManager.AnnotatedOffset(offset)
+    ao._count = counts
+    return ao
 
   def testGetOffsetToSymbolInfo(self):
     processor = TestSymbolOffsetProcessor(self.symbol_infos)
@@ -103,8 +104,9 @@
     self.assertEquals(5, process_profiles._Median([1, 4, 5, 6, 100]))
 
   def testRunGroups(self):
-    files = [self.File(40, 0), self.File(100, 0), self.File(200, 1),
-             self.File(35, 1), self.File(42, 0), self.File(95, 0)]
+    files = [ProfileFile(40, 0), ProfileFile(100, 0),
+             ProfileFile(200, 1), ProfileFile(35, 1),
+             ProfileFile(42, 0), ProfileFile(95, 0)]
     mgr = process_profiles.ProfileManager(files)
     mgr._ComputeRunGroups()
     self.assertEquals(3, len(mgr._run_groups))
@@ -118,11 +120,34 @@
     self.assertTrue(files[5] in mgr._run_groups[1].Filenames())
     self.assertTrue(files[2] in mgr._run_groups[2].Filenames())
 
+  def testRunGroupSanity(self):
+    files = []
+    # Generate 20 sets of files in groups separated by 60s.
+    for ts_base in xrange(0, 20):
+      ts = ts_base * 60
+      files.extend([ProfileFile(ts, 0, 'browser'),
+                    ProfileFile(ts + 1, 0, 'renderer'),
+                    ProfileFile(ts + 2, 1, 'browser'),
+                    ProfileFile(ts + 3, 0, 'gpu'),
+                    ProfileFile(ts + 2, 1, 'renderer'),
+                    ProfileFile(ts + 5, 1, 'gpu')])
+    # The following call should not assert.
+    process_profiles.ProfileManager(files)._ComputeRunGroups()
+
+    files.extend([ProfileFile(20 * 60, 0, 'browser'),
+                  ProfileFile(20 * 60 + 2, 1, 'renderer'),
+                  ProfileFile(21 * 60, 0, 'browser')] +
+                 [ProfileFile(22 * 60, 0, 'renderer')
+                  for _ in xrange(0, 10)])
+
+    self.assertRaises(AssertionError,
+                      process_profiles.ProfileManager(files)._ComputeRunGroups)
+
   def testReadOffsets(self):
     mgr = TestProfileManager({
-        self.File(30, 0): [1, 3, 5, 7],
-        self.File(40, 1): [8, 10],
-        self.File(50, 0): [13, 15]})
+        ProfileFile(30, 0): [1, 3, 5, 7],
+        ProfileFile(40, 1): [8, 10],
+        ProfileFile(50, 0): [13, 15]})
     self.assertListEqual([1, 3, 5, 7, 8, 10, 13, 15],
                          mgr.GetMergedOffsets())
     self.assertListEqual([8, 10], mgr.GetMergedOffsets(1))
@@ -130,9 +155,9 @@
 
   def testRunGroupOffsets(self):
     mgr = TestProfileManager({
-        self.File(30, 0): [1, 2, 3, 4],
-        self.File(150, 0): [9, 11, 13],
-        self.File(40, 1): [5, 6, 7]})
+        ProfileFile(30, 0): [1, 2, 3, 4],
+        ProfileFile(150, 0): [9, 11, 13],
+        ProfileFile(40, 1): [5, 6, 7]})
     offsets_list = mgr.GetRunGroupOffsets()
     self.assertEquals(2, len(offsets_list))
     self.assertListEqual([1, 2, 3, 4, 5, 6, 7], offsets_list[0])
@@ -150,22 +175,54 @@
     # The fact that the ProfileManager sorts by filename is implicit in the
     # other tests. It is tested explicitly here.
     mgr = TestProfileManager({
-        self.File(40, 0): [1, 2, 3, 4],
-        self.File(150, 0): [9, 11, 13],
-        self.File(30, 1): [5, 6, 7]})
+        ProfileFile(40, 0): [1, 2, 3, 4],
+        ProfileFile(150, 0): [9, 11, 13],
+        ProfileFile(30, 1): [5, 6, 7]})
     offsets_list = mgr.GetRunGroupOffsets()
     self.assertEquals(2, len(offsets_list))
     self.assertListEqual([5, 6, 7, 1, 2, 3, 4], offsets_list[0])
 
   def testPhases(self):
     mgr = TestProfileManager({
-        self.File(40, 0): [],
-        self.File(150, 0): [],
-        self.File(30, 1): [],
-        self.File(30, 2): [],
-        self.File(30, 0): []})
+        ProfileFile(40, 0): [],
+        ProfileFile(150, 0): [],
+        ProfileFile(30, 1): [],
+        ProfileFile(30, 2): [],
+        ProfileFile(30, 0): []})
     self.assertEquals(set([0,1,2]), mgr.GetPhases())
 
+  def testGetAnnotatedOffsets(self):
+    mgr = TestProfileManager({
+        ProfileFile(40, 0, ''): [1, 2, 3],
+        ProfileFile(50, 1, ''): [3, 4, 5],
+        ProfileFile(51, 0, 'renderer'): [2, 3, 6],
+        ProfileFile(51, 1, 'gpu-process'): [6, 7],
+        ProfileFile(70, 0, ''): [2, 8, 9],
+        ProfileFile(70, 1, ''): [9]})
+    offsets = mgr.GetAnnotatedOffsets()
+    self.assertListEqual([
+        self.MakeAnnotatedOffset(1, {(0, 'browser'): 1}),
+        self.MakeAnnotatedOffset(2, {(0, 'browser'): 2,
+                                     (0, 'renderer'): 1}),
+        self.MakeAnnotatedOffset(3, {(0, 'browser'): 1,
+                                     (1, 'browser'): 1,
+                                     (0, 'renderer'): 1}),
+        self.MakeAnnotatedOffset(4, {(1, 'browser'): 1}),
+        self.MakeAnnotatedOffset(5, {(1, 'browser'): 1}),
+        self.MakeAnnotatedOffset(6, {(0, 'renderer'): 1,
+                                     (1, 'gpu-process'): 1}),
+        self.MakeAnnotatedOffset(7, {(1, 'gpu-process'): 1}),
+        self.MakeAnnotatedOffset(8, {(0, 'browser'): 1}),
+        self.MakeAnnotatedOffset(9, {(0, 'browser'): 1,
+                                     (1, 'browser'): 1})],
+                         offsets)
+    self.assertListEqual(['browser', 'renderer'],
+                         sorted(offsets[1].Processes()))
+    self.assertListEqual(['browser'], list(offsets[0].Processes()))
+    self.assertListEqual([0], list(offsets[1].Phases()))
+    self.assertListEqual([0, 1], sorted(offsets[2].Phases()))
+    self.assertListEqual([0, 1], sorted(mgr.GetPhases()))
+
 
 if __name__ == '__main__':
   unittest.main()
diff --git a/tools/cygprofile/test_utils.py b/tools/cygprofile/test_utils.py
index 47043b15..6aacbc9 100644
--- a/tools/cygprofile/test_utils.py
+++ b/tools/cygprofile/test_utils.py
@@ -8,6 +8,9 @@
 
 import process_profiles
 
+# Used by ProfileFile to generate unique file names.
+_FILE_COUNTER = 0
+
 SimpleTestSymbol = collections.namedtuple(
     'SimpleTestSymbol', ['name', 'offset', 'size'])
 
@@ -25,3 +28,14 @@
 
   def _ReadOffsets(self, filename):
     return self._filecontents_mapping[filename]
+
+
+def ProfileFile(timestamp_sec, phase, process_name=None):
+  global _FILE_COUNTER
+  _FILE_COUNTER += 1
+  if process_name:
+    name_str = process_name + '-'
+  else:
+    name_str = ''
+  return 'test-directory/profile-hitmap-{}{}-{}.txt_{}'.format(
+      name_str, _FILE_COUNTER, timestamp_sec * 1000 * 1000 * 1000, phase)
diff --git a/tools/grit/grit.py b/tools/grit/grit.py
index b17ceb9..2813b93 100755
--- a/tools/grit/grit.py
+++ b/tools/grit/grit.py
@@ -13,4 +13,3 @@
 
 if __name__ == '__main__':
   grit.grit_runner.Main(sys.argv[1:])
-
diff --git a/tools/grit/grit/__init__.py b/tools/grit/grit/__init__.py
index 57e6709..a1eeaca 100755
--- a/tools/grit/grit/__init__.py
+++ b/tools/grit/grit/__init__.py
@@ -7,4 +7,3 @@
 '''
 
 pass
-
diff --git a/tools/grit/grit/clique.py b/tools/grit/grit/clique.py
index 7d920a8..7607e5d 100755
--- a/tools/grit/grit/clique.py
+++ b/tools/grit/grit/clique.py
@@ -483,4 +483,3 @@
         language, transl_msg.GetId())
 
     self.clique[language] = transl_msg
-
diff --git a/tools/grit/grit/extern/tclib.py b/tools/grit/grit/extern/tclib.py
index e84f177e..a9b1a39f 100755
--- a/tools/grit/grit/extern/tclib.py
+++ b/tools/grit/grit/extern/tclib.py
@@ -500,4 +500,3 @@
     Returns a copy of this Translation.
     """
     return Translation(None, clone_from=self)
-
diff --git a/tools/grit/grit/format/__init__.py b/tools/grit/grit/format/__init__.py
index 2a3c59ca..28a4dc7 100755
--- a/tools/grit/grit/format/__init__.py
+++ b/tools/grit/grit/format/__init__.py
@@ -7,4 +7,3 @@
 '''
 
 pass
-
diff --git a/tools/grit/grit/gather/regexp.py b/tools/grit/grit/gather/regexp.py
index 30488a6..16bc323a 100755
--- a/tools/grit/grit/gather/regexp.py
+++ b/tools/grit/grit/gather/regexp.py
@@ -82,4 +82,3 @@
 
     if self.single_message_:
       self.skeleton_.append(self.uberclique.MakeClique(self.single_message_))
-
diff --git a/tools/grit/grit/gather/tr_html.py b/tools/grit/grit/gather/tr_html.py
index 3487251..d5122f6e 100755
--- a/tools/grit/grit/gather/tr_html.py
+++ b/tools/grit/grit/gather/tr_html.py
@@ -742,4 +742,3 @@
           continue
       new_skel.append(chunk)
     self.skeleton_ = new_skel
-
diff --git a/tools/grit/grit/gather/txt_unittest.py b/tools/grit/grit/gather/txt_unittest.py
index e2ff8a5..80586526 100755
--- a/tools/grit/grit/gather/txt_unittest.py
+++ b/tools/grit/grit/gather/txt_unittest.py
@@ -31,4 +31,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/node/base.py b/tools/grit/grit/node/base.py
index e40bad88..76fbb31e 100755
--- a/tools/grit/grit/node/base.py
+++ b/tools/grit/grit/node/base.py
@@ -621,4 +621,3 @@
   '''Convenience baseclass for nodes that can have content.'''
   def _ContentType(self):
     return self._CONTENT_TYPE_MIXED
-
diff --git a/tools/grit/grit/node/base_unittest.py b/tools/grit/grit/node/base_unittest.py
index f7e7b75..58acdfb 100755
--- a/tools/grit/grit/node/base_unittest.py
+++ b/tools/grit/grit/node/base_unittest.py
@@ -255,4 +255,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/node/custom/filename_unittest.py b/tools/grit/grit/node/custom/filename_unittest.py
index 9099086..9ea8eb9 100755
--- a/tools/grit/grit/node/custom/filename_unittest.py
+++ b/tools/grit/grit/node/custom/filename_unittest.py
@@ -31,4 +31,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/node/io.py b/tools/grit/grit/node/io.py
index 69146199..83e14b1f 100755
--- a/tools/grit/grit/node/io.py
+++ b/tools/grit/grit/node/io.py
@@ -116,4 +116,3 @@
   def GetEmitType(self):
     '''Returns the emit_type for this node. Default is 'append'.'''
     return self.attrs['emit_type']
-
diff --git a/tools/grit/grit/node/mapping.py b/tools/grit/grit/node/mapping.py
index 259be97..71b250b2 100755
--- a/tools/grit/grit/node/mapping.py
+++ b/tools/grit/grit/node/mapping.py
@@ -58,4 +58,3 @@
   if name not in _ELEMENT_TO_CLASS:
     raise exception.UnknownElement()
   return _ELEMENT_TO_CLASS[name]
-
diff --git a/tools/grit/grit/node/structure.py b/tools/grit/grit/node/structure.py
index 7b6f7bf..3d2012a 100755
--- a/tools/grit/grit/node/structure.py
+++ b/tools/grit/grit/node/structure.py
@@ -356,4 +356,3 @@
     assert hasattr(self, 'gatherer')
     if self.ExpandVariables():
       self.gatherer.SubstituteMessages(substituter)
-
diff --git a/tools/grit/grit/node/variant.py b/tools/grit/grit/node/variant.py
index 4206712b..487f7f9 100755
--- a/tools/grit/grit/node/variant.py
+++ b/tools/grit/grit/node/variant.py
@@ -39,4 +39,3 @@
 
   def GetInputPath(self):
     return self.attrs['file']
-
diff --git a/tools/grit/grit/pseudo.py b/tools/grit/grit/pseudo.py
index 17a6ec6a..cb57daab 100755
--- a/tools/grit/grit/pseudo.py
+++ b/tools/grit/grit/pseudo.py
@@ -125,4 +125,3 @@
       transl.AppendText(PseudoString(part))
 
   return transl
-
diff --git a/tools/grit/grit/shortcuts.py b/tools/grit/grit/shortcuts.py
index 69a4386d..30bdbf7 100755
--- a/tools/grit/grit/shortcuts.py
+++ b/tools/grit/grit/shortcuts.py
@@ -90,4 +90,3 @@
   for group in groups.values():
     warnings += group.GenerateWarnings(tc_project)
   return warnings
-
diff --git a/tools/grit/grit/shortcuts_unittests.py b/tools/grit/grit/shortcuts_unittests.py
index 421cfb2..b7c37b5 100755
--- a/tools/grit/grit/shortcuts_unittests.py
+++ b/tools/grit/grit/shortcuts_unittests.py
@@ -77,4 +77,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/tool/__init__.py b/tools/grit/grit/tool/__init__.py
index c8565d5..4f769f2a 100755
--- a/tools/grit/grit/tool/__init__.py
+++ b/tools/grit/grit/tool/__init__.py
@@ -7,4 +7,3 @@
 '''
 
 pass
-
diff --git a/tools/grit/grit/tool/android2grd.py b/tools/grit/grit/tool/android2grd.py
index c0538ed..e9e0ed1 100755
--- a/tools/grit/grit/tool/android2grd.py
+++ b/tools/grit/grit/tool/android2grd.py
@@ -476,4 +476,3 @@
       return value == 'true'
     else:
       return True
-
diff --git a/tools/grit/grit/tool/count.py b/tools/grit/grit/tool/count.py
index e87c4900..1517b16 100755
--- a/tools/grit/grit/tool/count.py
+++ b/tools/grit/grit/tool/count.py
@@ -32,4 +32,3 @@
         count += 1
 
     print "There are %d occurrences of message %s." % (count, id)
-
diff --git a/tools/grit/grit/tool/menu_from_parts.py b/tools/grit/grit/tool/menu_from_parts.py
index 36d2d40..c7be7bd9 100755
--- a/tools/grit/grit/tool/menu_from_parts.py
+++ b/tools/grit/grit/tool/menu_from_parts.py
@@ -76,4 +76,3 @@
 
     with util.WrapOutputStream(open(output_file, 'w')) as f:
       transl2tc.TranslationToTc.WriteTranslations(f, translations)
-
diff --git a/tools/grit/grit/tool/postprocess_interface.py b/tools/grit/grit/tool/postprocess_interface.py
index 4a43254..c870b932 100755
--- a/tools/grit/grit/tool/postprocess_interface.py
+++ b/tools/grit/grit/tool/postprocess_interface.py
@@ -27,6 +27,3 @@
       The root node of the processed GRD tree.
     '''
     raise NotImplementedError()
-
-
-
diff --git a/tools/grit/grit/tool/postprocess_unittest.py b/tools/grit/grit/tool/postprocess_unittest.py
index 330db490..91f02d69 100755
--- a/tools/grit/grit/tool/postprocess_unittest.py
+++ b/tools/grit/grit/tool/postprocess_unittest.py
@@ -60,4 +60,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/tool/preprocess_interface.py b/tools/grit/grit/tool/preprocess_interface.py
index 4c5456c6..eea10b2 100755
--- a/tools/grit/grit/tool/preprocess_interface.py
+++ b/tools/grit/grit/tool/preprocess_interface.py
@@ -23,6 +23,3 @@
       The processed text.
     '''
     raise NotImplementedError()
-
-
-
diff --git a/tools/grit/grit/tool/preprocess_unittest.py b/tools/grit/grit/tool/preprocess_unittest.py
index 1fc7192..da4242b 100755
--- a/tools/grit/grit/tool/preprocess_unittest.py
+++ b/tools/grit/grit/tool/preprocess_unittest.py
@@ -46,4 +46,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/tool/rc2grd.py b/tools/grit/grit/tool/rc2grd.py
index 10d36f60..a0ef77fc 100755
--- a/tools/grit/grit/tool/rc2grd.py
+++ b/tools/grit/grit/tool/rc2grd.py
@@ -406,4 +406,3 @@
     except:
       print 'Exception processing message with text "%s"' % text
       raise
-
diff --git a/tools/grit/grit/tool/rc2grd_unittest.py b/tools/grit/grit/tool/rc2grd_unittest.py
index b41f5e43..0e48f07f 100755
--- a/tools/grit/grit/tool/rc2grd_unittest.py
+++ b/tools/grit/grit/tool/rc2grd_unittest.py
@@ -134,4 +134,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/tool/test.py b/tools/grit/grit/tool/test.py
index 246b3ded..8f4cdd9 100755
--- a/tools/grit/grit/tool/test.py
+++ b/tools/grit/grit/tool/test.py
@@ -21,4 +21,3 @@
     print 'Options: %s' % repr(global_options)
     print 'Arguments: %s' % repr(my_arguments)
     return 0
-
diff --git a/tools/grit/grit/tool/transl2tc.py b/tools/grit/grit/tool/transl2tc.py
index f3f06a9..a8af6ed16 100755
--- a/tools/grit/grit/tool/transl2tc.py
+++ b/tools/grit/grit/tool/transl2tc.py
@@ -249,4 +249,3 @@
       output_file.write(' ')
       output_file.write(text)
       output_file.write('\n')
-
diff --git a/tools/grit/grit/util_unittest.py b/tools/grit/grit/util_unittest.py
index 03f8cfe..ecadfa3c 100755
--- a/tools/grit/grit/util_unittest.py
+++ b/tools/grit/grit/util_unittest.py
@@ -116,4 +116,3 @@
 
 if __name__ == '__main__':
   unittest.main()
-
diff --git a/tools/grit/grit/xtb_reader.py b/tools/grit/grit/xtb_reader.py
index b92da39e..df6d9f8 100755
--- a/tools/grit/grit/xtb_reader.py
+++ b/tools/grit/grit/xtb_reader.py
@@ -138,4 +138,3 @@
   xml.sax.parse(xtb_file, handler)
   assert handler.language != ''
   return handler.language
-
diff --git a/tools/metrics/histograms/enums.xml b/tools/metrics/histograms/enums.xml
index 4d1aa96..0da6452 100644
--- a/tools/metrics/histograms/enums.xml
+++ b/tools/metrics/histograms/enums.xml
@@ -28641,6 +28641,7 @@
   <int value="341152650" label="SoundContentSetting:enabled"/>
   <int value="345664265" label="BlinkHeapIncrementalMarking:disabled"/>
   <int value="346711293" label="enable-save-password-bubble"/>
+  <int value="348115702" label="new-password-form-parsing-for-saving:enabled"/>
   <int value="348854923" label="v8-cache-strategies-for-cache-storage"/>
   <int value="350399958" label="ModuleScriptsImportMetaUrl:disabled"/>
   <int value="352191859" label="disabled-new-style-notification"/>
@@ -29065,6 +29066,8 @@
   <int value="1194496204" label="NewWallpaperPicker:enabled"/>
   <int value="1196644408" label="performance-monitor-gathering"/>
   <int value="1196834473" label="disable-smart-virtual-keyboard"/>
+  <int value="1201441984"
+      label="new-password-form-parsing-for-saving:disabled"/>
   <int value="1205849612" label="enable-sync-synced-notifications"/>
   <int value="1209221384" label="enable-experimental-accessibility-features"/>
   <int value="1210343926" label="enable-drop-sync-credential"/>
@@ -45702,6 +45705,18 @@
   <int value="2" label="EMPTY_FIELD"/>
 </enum>
 
+<enum name="SyncEntityChange">
+  <summary>
+    Type of change of a sync entity. Recorded once for every sync entity
+    whenever it is commited to the server or updated from the server.
+  </summary>
+  <int value="0" label="Local deletion (commited upstream)"/>
+  <int value="1" label="Local creation (commited upstream)"/>
+  <int value="2" label="Local update (commited upstream)"/>
+  <int value="3" label="Remote deletion (updated downstream)"/>
+  <int value="4" label="Remote update (updated downstream)"/>
+</enum>
+
 <enum name="SyncErrorInfobarTypes">
   <summary>Possible errors that can trigger a sync error infobar.</summary>
   <int value="1" label="Sign in needs update"/>
diff --git a/tools/metrics/histograms/histograms.xml b/tools/metrics/histograms/histograms.xml
index b1d326c..9013798e 100644
--- a/tools/metrics/histograms/histograms.xml
+++ b/tools/metrics/histograms/histograms.xml
@@ -45186,6 +45186,9 @@
 </histogram>
 
 <histogram name="MessageLoop.DelayedTaskQueue.PendingTasksCountOnIdle">
+  <obsolete>
+    Deprecated as of 07/2018.
+  </obsolete>
   <owner>gab@chromium.org</owner>
   <summary>
     The size of the delayed task queue when the loop becomes idle. Diagnosis
@@ -45194,6 +45197,9 @@
 </histogram>
 
 <histogram name="MessageLoop.DelayedTaskQueue.PostedDelay" units="ms">
+  <obsolete>
+    Deprecated as of 07/2018.
+  </obsolete>
   <owner>gab@chromium.org</owner>
   <summary>
     Reports the delay of a delayed task posted to a MessageLoop. Reported once
@@ -45201,7 +45207,20 @@
   </summary>
 </histogram>
 
+<histogram name="MessageLoop.DelayedTaskQueueForUI.PendingTasksCountOnIdle">
+  <owner>gab@chromium.org</owner>
+  <summary>
+    The size of the delayed task queue when the loop becomes idle on a UI
+    thread. Diagnosis metric for https://crbug.com/850450#c4. Note: this metric
+    is a bit broken on Mac OSX as CFRunLoop doesn't deterministically invoke
+    MessageLoop::DoIdleWork().
+  </summary>
+</histogram>
+
 <histogram name="MessageLoop.ScheduledSleep.Completed" units="ms">
+  <obsolete>
+    Deprecated as of 07/2018.
+  </obsolete>
   <owner>gab@chromium.org</owner>
   <summary>
     Reports the delay for which the MessageLoop successfully slept until an
@@ -45211,6 +45230,9 @@
 </histogram>
 
 <histogram name="MessageLoop.ScheduledSleep.Interrupted" units="ms">
+  <obsolete>
+    Deprecated as of 07/2018.
+  </obsolete>
   <owner>gab@chromium.org</owner>
   <summary>
     Reports the delay for which the MessageLoop had planned to sleep (next
@@ -100122,6 +100144,17 @@
   </summary>
 </histogram>
 
+<histogram base="true" name="Sync.ModelTypeEntityChange"
+    enum="SyncEntityChange" expires_after="2020-02-01">
+  <owner>jkrcal@chromium.org</owner>
+  <summary>
+    Recorded once for every sync entity change (whenever it is commited to the
+    server or updated from the server). This metric is used for monitoring
+    general health of sync client-side code. Note: This is only recorded with a
+    data type suffix. The base version is never recorded.
+  </summary>
+</histogram>
+
 <histogram name="Sync.ModelTypeMemoryKB" units="KB">
   <owner>pavely@chromium.org</owner>
   <summary>
@@ -126488,6 +126521,7 @@
   <suffix name="USER_EVENT" label="USER_EVENT"/>
   <suffix name="WIFI_CREDENTIAL" label="WIFI_CREDENTIAL"/>
   <affected-histogram name="Sync.ModelTypeCount"/>
+  <affected-histogram name="Sync.ModelTypeEntityChange"/>
   <affected-histogram name="Sync.ModelTypeMemoryKB"/>
 </histogram_suffixes>
 
diff --git a/tools/perf/expectations.config b/tools/perf/expectations.config
index 6407a9f..b3374da 100644
--- a/tools/perf/expectations.config
+++ b/tools/perf/expectations.config
@@ -255,8 +255,6 @@
 crbug.com/738854 [ Nexus_5X ] system_health.common_mobile/load:tools:drive [ Skip ]
 crbug.com/738854 [ Android_Webview ] system_health.common_mobile/load:tools:drive [ Skip ]
 crbug.com/797261 [ Android_Webview ] system_health.common_mobile/load:games:spychase [ Skip ]
-crbug.com/798536 [ Android ] system_health.common_mobile/background:news:nytimes [ Skip ]
-crbug.com/798536 [ Android ] system_health.common_mobile/load:games:spychase [ Skip ]
 crbug.com/736147 [ Android_One ] system_health.common_mobile/browse:social:tumblr_infinite_scroll [ Skip ]
 crbug.com/803461 [ Nexus_5 ] system_health.common_mobile/browse:chrome:newtab [ Skip ]
 
diff --git a/ui/accelerated_widget_mac/ca_layer_tree_coordinator.mm b/ui/accelerated_widget_mac/ca_layer_tree_coordinator.mm
index 9bc48ec..fb4e271 100644
--- a/ui/accelerated_widget_mac/ca_layer_tree_coordinator.mm
+++ b/ui/accelerated_widget_mac/ca_layer_tree_coordinator.mm
@@ -48,7 +48,7 @@
   if (pending_ca_renderer_layer_tree_) {
     pending_ca_renderer_layer_tree_->CommitScheduledCALayers(
         root_ca_layer_.get(), std::move(current_ca_renderer_layer_tree_),
-        scale_factor_);
+        pixel_size_, scale_factor_);
     current_ca_renderer_layer_tree_.swap(pending_ca_renderer_layer_tree_);
   } else {
     TRACE_EVENT0("gpu", "Blank frame: No overlays or CALayers");
diff --git a/ui/accelerated_widget_mac/ca_layer_tree_unittest_mac.mm b/ui/accelerated_widget_mac/ca_layer_tree_unittest_mac.mm
index 93c7fe4c..ba08f5f6 100644
--- a/ui/accelerated_widget_mac/ca_layer_tree_unittest_mac.mm
+++ b/ui/accelerated_widget_mac/ca_layer_tree_unittest_mac.mm
@@ -80,7 +80,8 @@
   bool result = ScheduleCALayer(new_ca_layer_tree.get(), properties);
   EXPECT_TRUE(result);
   new_ca_layer_tree->CommitScheduledCALayers(
-      superlayer, std::move(ca_layer_tree), properties->scale_factor);
+      superlayer, std::move(ca_layer_tree), properties->rect.size(),
+      properties->scale_factor);
   std::swap(new_ca_layer_tree, ca_layer_tree);
 }
 
@@ -508,8 +509,8 @@
     bool result = ScheduleCALayer(ca_layer_tree.get(), &properties);
     EXPECT_TRUE(result);
   }
-  ca_layer_tree->CommitScheduledCALayers(superlayer_, nullptr,
-                                         properties.scale_factor);
+  ca_layer_tree->CommitScheduledCALayers(
+      superlayer_, nullptr, properties.rect.size(), properties.scale_factor);
 
   // Validate the root layer.
   EXPECT_EQ(1u, [[superlayer_ sublayers] count]);
@@ -588,8 +589,8 @@
     bool result = ScheduleCALayer(ca_layer_tree.get(), &properties);
     EXPECT_TRUE(result);
   }
-  ca_layer_tree->CommitScheduledCALayers(superlayer_, nullptr,
-                                         properties.scale_factor);
+  ca_layer_tree->CommitScheduledCALayers(
+      superlayer_, nullptr, properties.rect.size(), properties.scale_factor);
 
   // Validate the root layer.
   EXPECT_EQ(1u, [[superlayer_ sublayers] count]);
@@ -882,7 +883,8 @@
     bool result = ScheduleCALayer(new_ca_layer_tree.get(), &properties);
     EXPECT_TRUE(result);
     new_ca_layer_tree->CommitScheduledCALayers(
-        superlayer_, std::move(ca_layer_tree), properties.scale_factor);
+        superlayer_, std::move(ca_layer_tree), properties.rect.size(),
+        properties.scale_factor);
     std::swap(new_ca_layer_tree, ca_layer_tree);
 
     // Validate the tree structure.
@@ -896,7 +898,7 @@
     EXPECT_EQ(1u, [[transform_layer sublayers] count]);
 
     // Validate the content layer and fullscreen low power mode.
-    EXPECT_TRUE(CGRectEqualToRect([root_layer frame], CGRectZero));
+    EXPECT_FALSE(CGRectEqualToRect([root_layer frame], CGRectZero));
     EXPECT_NE([root_layer backgroundColor], nil);
   }
 
@@ -909,7 +911,8 @@
     result = ScheduleCALayer(new_ca_layer_tree.get(), &properties);
     EXPECT_TRUE(result);
     new_ca_layer_tree->CommitScheduledCALayers(
-        superlayer_, std::move(ca_layer_tree), properties.scale_factor);
+        superlayer_, std::move(ca_layer_tree), properties.rect.size(),
+        properties.scale_factor);
     std::swap(new_ca_layer_tree, ca_layer_tree);
 
     // Validate the tree structure.
@@ -936,7 +939,8 @@
     result = ScheduleCALayer(new_ca_layer_tree.get(), &properties);
     EXPECT_TRUE(result);
     new_ca_layer_tree->CommitScheduledCALayers(
-        superlayer_, std::move(ca_layer_tree), properties.scale_factor);
+        superlayer_, std::move(ca_layer_tree), properties.rect.size(),
+        properties.scale_factor);
     std::swap(new_ca_layer_tree, ca_layer_tree);
 
     // Validate the tree structure.
@@ -963,7 +967,8 @@
     result = ScheduleCALayer(new_ca_layer_tree.get(), &properties_black);
     EXPECT_TRUE(result);
     new_ca_layer_tree->CommitScheduledCALayers(
-        superlayer_, std::move(ca_layer_tree), properties.scale_factor);
+        superlayer_, std::move(ca_layer_tree), properties.rect.size(),
+        properties.scale_factor);
     std::swap(new_ca_layer_tree, ca_layer_tree);
 
     // Validate the tree structure.
diff --git a/ui/accelerated_widget_mac/ca_renderer_layer_tree.h b/ui/accelerated_widget_mac/ca_renderer_layer_tree.h
index 7c90bae6..a4e7fe3 100644
--- a/ui/accelerated_widget_mac/ca_renderer_layer_tree.h
+++ b/ui/accelerated_widget_mac/ca_renderer_layer_tree.h
@@ -51,6 +51,7 @@
   // not re-used by |this| will be removed from the CALayer hierarchy.
   void CommitScheduledCALayers(CALayer* superlayer,
                                std::unique_ptr<CARendererLayerTree> old_tree,
+                               const gfx::Size& pixel_size,
                                float scale_factor);
 
   // Returns the contents used for a given solid color.
@@ -86,15 +87,12 @@
     // to nil, so that its destructor will not remove an active CALayer.
     void CommitToCA(CALayer* superlayer,
                     RootLayer* old_layer,
+                    const gfx::Size& pixel_size,
                     float scale_factor);
 
-    // Check to see if the CALayer tree is just a video layer on a black
-    // background. If so, return true and set background_rect to the
-    // background's bounding rect, otherwise return false. CommitToCA() calls
-    // this function and, based on its return value, either gives the root
-    // layer this frame and a black background color or clears them.
-    bool WantsFullcreenLowPowerBackdrop(float scale_factor,
-                                        gfx::RectF* background_rect);
+    // Return true if the CALayer tree is just a video layer on a black or
+    // transparent background, false otherwise.
+    bool WantsFullcreenLowPowerBackdrop();
 
     std::vector<ClipAndSortingLayer> clip_and_sorting_layers;
     base::scoped_nsobject<CALayer> ca_layer;
diff --git a/ui/accelerated_widget_mac/ca_renderer_layer_tree.mm b/ui/accelerated_widget_mac/ca_renderer_layer_tree.mm
index 9b64866..1c5988fe 100644
--- a/ui/accelerated_widget_mac/ca_renderer_layer_tree.mm
+++ b/ui/accelerated_widget_mac/ca_renderer_layer_tree.mm
@@ -220,6 +220,7 @@
 void CARendererLayerTree::CommitScheduledCALayers(
     CALayer* superlayer,
     std::unique_ptr<CARendererLayerTree> old_tree,
+    const gfx::Size& pixel_size,
     float scale_factor) {
   TRACE_EVENT0("gpu", "CARendererLayerTree::CommitScheduledCALayers");
   RootLayer* old_root_layer = nullptr;
@@ -229,7 +230,7 @@
       old_root_layer = &old_tree->root_layer_;
   }
 
-  root_layer_.CommitToCA(superlayer, old_root_layer, scale_factor);
+  root_layer_.CommitToCA(superlayer, old_root_layer, pixel_size, scale_factor);
   // If there are any extra CALayers in |old_tree| that were not stolen by this
   // tree, they will be removed from the CALayer tree in this deallocation.
   old_tree.reset();
@@ -237,9 +238,7 @@
   scale_factor_ = scale_factor;
 }
 
-bool CARendererLayerTree::RootLayer::WantsFullcreenLowPowerBackdrop(
-    float scale_factor,
-    gfx::RectF* background_rect) {
+bool CARendererLayerTree::RootLayer::WantsFullcreenLowPowerBackdrop() {
   bool found_video_layer = false;
   for (auto& clip_layer : clip_and_sorting_layers) {
     for (auto& transform_layer : clip_layer.transform_layers) {
@@ -250,7 +249,6 @@
 
         // See if this is the video layer.
         if (content_layer.use_av_layer) {
-          background_rect->Union(gfx::RectF(content_layer.rect));
           found_video_layer = true;
           if (!transform_layer.transform.IsPositiveScaleOrTranslation())
             return false;
@@ -263,15 +261,13 @@
         // solid black or transparent
         if (content_layer.io_surface)
           return false;
-        if (content_layer.background_color == SK_ColorBLACK) {
-          background_rect->Union(gfx::RectF(content_layer.rect));
-        } else if (content_layer.background_color != SK_ColorTRANSPARENT) {
+        if (content_layer.background_color != SK_ColorBLACK &&
+            content_layer.background_color != SK_ColorTRANSPARENT) {
           return false;
         }
       }
     }
   }
-  background_rect->Scale(1 / scale_factor);
   return found_video_layer;
 }
 
@@ -554,6 +550,7 @@
 
 void CARendererLayerTree::RootLayer::CommitToCA(CALayer* superlayer,
                                                 RootLayer* old_layer,
+                                                const gfx::Size& pixel_size,
                                                 float scale_factor) {
   if (old_layer) {
     DCHECK(old_layer->ca_layer);
@@ -569,8 +566,9 @@
     DLOG(ERROR) << "CARendererLayerTree root layer not attached to tree.";
   }
 
-  gfx::RectF bg_rect;
-  if (WantsFullcreenLowPowerBackdrop(scale_factor, &bg_rect)) {
+  if (WantsFullcreenLowPowerBackdrop()) {
+    const gfx::RectF bg_rect(
+        ScaleSize(gfx::SizeF(pixel_size), 1 / scale_factor));
     if (gfx::RectF([ca_layer frame]) != bg_rect)
       [ca_layer setFrame:bg_rect.ToCGRect()];
     if (![ca_layer backgroundColor])
diff --git a/ui/android/java/src/org/chromium/ui/base/SelectFileDialog.java b/ui/android/java/src/org/chromium/ui/base/SelectFileDialog.java
index df8c615..c5d5f32 100644
--- a/ui/android/java/src/org/chromium/ui/base/SelectFileDialog.java
+++ b/ui/android/java/src/org/chromium/ui/base/SelectFileDialog.java
@@ -390,17 +390,18 @@
 
                 if (photos.length == 1) {
                     GetDisplayNameTask task =
-                            new GetDisplayNameTask(ContextUtils.getApplicationContext(), false);
-                    task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, Uri.parse(photos[0]));
+                            new GetDisplayNameTask(ContextUtils.getApplicationContext(), false,
+                                    new Uri[] {Uri.parse(photos[0])});
+                    task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
                     return;
                 } else {
                     Uri[] filePathArray = new Uri[photos.length];
                     for (int i = 0; i < photos.length; ++i) {
                         filePathArray[i] = Uri.parse(photos[i]);
                     }
-                    GetDisplayNameTask task =
-                            new GetDisplayNameTask(ContextUtils.getApplicationContext(), true);
-                    task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, filePathArray);
+                    GetDisplayNameTask task = new GetDisplayNameTask(
+                            ContextUtils.getApplicationContext(), true, filePathArray);
+                    task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
                 }
                 break;
 
@@ -565,9 +566,9 @@
             for (int i = 0; i < itemCount; ++i) {
                 filePathArray[i] = clipData.getItemAt(i).getUri();
             }
-            GetDisplayNameTask task =
-                    new GetDisplayNameTask(ContextUtils.getApplicationContext(), true);
-            task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, filePathArray);
+            GetDisplayNameTask task = new GetDisplayNameTask(
+                    ContextUtils.getApplicationContext(), true, filePathArray);
+            task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
             return;
         }
 
@@ -577,9 +578,9 @@
         }
 
         if (ContentResolver.SCHEME_CONTENT.equals(results.getScheme())) {
-            GetDisplayNameTask task =
-                    new GetDisplayNameTask(ContextUtils.getApplicationContext(), false);
-            task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, results.getData());
+            GetDisplayNameTask task = new GetDisplayNameTask(
+                    ContextUtils.getApplicationContext(), false, new Uri[] {results.getData()});
+            task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
             return;
         }
 
@@ -699,32 +700,34 @@
         return count;
     }
 
-    class GetDisplayNameTask extends AsyncTask<Uri, Void, String[]> {
+    class GetDisplayNameTask extends AsyncTask<Void, Void, String[]> {
         String[] mFilePaths;
         final Context mContext;
         final boolean mIsMultiple;
+        final Uri[] mUris;
 
-        public GetDisplayNameTask(Context context, boolean isMultiple) {
+        public GetDisplayNameTask(Context context, boolean isMultiple, Uri[] uris) {
             mContext = context;
             mIsMultiple = isMultiple;
+            mUris = uris;
         }
 
         @Override
-        public String[] doInBackground(Uri...uris) {
-            mFilePaths = new String[uris.length];
-            String[] displayNames = new String[uris.length];
+        public String[] doInBackground(Void... params) {
+            mFilePaths = new String[mUris.length];
+            String[] displayNames = new String[mUris.length];
             try {
-                for (int i = 0; i < uris.length; i++) {
+                for (int i = 0; i < mUris.length; i++) {
                     // The selected files must be returned as a list of absolute paths. A MIUI 8.5
                     // device was observed to return a file:// URI instead, so convert if necessary.
                     // See https://crbug.com/752834 for context.
-                    if (ContentResolver.SCHEME_FILE.equals(uris[i].getScheme())) {
-                        mFilePaths[i] = uris[i].getSchemeSpecificPart();
+                    if (ContentResolver.SCHEME_FILE.equals(mUris[i].getScheme())) {
+                        mFilePaths[i] = mUris[i].getSchemeSpecificPart();
                     } else {
-                        mFilePaths[i] = uris[i].toString();
+                        mFilePaths[i] = mUris[i].toString();
                     }
                     displayNames[i] = ContentUriUtils.getDisplayName(
-                            uris[i], mContext, MediaStore.MediaColumns.DISPLAY_NAME);
+                            mUris[i], mContext, MediaStore.MediaColumns.DISPLAY_NAME);
                 }
             }  catch (SecurityException e) {
                 // Some third party apps will present themselves as being able
diff --git a/ui/android/junit/src/org/chromium/ui/base/SelectFileDialogTest.java b/ui/android/junit/src/org/chromium/ui/base/SelectFileDialogTest.java
index 101215af..ebcc7d3 100644
--- a/ui/android/junit/src/org/chromium/ui/base/SelectFileDialogTest.java
+++ b/ui/android/junit/src/org/chromium/ui/base/SelectFileDialogTest.java
@@ -133,12 +133,12 @@
     @Test
     public void testMultipleFileSelectorWithFileUris() throws Throwable {
         SelectFileDialog selectFileDialog = new SelectFileDialog(0);
-        SelectFileDialog.GetDisplayNameTask task =
-                selectFileDialog.new GetDisplayNameTask(ContextUtils.getApplicationContext(), true);
         Uri[] filePathArray = new Uri[] {
                 Uri.parse("file:///storage/emulated/0/DCIM/Camera/IMG_0.jpg"),
                 Uri.parse("file:///storage/emulated/0/DCIM/Camera/IMG_1.jpg")};
-        task.doInBackground(filePathArray);
+        SelectFileDialog.GetDisplayNameTask task = selectFileDialog.new GetDisplayNameTask(
+                ContextUtils.getApplicationContext(), true, filePathArray);
+        task.doInBackground();
         assertEquals(task.mFilePaths[0].toString(),
                 "///storage/emulated/0/DCIM/Camera/IMG_0.jpg");
         assertEquals(task.mFilePaths[1].toString(),
diff --git a/ui/gfx/BUILD.gn b/ui/gfx/BUILD.gn
index beae9238..3e0f7dc 100644
--- a/ui/gfx/BUILD.gn
+++ b/ui/gfx/BUILD.gn
@@ -258,29 +258,6 @@
     "//third_party/zlib",
   ]
 
-  # Text rendering conditions (complicated so separated out).
-  if (use_aura || is_mac || is_android || is_fuchsia) {
-    # Mac doesn't use RenderTextHarfBuzz by default yet.
-    sources += [
-      "harfbuzz_font_skia.cc",
-      "harfbuzz_font_skia.h",
-      "render_text_harfbuzz.cc",
-      "render_text_harfbuzz.h",
-      "render_text_mac.h",
-      "render_text_mac.mm",
-      "text_utils_skia.cc",
-    ]
-
-    if (!is_ios) {
-      sources += [
-        "render_text.cc",
-        "render_text.h",
-      ]
-    }
-  } else {
-    # We don't support RenderText on these platforms.
-  }
-
   if (is_android) {
     set_sources_assignment_filter([])
     sources += [
@@ -316,8 +293,17 @@
 
   if (!is_ios) {
     sources += [
+      "harfbuzz_font_skia.cc",
+      "harfbuzz_font_skia.h",
       "paint_vector_icon.cc",
       "paint_vector_icon.h",
+      "render_text.cc",
+      "render_text.h",
+      "render_text_harfbuzz.cc",
+      "render_text_harfbuzz.h",
+      "render_text_mac.h",
+      "render_text_mac.mm",
+      "text_utils_skia.cc",
       "vector_icon_types.h",
     ]
   }