diff --git a/core/lib/Drupal/Core/Datetime/Entity/DateFormat.php b/core/lib/Drupal/Core/Datetime/Entity/DateFormat.php
index bb4efe0aa8c3b730ac8982111c2c01437c83be53..5eb0faf5bf8e805bda256aa28d8f3624c8d9dbc5 100644
--- a/core/lib/Drupal/Core/Datetime/Entity/DateFormat.php
+++ b/core/lib/Drupal/Core/Datetime/Entity/DateFormat.php
@@ -25,6 +25,7 @@
  *     "label" = "label"
  *   },
  *   admin_permission = "administer site configuration",
+ *   list_cache_tags = { "rendered" }
  * )
  */
 class DateFormat extends ConfigEntityBase implements DateFormatInterface {
@@ -98,11 +99,4 @@ public function getCacheTag() {
     return ['rendered'];
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function getListCacheTags() {
-    return ['rendered'];
-  }
-
 }
diff --git a/core/lib/Drupal/Core/Entity/Entity.php b/core/lib/Drupal/Core/Entity/Entity.php
index 07e50c7b6021266e2dc576312d258e996cfa935c..558297b66c9998dc83f17fdfeedd3b2c68e95eae 100644
--- a/core/lib/Drupal/Core/Entity/Entity.php
+++ b/core/lib/Drupal/Core/Entity/Entity.php
@@ -380,7 +380,6 @@ public function preSave(EntityStorageInterface $storage) {
    * {@inheritdoc}
    */
   public function postSave(EntityStorageInterface $storage, $update = TRUE) {
-    $this->onSaveOrDelete();
     $this->invalidateTagsOnSave($update);
   }
 
@@ -406,7 +405,7 @@ public static function preDelete(EntityStorageInterface $storage, array $entitie
    * {@inheritdoc}
    */
   public static function postDelete(EntityStorageInterface $storage, array $entities) {
-    self::invalidateTagsOnDelete($entities);
+    self::invalidateTagsOnDelete($storage->getEntityType(), $entities);
   }
 
   /**
@@ -426,15 +425,8 @@ public function referencedEntities() {
    * {@inheritdoc}
    */
   public function getCacheTag() {
-    return [$this->entityTypeId . ':' . $this->id()];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getListCacheTags() {
     // @todo Add bundle-specific listing cache tag? https://drupal.org/node/2145751
-    return [$this->entityTypeId . 's'];
+    return [$this->entityTypeId . ':' . $this->id()];
   }
 
   /**
@@ -461,26 +453,6 @@ public static function create(array $values = array()) {
     return $entity_manager->getStorage($entity_manager->getEntityTypeFromClass(get_called_class()))->create($values);
   }
 
-
-  /**
-   * Acts on an entity after it was saved or deleted.
-   */
-  protected function onSaveOrDelete() {
-    $referenced_entities = array(
-      $this->getEntityTypeId() => array($this->id() => $this),
-    );
-
-    foreach ($this->referencedEntities() as $referenced_entity) {
-      $referenced_entities[$referenced_entity->getEntityTypeId()][$referenced_entity->id()] = $referenced_entity;
-    }
-
-    foreach ($referenced_entities as $entity_type => $entities) {
-      if ($this->entityManager()->hasHandler($entity_type, 'view_builder')) {
-        $this->entityManager()->getViewBuilder($entity_type)->resetCache($entities);
-      }
-    }
-  }
-
   /**
    * Invalidates an entity's cache tags upon save.
    *
@@ -492,7 +464,7 @@ protected function invalidateTagsOnSave($update) {
     // updated entity may start to appear in a listing because it now meets that
     // listing's filtering requirements. A newly created entity may start to
     // appear in listings because it did not exist before.)
-    $tags = $this->getListCacheTags();
+    $tags = $this->getEntityType()->getListCacheTags();
     if ($update) {
       // An existing entity was updated, also invalidate its unique cache tag.
       $tags = Cache::mergeTags($tags, $this->getCacheTag());
@@ -504,19 +476,20 @@ protected function invalidateTagsOnSave($update) {
   /**
    * Invalidates an entity's cache tags upon delete.
    *
+   * @param \Drupal\Core\Entity\EntityTypeInterface $entity_type
+   *   The entity type definition.
    * @param \Drupal\Core\Entity\EntityInterface[] $entities
    *   An array of entities.
    */
-  protected static function invalidateTagsOnDelete(array $entities) {
-    $tags = array();
+  protected static function invalidateTagsOnDelete(EntityTypeInterface $entity_type, array $entities) {
+    $tags = $entity_type->getListCacheTags();
     foreach ($entities as $entity) {
       // An entity was deleted: invalidate its own cache tag, but also its list
       // cache tags. (A deleted entity may cause changes in a paged list on
       // other pages than the one it's on. The one it's on is handled by its own
       // cache tag, but subsequent list pages would not be invalidated, hence we
       // must invalidate its list cache tags as well.)
-      $tags = Cache::mergeTags($tags, $entity->getCacheTag(), $entity->getListCacheTags());
-      $entity->onSaveOrDelete();
+      $tags = Cache::mergeTags($tags, $entity->getCacheTag());
     }
     Cache::invalidateTags($tags);
   }
diff --git a/core/lib/Drupal/Core/Entity/EntityInterface.php b/core/lib/Drupal/Core/Entity/EntityInterface.php
index 9280d79a4ce82450aa1a5395247f17deda0162fc..4023e10d0e95a7ebc5e51307eed166f8666a4fce 100644
--- a/core/lib/Drupal/Core/Entity/EntityInterface.php
+++ b/core/lib/Drupal/Core/Entity/EntityInterface.php
@@ -399,15 +399,4 @@ public function getTypedData();
    */
   public function getCacheTag();
 
-  /**
-   * The list cache tags associated with this entity.
-   *
-   * Enables code listing entities of this type to ensure that newly created
-   * entities show up immediately.
-   *
-   * @return array
-   *   An array of cache tags.
-   */
-  public function getListCacheTags();
-
 }
diff --git a/core/lib/Drupal/Core/Entity/EntityStorageInterface.php b/core/lib/Drupal/Core/Entity/EntityStorageInterface.php
index 7a329cc9af67b0bfc75f9461e739dd2e37e11759..082e9893b9774521f987bd0a4cfd7f4394f8bc32 100644
--- a/core/lib/Drupal/Core/Entity/EntityStorageInterface.php
+++ b/core/lib/Drupal/Core/Entity/EntityStorageInterface.php
@@ -45,7 +45,7 @@ public function resetCache(array $ids = NULL);
    * @param $ids
    *   An array of entity IDs, or NULL to load all entities.
    *
-   * @return array
+   * @return \Drupal\Core\Entity\EntityInterface[]
    *   An array of entity objects indexed by their IDs. Returns an empty array
    *   if no matching entities found.
    */
diff --git a/core/lib/Drupal/Core/Entity/EntityType.php b/core/lib/Drupal/Core/Entity/EntityType.php
index 8cc1cbe4b63c6a0f97fa81ec3d65cc4bcc4868ad..5ea863d94101523bce04a1f373ae08f8a4f827ac 100644
--- a/core/lib/Drupal/Core/Entity/EntityType.php
+++ b/core/lib/Drupal/Core/Entity/EntityType.php
@@ -202,6 +202,13 @@ class EntityType implements EntityTypeInterface {
    */
   protected $field_ui_base_route;
 
+  /**
+   * The list cache tags for this entity type.
+   *
+   * @var array
+   */
+  protected $list_cache_tags = array();
+
   /**
    * Constructs a new EntityType.
    *
@@ -234,6 +241,12 @@ public function __construct($definition) {
     $this->handlers += array(
       'access' => 'Drupal\Core\Entity\EntityAccessControlHandler',
     );
+
+    // Ensure a default list cache tag is set.
+    if (empty($this->list_cache_tags)) {
+      $this->list_cache_tags = [$definition['id'] . '_list'];
+    }
+
   }
 
   /**
@@ -660,4 +673,11 @@ public function getGroupLabel() {
     return !empty($this->group_label) ? (string) $this->group_label : $this->t('Other', array(), array('context' => 'Entity type group'));
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getListCacheTags() {
+    return $this->list_cache_tags;
+  }
+
 }
diff --git a/core/lib/Drupal/Core/Entity/EntityTypeInterface.php b/core/lib/Drupal/Core/Entity/EntityTypeInterface.php
index 40f5d5d31ab04d3ee4e8a4de9a7cc93b1295b313..f8237a5a37d6b26809d927259f6d33fc3d03eb4c 100644
--- a/core/lib/Drupal/Core/Entity/EntityTypeInterface.php
+++ b/core/lib/Drupal/Core/Entity/EntityTypeInterface.php
@@ -629,4 +629,13 @@ public function getUriCallback();
    */
   public function setUriCallback($callback);
 
+  /**
+   * The list cache tags associated with this entity type.
+   *
+   * Enables code listing entities of this type to ensure that newly created
+   * entities show up immediately.
+   *
+   * @return string[]
+   */
+  public function getListCacheTags();
 }
diff --git a/core/lib/Drupal/Core/Entity/EntityViewBuilder.php b/core/lib/Drupal/Core/Entity/EntityViewBuilder.php
index e24735a005a35ee9b4c9b5f43547e88f66310bd9..42f962b0ba88b9b444e7dfefc2f68ba655fe4cd1 100644
--- a/core/lib/Drupal/Core/Entity/EntityViewBuilder.php
+++ b/core/lib/Drupal/Core/Entity/EntityViewBuilder.php
@@ -349,11 +349,20 @@ public function getCacheTag() {
    * {@inheritdoc}
    */
   public function resetCache(array $entities = NULL) {
+    // If no set of specific entities is provided, invalidate the entity view
+    // builder's cache tag. This will invalidate all entities rendered by this
+    // view builder.
+    // Otherwise, if a set of specific entities is provided, invalidate those
+    // specific entities only, plus their list cache tags, because any lists in
+    // which these entities are rendered, must be invalidated as well. However,
+    // even in this case, we might invalidate more cache items than necessary.
+    // When we have a way to invalidate only those cache items that have both
+    // the individual entity's cache tag and the view builder's cache tag, we'll
+    // be able to optimize this further.
     if (isset($entities)) {
-      // Always invalidate the ENTITY_TYPE_list tag.
-      $tags = array($this->entityTypeId . '_list');
+      $tags = [];
       foreach ($entities as $entity) {
-        $tags = Cache::mergeTags($tags, $entity->getCacheTag());
+        $tags = Cache::mergeTags($tags, $entity->getCacheTag(), $entity->getEntityType()->getListCacheTags());
       }
       Cache::invalidateTags($tags);
     }
diff --git a/core/modules/aggregator/src/Entity/Item.php b/core/modules/aggregator/src/Entity/Item.php
index 0cebc3b32e76b692e611ffa9d36891e43be68ea8..09eed9b66d7d2c0776b5fa14167db5afd92395f2 100644
--- a/core/modules/aggregator/src/Entity/Item.php
+++ b/core/modules/aggregator/src/Entity/Item.php
@@ -31,6 +31,7 @@
  *   uri_callback = "Drupal\aggregator\Entity\Item::buildUri",
  *   base_table = "aggregator_item",
  *   render_cache = FALSE,
+ *   list_cache_tags = { "aggregator_feed_list" },
  *   entity_keys = {
  *     "id" = "iid",
  *     "label" = "title",
@@ -232,13 +233,6 @@ public function getCacheTag() {
     return Feed::load($this->getFeedId())->getCacheTag();
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function getListCacheTags() {
-    return Feed::load($this->getFeedId())->getListCacheTags();
-  }
-
 
   /**
    * Entity URI callback.
diff --git a/core/modules/block/src/BlockViewBuilder.php b/core/modules/block/src/BlockViewBuilder.php
index d29b63c4efb6f8b6e71306f595a751f365e362de..32d75cff3425345991c05e5acf6546c107be7fb7 100644
--- a/core/modules/block/src/BlockViewBuilder.php
+++ b/core/modules/block/src/BlockViewBuilder.php
@@ -73,7 +73,6 @@ public function viewMultiple(array $entities = array(), $view_mode = 'full', $la
       $build[$entity_id]['#cache']['tags'] = Cache::mergeTags(
         $this->getCacheTag(), // Block view builder cache tag.
         $entity->getCacheTag(), // Block entity cache tag.
-        $entity->getListCacheTags(), // Block entity list cache tags.
         $plugin->getCacheTags() // Block plugin cache tags.
       );
 
diff --git a/core/modules/block/src/Entity/Block.php b/core/modules/block/src/Entity/Block.php
index 0036eb835fc79b2e33419e399d03bd9d85bc8b86..20d70a40cd3dcb6a5b38b3eca84d19cde33dc136 100644
--- a/core/modules/block/src/Entity/Block.php
+++ b/core/modules/block/src/Entity/Block.php
@@ -154,16 +154,33 @@ public function calculateDependencies() {
     return $this->dependencies;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function postSave(EntityStorageInterface $storage, $update = TRUE) {
+    parent::postSave($storage, $update);
+
+    // Entity::postSave() calls Entity::invalidateTagsOnSave(), which only
+    // handles the regular cases. The Block entity has one special case: a
+    // newly created block may *also* appear on any page in the current theme,
+    // so we must invalidate the associated block's cache tag (which includes
+    // the theme cache tag).
+    if (!$update) {
+      Cache::invalidateTags($this->getCacheTag());
+    }
+  }
+
   /**
    * {@inheritdoc}
    *
    * Block configuration entities are a special case: one block entity stores
-   * the placement of one block in one theme. Instead of using an entity type-
-   * specific list cache tag like most entities, use the cache tag of the theme
-   * this block is placed in instead.
+   * the placement of one block in one theme. Changing these entities may affect
+   * any page that is rendered in a certain theme, even if the block doesn't
+   * appear there currently. Hence a block configuration entity must also return
+   * the associated theme's cache tag.
    */
-  public function getListCacheTags() {
-    return array('theme:' . $this->theme);
+  public function getCacheTag() {
+    return Cache::mergeTags(parent::getCacheTag(), ['theme:' . $this->theme]);
   }
 
   /**
diff --git a/core/modules/book/book.module b/core/modules/book/book.module
index dc20b1d760a6949ec3e52560b4ac85480a25330c..6b631c7520172248062592f6279a95558d5bee23 100644
--- a/core/modules/book/book.module
+++ b/core/modules/book/book.module
@@ -14,6 +14,7 @@
 use Drupal\Core\Routing\RouteMatchInterface;
 use Drupal\node\NodeInterface;
 use Drupal\node\NodeTypeInterface;
+use Drupal\node\Entity\Node;
 use Drupal\Core\Language\LanguageInterface;
 use Drupal\Core\Entity\Display\EntityViewDisplayInterface;
 use Drupal\Core\Template\Attribute;
@@ -243,6 +244,11 @@ function book_node_view(array &$build, EntityInterface $node, EntityViewDisplayI
             drupal_get_path('module', 'book') . '/css/book.theme.css',
           ),
         ),
+        // The book navigation is a listing of Node entities, so associate its
+        // list cache tag for correct invalidation.
+        '#cache' => [
+          'tags' => $node->getEntityType()->getListCacheTags(),
+        ],
       );
     }
   }
diff --git a/core/modules/book/src/BookExport.php b/core/modules/book/src/BookExport.php
index 4b4507804987214971a4208ce7c8ed0c5e04a738..b71c7a0874d1802011bcb0a23540e30a3ea01a54 100644
--- a/core/modules/book/src/BookExport.php
+++ b/core/modules/book/src/BookExport.php
@@ -8,6 +8,7 @@
 namespace Drupal\book;
 
 use Drupal\Core\Entity\EntityManagerInterface;
+use Drupal\node\Entity\Node;
 use Drupal\node\NodeInterface;
 
 /**
@@ -84,6 +85,9 @@ public function bookExportHtml(NodeInterface $node) {
       '#title' => $node->label(),
       '#contents' => $contents,
       '#depth' => $node->book['depth'],
+      '#cache' => [
+        'tags' => $node->getEntityType()->getListCacheTags(),
+      ],
     );
   }
 
diff --git a/core/modules/book/src/Controller/BookController.php b/core/modules/book/src/Controller/BookController.php
index 7a271b335cc50a1ee15d055cc52a71ee2af64f4b..9e7a11f4acc8fe0639a8ded3def423ddc96ec5b4 100644
--- a/core/modules/book/src/Controller/BookController.php
+++ b/core/modules/book/src/Controller/BookController.php
@@ -10,6 +10,7 @@
 use Drupal\book\BookExport;
 use Drupal\book\BookManagerInterface;
 use Drupal\Core\Controller\ControllerBase;
+use Drupal\node\Entity\Node;
 use Drupal\node\NodeInterface;
 use Symfony\Component\DependencyInjection\Container;
 use Symfony\Component\DependencyInjection\ContainerInterface;
@@ -115,6 +116,9 @@ public function bookRender() {
     return array(
       '#theme' => 'item_list',
       '#items' => $book_list,
+      '#cache' => [
+        'tags' => \Drupal::entityManager()->getDefinition('node')->getListCacheTags(),
+      ],
     );
   }
 
diff --git a/core/modules/comment/src/Plugin/Field/FieldFormatter/CommentDefaultFormatter.php b/core/modules/comment/src/Plugin/Field/FieldFormatter/CommentDefaultFormatter.php
index aa41d1223a6fd7c202f0ba45c5b97d7dff5d499c..02c310775bc382a23753b43f742dd4533aad6074 100644
--- a/core/modules/comment/src/Plugin/Field/FieldFormatter/CommentDefaultFormatter.php
+++ b/core/modules/comment/src/Plugin/Field/FieldFormatter/CommentDefaultFormatter.php
@@ -9,7 +9,9 @@
 
 use Drupal\comment\CommentManagerInterface;
 use Drupal\comment\CommentStorageInterface;
+use Drupal\comment\Entity\Comment;
 use Drupal\comment\Plugin\Field\FieldType\CommentItemInterface;
+use Drupal\Core\Entity\EntityManagerInterface;
 use Drupal\Core\Entity\EntityViewBuilderInterface;
 use Drupal\Core\Entity\EntityFormBuilderInterface;
 use Drupal\Core\Field\FieldItemListInterface;
@@ -67,6 +69,13 @@ public static function defaultSettings() {
    */
   protected $viewBuilder;
 
+  /**
+   * The entity manager.
+   *
+   * @var \Drupal\Core\Entity\EntityManagerInterface
+   */
+  protected $entityManager;
+
   /**
    * The entity form builder.
    *
@@ -87,8 +96,7 @@ public static function create(ContainerInterface $container, array $configuratio
       $configuration['view_mode'],
       $configuration['third_party_settings'],
       $container->get('current_user'),
-      $container->get('entity.manager')->getStorage('comment'),
-      $container->get('entity.manager')->getViewBuilder('comment'),
+      $container->get('entity.manager'),
       $container->get('entity.form_builder')
     );
   }
@@ -112,18 +120,17 @@ public static function create(ContainerInterface $container, array $configuratio
    *   Third party settings.
    * @param \Drupal\Core\Session\AccountInterface $current_user
    *   The current user.
-   * @param \Drupal\comment\CommentStorageInterface $comment_storage
-   *   The comment storage.
-   * @param \Drupal\Core\Entity\EntityViewBuilderInterface $comment_view_builder
-   *   The comment view builder.
+   * @param \Drupal\Core\Entity\EntityManagerInterface $entity_manager
+   *   The entity manager
    * @param \Drupal\Core\Entity\EntityFormBuilderInterface $entity_form_builder
    *   The entity form builder.
    */
-  public function __construct($plugin_id, $plugin_definition, FieldDefinitionInterface $field_definition, array $settings, $label, $view_mode, array $third_party_settings, AccountInterface $current_user, CommentStorageInterface $comment_storage, EntityViewBuilderInterface $comment_view_builder, EntityFormBuilderInterface $entity_form_builder) {
+  public function __construct($plugin_id, $plugin_definition, FieldDefinitionInterface $field_definition, array $settings, $label, $view_mode, array $third_party_settings, AccountInterface $current_user, EntityManagerInterface $entity_manager, EntityFormBuilderInterface $entity_form_builder) {
     parent::__construct($plugin_id, $plugin_definition, $field_definition, $settings, $label, $view_mode, $third_party_settings);
-    $this->viewBuilder = $comment_view_builder;
-    $this->storage = $comment_storage;
+    $this->viewBuilder = $entity_manager->getViewBuilder('comment');
+    $this->storage = $entity_manager->getStorage('comment');
     $this->currentUser = $current_user;
+    $this->entityManager = $entity_manager;
     $this->entityFormBuilder = $entity_form_builder;
   }
 
@@ -150,19 +157,24 @@ public function viewElements(FieldItemListInterface $items) {
       // Unpublished comments are not included in
       // $entity->get($field_name)->comment_count, but unpublished comments
       // should display if the user is an administrator.
-      if ((($entity->get($field_name)->comment_count && $this->currentUser->hasPermission('access comments')) ||
-        $this->currentUser->hasPermission('administer comments'))) {
-        $mode = $comment_settings['default_mode'];
-        $comments_per_page = $comment_settings['per_page'];
-        $comments = $this->storage->loadThread($entity, $field_name, $mode, $comments_per_page, $this->getSetting('pager_id'));
-        if ($comments) {
-          comment_prepare_thread($comments);
-          $build = $this->viewBuilder->viewMultiple($comments);
-          $build['pager']['#theme'] = 'pager';
-          if ($this->getSetting('pager_id')) {
-            $build['pager']['#element'] = $this->getSetting('pager_id');
+      if ($this->currentUser->hasPermission('access comments') || $this->currentUser->hasPermission('administer comments')) {
+        // This is a listing of Comment entities, so associate its list cache
+        // tag for correct invalidation.
+        $output['comments']['#cache']['tags'] = $this->entityManager->getDefinition('comment')->getListCacheTags();
+
+        if ($entity->get($field_name)->comment_count || $this->currentUser->hasPermission('administer comments')) {
+          $mode = $comment_settings['default_mode'];
+          $comments_per_page = $comment_settings['per_page'];
+          $comments = $this->storage->loadThread($entity, $field_name, $mode, $comments_per_page, $this->getSetting('pager_id'));
+          if ($comments) {
+            comment_prepare_thread($comments);
+            $build = $this->viewBuilder->viewMultiple($comments);
+            $build['pager']['#theme'] = 'pager';
+            if ($this->getSetting('pager_id')) {
+              $build['pager']['#element'] = $this->getSetting('pager_id');
+            }
+            $output['comments'] += $build;
           }
-          $output['comments'] = $build;
         }
       }
 
diff --git a/core/modules/comment/src/Tests/CommentDefaultFormatterCacheTagsTest.php b/core/modules/comment/src/Tests/CommentDefaultFormatterCacheTagsTest.php
index 9c1468541b8ef9b7ace964757eaac7eb72751953..8747737f2d872de0506a8d098ac10568fd2673e8 100644
--- a/core/modules/comment/src/Tests/CommentDefaultFormatterCacheTagsTest.php
+++ b/core/modules/comment/src/Tests/CommentDefaultFormatterCacheTagsTest.php
@@ -68,6 +68,7 @@ public function testCacheTags() {
     $expected_cache_tags = array(
       'entity_test_view',
       'entity_test:'  . $commented_entity->id(),
+      'comment_list',
     );
     sort($expected_cache_tags);
     $this->assertEqual($build['#cache']['tags'], $expected_cache_tags, 'The test entity has the expected cache tags before it has comments.');
@@ -96,7 +97,7 @@ public function testCacheTags() {
     //   https://drupal.org/node/597236 lands, it's a temporary work-around.
     $commented_entity = entity_load('entity_test', $commented_entity->id(), TRUE);
 
-    // Verify cache tags on the rendered entity before it has comments.
+    // Verify cache tags on the rendered entity when it has comments.
     $build = \Drupal::entityManager()
       ->getViewBuilder('entity_test')
       ->view($commented_entity);
@@ -104,6 +105,7 @@ public function testCacheTags() {
     $expected_cache_tags = array(
       'entity_test_view',
       'entity_test:' . $commented_entity->id(),
+      'comment_list',
       'comment_view',
       'comment:' . $comment->id(),
       'filter_format:plain_text',
diff --git a/core/modules/comment/tests/src/Unit/Entity/CommentLockTest.php b/core/modules/comment/tests/src/Unit/Entity/CommentLockTest.php
index aef189e7cf183d8f82d9a348b7726ebbe32b3c52..c6f11bcb1b02c1e158118cad70a19a133eab2814 100644
--- a/core/modules/comment/tests/src/Unit/Entity/CommentLockTest.php
+++ b/core/modules/comment/tests/src/Unit/Entity/CommentLockTest.php
@@ -49,8 +49,8 @@ public function testLocks() {
     $methods = get_class_methods('Drupal\comment\Entity\Comment');
     unset($methods[array_search('preSave', $methods)]);
     unset($methods[array_search('postSave', $methods)]);
-    $methods[] = 'onSaveOrDelete';
     $methods[] = 'onUpdateBundleEntity';
+    $methods[] = 'invalidateTagsOnSave';
     $comment = $this->getMockBuilder('Drupal\comment\Entity\Comment')
       ->disableOriginalConstructor()
       ->setMethods($methods)
@@ -79,12 +79,6 @@ public function testLocks() {
       ->method('get')
       ->with('status')
       ->will($this->returnValue((object) array('value' => NULL)));
-    $comment->expects($this->once())
-      ->method('getCacheTag')
-      ->will($this->returnValue(array('comment:' . $cid)));
-    $comment->expects($this->once())
-      ->method('getListCacheTags')
-      ->will($this->returnValue(array('comments')));
     $storage = $this->getMock('Drupal\comment\CommentStorageInterface');
 
     // preSave() should acquire the lock. (This is what's really being tested.)
diff --git a/core/modules/filter/filter.module b/core/modules/filter/filter.module
index 9040b56f81a685e222a12c0337dea7c1c0c7c678..cad200c202a695ea01eb3f3ec65426aff4971e33 100644
--- a/core/modules/filter/filter.module
+++ b/core/modules/filter/filter.module
@@ -14,6 +14,7 @@
 use Drupal\Core\Routing\RouteMatchInterface;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Template\Attribute;
+use Drupal\filter\Entity\FilterFormat;
 use Drupal\filter\FilterFormatInterface;
 
 /**
@@ -139,7 +140,7 @@ function filter_formats(AccountInterface $account = NULL) {
     else {
       $formats['all'] = \Drupal::entityManager()->getStorage('filter_format')->loadByProperties(array('status' => TRUE));
       uasort($formats['all'], 'Drupal\Core\Config\Entity\ConfigEntityBase::sort');
-      \Drupal::cache()->set("filter_formats:{$language_interface->id}", $formats['all'], Cache::PERMANENT, array('filter_formats'));
+      \Drupal::cache()->set("filter_formats:{$language_interface->id}", $formats['all'], Cache::PERMANENT, \Drupal::entityManager()->getDefinition('filter_format')->getListCacheTags());
     }
   }
 
diff --git a/core/modules/filter/src/FilterPluginManager.php b/core/modules/filter/src/FilterPluginManager.php
index 43b6568644c33ed40c457b06d8a9b520026168ef..ccbaf87b0801d39965c98cd86c6254c9774def8f 100644
--- a/core/modules/filter/src/FilterPluginManager.php
+++ b/core/modules/filter/src/FilterPluginManager.php
@@ -37,7 +37,7 @@ class FilterPluginManager extends DefaultPluginManager implements FallbackPlugin
   public function __construct(\Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler) {
     parent::__construct('Plugin/Filter', $namespaces, $module_handler, 'Drupal\filter\Plugin\FilterInterface', 'Drupal\filter\Annotation\Filter');
     $this->alterInfo('filter_info');
-    $this->setCacheBackend($cache_backend, 'filter_plugins', array('filter_formats'));
+    $this->setCacheBackend($cache_backend, 'filter_plugins');
   }
 
   /**
diff --git a/core/modules/hal/src/Tests/EntityTest.php b/core/modules/hal/src/Tests/EntityTest.php
index 591f7ffa2734c25b8e0ca743ef6cdfde79cc9e7a..47aa4d3a0901c0d18caed4347ef4559e4e2f86c0 100644
--- a/core/modules/hal/src/Tests/EntityTest.php
+++ b/core/modules/hal/src/Tests/EntityTest.php
@@ -64,7 +64,8 @@ public function testNode() {
       'body' => array(
         'value' => $this->randomMachineName(),
         'format' => $this->randomMachineName(),
-      )
+      ),
+      'revision_log' => $this->randomString(),
     ));
     $node->save();
 
@@ -160,13 +161,32 @@ public function testComment() {
     ));
     $node->save();
 
+    $parent_comment = entity_create('comment', array(
+      'uid' => $user->id(),
+      'subject' => $this->randomMachineName(),
+      'comment_body' => [
+        'value' => $this->randomMachineName(),
+        'format' => NULL,
+      ],
+      'entity_id' => $node->id(),
+      'entity_type' => 'node',
+      'field_name' => 'comment',
+    ));
+    $parent_comment->save();
+
     $comment = entity_create('comment', array(
       'uid' => $user->id(),
       'subject' => $this->randomMachineName(),
-      'comment_body' => $this->randomMachineName(),
+      'comment_body' => [
+        'value' => $this->randomMachineName(),
+        'format' => NULL,
+      ],
       'entity_id' => $node->id(),
       'entity_type' => 'node',
-      'field_name' => 'comment'
+      'field_name' => 'comment',
+      'pid' => $parent_comment->id(),
+      'mail' => 'dries@drupal.org',
+      'homepage' => 'http://buytaert.net',
     ));
     $comment->save();
 
diff --git a/core/modules/shortcut/src/Entity/Shortcut.php b/core/modules/shortcut/src/Entity/Shortcut.php
index 4bd041efa8c3e438348f07915b0ab4876a6676dd..75d5b2283d51a45cb6e9ec1c9017a607c944e325 100644
--- a/core/modules/shortcut/src/Entity/Shortcut.php
+++ b/core/modules/shortcut/src/Entity/Shortcut.php
@@ -46,6 +46,7 @@
  *     "delete-form" = "entity.shortcut.delete_form",
  *     "edit-form" = "entity.shortcut.canonical",
  *   },
+ *   list_cache_tags = { "shortcut_set_list" },
  *   bundle_entity_type = "shortcut_set"
  * )
  */
@@ -235,11 +236,4 @@ public function getCacheTag() {
     return $this->shortcut_set->entity->getCacheTag();
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function getListCacheTags() {
-    return $this->shortcut_set->entity->getListCacheTags();
-  }
-
 }
diff --git a/core/modules/system/core.api.php b/core/modules/system/core.api.php
index 1863c287caba31c091b0c0f5652bdff7efdc76d0..9dc75de61782dc2a5ae733bfa61c74265b74b043 100644
--- a/core/modules/system/core.api.php
+++ b/core/modules/system/core.api.php
@@ -484,7 +484,7 @@
  * exact same cache tag invalidation as any of the built-in entity types, with
  * the ability to override any of the default behavior if needed.
  * See \Drupal\Core\Entity\EntityInterface::getCacheTag(),
- * \Drupal\Core\Entity\EntityInterface::getListCacheTags(),
+ * \Drupal\Core\Entity\EntityTypeInterface::getListCacheTags(),
  * \Drupal\Core\Entity\Entity::invalidateTagsOnSave() and
  * \Drupal\Core\Entity\Entity::invalidateTagsOnDelete().
  *
diff --git a/core/modules/system/src/Tests/Cache/PageCacheTagsTestBase.php b/core/modules/system/src/Tests/Cache/PageCacheTagsTestBase.php
index e54f97b55dd7443c4d58d6c81b3a0ae5098b9197..8d95233c5d9bbc57fb49b957bcc5cfbd47248020 100644
--- a/core/modules/system/src/Tests/Cache/PageCacheTagsTestBase.php
+++ b/core/modules/system/src/Tests/Cache/PageCacheTagsTestBase.php
@@ -15,6 +15,13 @@
  */
 abstract class PageCacheTagsTestBase extends WebTestBase {
 
+  /**
+   * {@inheritdoc}
+   *
+   * Always enable header dumping in page cache tags tests, this aids debugging.
+   */
+  protected $dumpHeaders = TRUE;
+
   /**
    * {@inheritdoc}
    */
@@ -50,6 +57,7 @@ protected function verifyPageCache($path, $hit_or_miss, $tags = FALSE) {
       $cid = sha1(implode(':', $cid_parts));
       $cache_entry = \Drupal::cache('render')->get($cid);
       sort($cache_entry->tags);
+      $tags = array_unique($tags);
       sort($tags);
       $this->assertIdentical($cache_entry->tags, $tags);
     }
diff --git a/core/modules/system/src/Tests/Entity/EntityCacheTagsTestBase.php b/core/modules/system/src/Tests/Entity/EntityCacheTagsTestBase.php
index b4e3fcedf8e8a7757334f1bcf6cbc10c9219a1de..3c981c12ab4b1c3a4db4431553564a39797a65bf 100644
--- a/core/modules/system/src/Tests/Entity/EntityCacheTagsTestBase.php
+++ b/core/modules/system/src/Tests/Entity/EntityCacheTagsTestBase.php
@@ -137,6 +137,19 @@ protected function getAdditionalCacheTagsForEntity(EntityInterface $entity) {
     return array();
   }
 
+  /**
+   * Returns the additional cache tags for the tested entity's listing by type.
+   *
+   * Necessary when there are unavoidable default entities of this type, e.g.
+   * the anonymous and administrator User entities always exist.
+   *
+   * @return array
+   *   An array of the additional cache tags.
+   */
+  protected function getAdditionalCacheTagsForEntityListing() {
+    return [];
+  }
+
   /**
    * Selects the preferred view mode for the given entity type.
    *
@@ -253,16 +266,19 @@ protected function createReferenceTestEntities($referenced_entity) {
    * Tests cache tags presence and invalidation of the entity when referenced.
    *
    * Tests the following cache tags:
-   * - "<entity type>_view"
-   * - "<entity type>:<entity ID>"
-   * - "<referencing entity type>_view"
-   * * - "<referencing entity type>:<referencing entity ID>"
+   * - entity type view cache tag: "<entity type>_view"
+   * - entity cache tag: "<entity type>:<entity ID>"
+   * - entity type list cache tag: "<entity type>_list"
+   * - referencing entity type view cache tag: "<referencing entity type>_view"
+   * - referencing entity type cache tag: "<referencing entity type>:<referencing entity ID>"
    */
   public function testReferencedEntity() {
     $entity_type = $this->entity->getEntityTypeId();
     $referencing_entity_path = $this->referencing_entity->getSystemPath();
     $non_referencing_entity_path = $this->non_referencing_entity->getSystemPath();
     $listing_path = 'entity_test/list/' . $entity_type . '_reference/' . $entity_type . '/' . $this->entity->id();
+    $empty_entity_listing_path = 'entity_test/list_empty/' . $entity_type;
+    $nonempty_entity_listing_path = 'entity_test/list_labels_alphabetically/' . $entity_type;
 
     $render_cache_tags = array('rendered');
     $theme_cache_tags = array('theme:stark', 'theme_global_settings');
@@ -287,6 +303,21 @@ public function testReferencedEntity() {
       \Drupal::entityManager()->getViewBuilder('entity_test')->getCacheTag()
     );
 
+    // Generate the cache tags for all two possible entity listing paths.
+    // 1. list cache tag only (listing query has no match)
+    // 2. list cache tag plus entity cache tag (listing query has a match)
+    $empty_entity_listing_cache_tags = Cache::mergeTags(
+      $this->entity->getEntityType()->getListCacheTags(),
+      $theme_cache_tags,
+      $render_cache_tags
+    );
+    $nonempty_entity_listing_cache_tags = Cache::mergeTags(
+      $this->entity->getEntityType()->getListCacheTags(),
+      $this->entity->getCacheTag(),
+      $this->getAdditionalCacheTagsForEntityListing($this->entity),
+      $theme_cache_tags,
+      $render_cache_tags
+    );
 
     $this->pass("Test referencing entity.", 'Debug');
     $this->verifyPageCache($referencing_entity_path, 'MISS');
@@ -317,41 +348,62 @@ public function testReferencedEntity() {
     $this->verifyPageCache($listing_path, 'HIT', $tags);
 
 
+    $this->pass("Test empty listing.", 'Debug');
+    // Prime the page cache for the empty listing.
+    $this->verifyPageCache($empty_entity_listing_path, 'MISS');
+    // Verify a cache hit, but also the presence of the correct cache tags.
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT', $empty_entity_listing_cache_tags);
+
+
+    $this->pass("Test listing containing referenced entity.", 'Debug');
+    // Prime the page cache for the listing containing the referenced entity.
+    $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
+    // Verify a cache hit, but also the presence of the correct cache tags.
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT', $nonempty_entity_listing_cache_tags);
+
+
     // Verify that after modifying the referenced entity, there is a cache miss
-    // for both the referencing entity, and the listing of referencing entities,
-    // but not for the non-referencing entity.
+    // for every route except the one for the non-referencing entity.
     $this->pass("Test modification of referenced entity.", 'Debug');
     $this->entity->save();
     $this->verifyPageCache($referencing_entity_path, 'MISS');
     $this->verifyPageCache($listing_path, 'MISS');
+    $this->verifyPageCache($empty_entity_listing_path, 'MISS');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
     $this->verifyPageCache($non_referencing_entity_path, 'HIT');
 
     // Verify cache hits.
     $this->verifyPageCache($referencing_entity_path, 'HIT');
     $this->verifyPageCache($listing_path, 'HIT');
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
 
 
     // Verify that after modifying the referencing entity, there is a cache miss
-    // for both the referencing entity, and the listing of referencing entities,
-    // but not for the non-referencing entity.
+    // for every route except the ones for the non-referencing entity and the
+    // empty entity listing.
     $this->pass("Test modification of referencing entity.", 'Debug');
     $this->referencing_entity->save();
     $this->verifyPageCache($referencing_entity_path, 'MISS');
     $this->verifyPageCache($listing_path, 'MISS');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
     $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT');
 
     // Verify cache hits.
     $this->verifyPageCache($referencing_entity_path, 'HIT');
     $this->verifyPageCache($listing_path, 'HIT');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
 
 
     // Verify that after modifying the non-referencing entity, there is a cache
-    // miss for only the non-referencing entity, not for the referencing entity,
-    // nor for the listing of referencing entities.
+    // miss only for the non-referencing entity route.
     $this->pass("Test modification of non-referencing entity.", 'Debug');
     $this->non_referencing_entity->save();
     $this->verifyPageCache($referencing_entity_path, 'HIT');
     $this->verifyPageCache($listing_path, 'HIT');
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
     $this->verifyPageCache($non_referencing_entity_path, 'MISS');
 
     // Verify cache hits.
@@ -361,7 +413,7 @@ public function testReferencedEntity() {
     if ($this->entity->getEntityType()->hasHandlerClass('view_builder')) {
       // Verify that after modifying the entity's display, there is a cache miss
       // for both the referencing entity, and the listing of referencing
-      // entities, but not for the non-referencing entity.
+      // entities, but not for any other routes.
       $referenced_entity_view_mode = $this->selectViewMode($this->entity->getEntityTypeId());
       $this->pass("Test modification of referenced entity's '$referenced_entity_view_mode' display.", 'Debug');
       $entity_display = entity_get_display($entity_type, $this->entity->bundle(), $referenced_entity_view_mode);
@@ -369,6 +421,8 @@ public function testReferencedEntity() {
       $this->verifyPageCache($referencing_entity_path, 'MISS');
       $this->verifyPageCache($listing_path, 'MISS');
       $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+      $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+      $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
 
       // Verify cache hits.
       $this->verifyPageCache($referencing_entity_path, 'HIT');
@@ -380,17 +434,32 @@ public function testReferencedEntity() {
     if ($bundle_entity_type !== 'bundle') {
       // Verify that after modifying the corresponding bundle entity, there is a
       // cache miss for both the referencing entity, and the listing of
-      // referencing entities, but not for the non-referencing entity.
+      // referencing entities, but not for any other routes.
       $this->pass("Test modification of referenced entity's bundle entity.", 'Debug');
       $bundle_entity = entity_load($bundle_entity_type, $this->entity->bundle());
       $bundle_entity->save();
       $this->verifyPageCache($referencing_entity_path, 'MISS');
       $this->verifyPageCache($listing_path, 'MISS');
       $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+      // Special case: entity types may choose to use their bundle entity type
+      // cache tags, to avoid having excessively granular invalidation.
+      $is_special_case = $bundle_entity->getCacheTag() == $this->entity->getCacheTag() && $bundle_entity->getEntityType()->getListCacheTags() == $this->entity->getEntityType()->getListCacheTags();
+      if ($is_special_case) {
+        $this->verifyPageCache($empty_entity_listing_path, 'MISS');
+        $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
+      }
+      else {
+        $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+        $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
+      }
 
       // Verify cache hits.
       $this->verifyPageCache($referencing_entity_path, 'HIT');
       $this->verifyPageCache($listing_path, 'HIT');
+      if ($is_special_case) {
+        $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+        $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
+      }
     }
 
 
@@ -403,6 +472,8 @@ public function testReferencedEntity() {
       $field_storage->save();
       $this->verifyPageCache($referencing_entity_path, 'MISS');
       $this->verifyPageCache($listing_path, 'MISS');
+      $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+      $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
       $this->verifyPageCache($non_referencing_entity_path, 'HIT');
 
       // Verify cache hits.
@@ -418,6 +489,8 @@ public function testReferencedEntity() {
       $field->save();
       $this->verifyPageCache($referencing_entity_path, 'MISS');
       $this->verifyPageCache($listing_path, 'MISS');
+      $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+      $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
       $this->verifyPageCache($non_referencing_entity_path, 'HIT');
 
       // Verify cache hits.
@@ -426,42 +499,63 @@ public function testReferencedEntity() {
     }
 
 
-    // Verify that after invalidating the entity's cache tag directly,  there is
-    // a cache miss for both the referencing entity, and the listing of
-    // referencing entities, but not for the non-referencing entity.
+    // Verify that after invalidating the entity's cache tag directly, there is
+    // a cache miss for every route except the ones for the non-referencing
+    // entity and the empty entity listing.
     $this->pass("Test invalidation of referenced entity's cache tag.", 'Debug');
     Cache::invalidateTags($this->entity->getCacheTag());
     $this->verifyPageCache($referencing_entity_path, 'MISS');
     $this->verifyPageCache($listing_path, 'MISS');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
     $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT');
 
     // Verify cache hits.
     $this->verifyPageCache($referencing_entity_path, 'HIT');
     $this->verifyPageCache($listing_path, 'HIT');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
+
+    // Verify that after invalidating the entity's list cache tag directly,
+    // there is a cache miss for both the empty entity listing and the non-empty
+    // entity listing routes, but not for other routes.
+    $this->pass("Test invalidation of referenced entity's list cache tag.", 'Debug');
+    Cache::invalidateTags($this->entity->getEntityType()->getListCacheTags());
+    $this->verifyPageCache($empty_entity_listing_path, 'MISS');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
+    $this->verifyPageCache($referencing_entity_path, 'HIT');
+    $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+    $this->verifyPageCache($listing_path, 'HIT');
+
+    // Verify cache hits.
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
 
 
     if (!empty($view_cache_tag)) {
       // Verify that after invalidating the generic entity type's view cache tag
       // directly, there is a cache miss for both the referencing entity, and the
-      // listing of referencing entities, but not for the non-referencing entity.
+      // listing of referencing entities, but not for other routes.
       $this->pass("Test invalidation of referenced entity's 'view' cache tag.", 'Debug');
       Cache::invalidateTags($view_cache_tag);
       $this->verifyPageCache($referencing_entity_path, 'MISS');
       $this->verifyPageCache($listing_path, 'MISS');
       $this->verifyPageCache($non_referencing_entity_path, 'HIT');
+      $this->verifyPageCache($empty_entity_listing_path, 'HIT');
+      $this->verifyPageCache($nonempty_entity_listing_path, 'HIT');
 
       // Verify cache hits.
       $this->verifyPageCache($referencing_entity_path, 'HIT');
       $this->verifyPageCache($listing_path, 'HIT');
     }
 
-    // Verify that after deleting the entity, there is a cache miss for both the
-    // referencing entity, and the listing of referencing entities, but not for
-    // the non-referencing entity.
+    // Verify that after deleting the entity, there is a cache miss for every
+    // route except for the the non-referencing entity one.
     $this->pass('Test deletion of referenced entity.', 'Debug');
     $this->entity->delete();
     $this->verifyPageCache($referencing_entity_path, 'MISS');
     $this->verifyPageCache($listing_path, 'MISS');
+    $this->verifyPageCache($empty_entity_listing_path, 'MISS');
+    $this->verifyPageCache($nonempty_entity_listing_path, 'MISS');
     $this->verifyPageCache($non_referencing_entity_path, 'HIT');
 
     // Verify cache hits.
@@ -473,6 +567,10 @@ public function testReferencedEntity() {
     $this->verifyPageCache($referencing_entity_path, 'HIT', $tags);
     $tags = Cache::mergeTags($render_cache_tags, $theme_cache_tags);
     $this->verifyPageCache($listing_path, 'HIT', $tags);
+    $tags = Cache::mergeTags($render_cache_tags, $theme_cache_tags, $this->entity->getEntityType()->getListCacheTags());
+    $this->verifyPageCache($empty_entity_listing_path, 'HIT', $tags);
+    $tags = Cache::mergeTags($tags, $this->getAdditionalCacheTagsForEntityListing());
+    $this->verifyPageCache($nonempty_entity_listing_path, 'HIT', $tags);
   }
 
   /**
diff --git a/core/modules/system/src/Tests/Entity/EntityViewBuilderTest.php b/core/modules/system/src/Tests/Entity/EntityViewBuilderTest.php
index 547663ce3967080c0fecb77995050ad51b681094..d57b79c663cf0fdb73bdec92e70e5371e343e20b 100644
--- a/core/modules/system/src/Tests/Entity/EntityViewBuilderTest.php
+++ b/core/modules/system/src/Tests/Entity/EntityViewBuilderTest.php
@@ -87,6 +87,7 @@ public function testEntityViewBuilderCacheWithReferences() {
     // Create an entity reference field and an entity that will be referenced.
     entity_reference_create_field('entity_test', 'entity_test', 'reference_field', 'Reference', 'entity_test');
     entity_get_display('entity_test', 'entity_test', 'full')->setComponent('reference_field', [
+      'type' => 'entity_reference_entity_view',
       'settings' => ['link' => FALSE],
     ])->save();
     $entity_test_reference = $this->createTestEntity('entity_test');
@@ -108,6 +109,7 @@ public function testEntityViewBuilderCacheWithReferences() {
     // Create another entity that references the first one.
     $entity_test = $this->createTestEntity('entity_test');
     $entity_test->reference_field->entity = $entity_test_reference;
+    $entity_test->reference_field->access = TRUE;
     $entity_test->save();
 
     // Get a fully built entity view render array.
@@ -124,7 +126,7 @@ public function testEntityViewBuilderCacheWithReferences() {
     $this->assertTrue($this->container->get('cache.' . $bin)->get($cid), 'The entity render element has been cached.');
 
     // Save the entity and verify that both cache entries have been deleted.
-    $entity_test->save();
+    $entity_test_reference->save();
     $this->assertFalse($this->container->get('cache.' . $bin)->get($cid), 'The entity render cache has been cleared when the entity was deleted.');
     $this->assertFalse($this->container->get('cache.' . $bin_reference)->get($cid_reference), 'The entity render cache for the referenced entity has been cleared when the entity was deleted.');
 
diff --git a/core/modules/system/tests/modules/entity_test/entity_test.routing.yml b/core/modules/system/tests/modules/entity_test/entity_test.routing.yml
index 25c159d1bc23ea4603a3c0f1ba447657780aa1c9..e3d0adca2faa6bf908b8ed3227d31c3885bd1df5 100644
--- a/core/modules/system/tests/modules/entity_test/entity_test.routing.yml
+++ b/core/modules/system/tests/modules/entity_test/entity_test.routing.yml
@@ -60,5 +60,22 @@ entity.entity_test.list_referencing_entities:
   requirements:
     _access: 'TRUE'
 
+entity.entity_test.list_labels_alphabetically:
+  path: '/entity_test/list_labels_alphabetically/{entity_type_id}'
+  defaults:
+    _content: '\Drupal\entity_test\Controller\EntityTestController::listEntitiesAlphabetically'
+    _title: 'List labels of entities of the given entity type alphabetically'
+  requirements:
+    _access: 'TRUE'
+
+entity.entity_test.list_empty:
+  path: '/entity_test/list_empty/{entity_type_id}'
+  defaults:
+    _content: '\Drupal\entity_test\Controller\EntityTestController::listEntitiesEmpty'
+    _title: 'Empty list of entities of the given entity type, empty because no entities match the query'
+  requirements:
+    _access: 'TRUE'
+
+
 route_callbacks:
   - '\Drupal\entity_test\Routing\EntityTestRoutes::routes'
diff --git a/core/modules/system/tests/modules/entity_test/src/Controller/EntityTestController.php b/core/modules/system/tests/modules/entity_test/src/Controller/EntityTestController.php
index a9d114aaa6edfe23d3698066af07eb82bda5320f..d189747f4e54dcc2eae5cff4f515da5d0aa14a38 100644
--- a/core/modules/system/tests/modules/entity_test/src/Controller/EntityTestController.php
+++ b/core/modules/system/tests/modules/entity_test/src/Controller/EntityTestController.php
@@ -7,6 +7,7 @@
 
 namespace Drupal\entity_test\Controller;
 
+use Drupal\Core\Cache\Cache;
 use Drupal\Core\Controller\ControllerBase;
 use Drupal\Core\Entity\Query\QueryFactory;
 use Symfony\Component\DependencyInjection\ContainerInterface;
@@ -121,4 +122,71 @@ public function listReferencingEntities($entity_reference_field_name, $reference
       ->viewMultiple($entities, 'full');
   }
 
+  /**
+   * List entities of the given entity type labels, sorted alphabetically.
+   *
+   * @param string $entity_type_id
+   *   The type of the entity being listed.
+   *
+   * @return array
+   *   A renderable array.
+   */
+  public function listEntitiesAlphabetically($entity_type_id) {
+    $entity_type_definition = $this->entityManager()->getDefinition($entity_type_id);
+    $query = $this->entityQueryFactory->get($entity_type_id);
+
+    // Sort by label field, if any.
+    if ($label_field = $entity_type_definition->getKey('label')) {
+      $query->sort($label_field);
+    }
+
+    $entities = $this->entityManager()
+      ->getStorage($entity_type_id)
+      ->loadMultiple($query->execute());
+
+    $cache_tags = [];
+    $labels = [];
+    foreach ($entities as $entity) {
+      $labels[] = $entity->label();
+      $cache_tags = Cache::mergeTags($cache_tags, $entity->getCacheTag());
+    }
+    // Always associate the list cache tag, otherwise the cached empty result
+    // wouldn't be invalidated. This would continue to show nothing matches the
+    // query, even though a newly created entity might match the query.
+    $cache_tags = Cache::mergeTags($cache_tags, $entity_type_definition->getListCacheTags());
+
+    return [
+      '#theme' => 'item_list',
+      '#items' => $labels,
+      '#title' => $entity_type_id . ' entities',
+      '#cache' => [
+        'tags' => $cache_tags,
+      ],
+    ];
+  }
+
+
+  /**
+   * Empty list of entities of the given entity type.
+   *
+   * Empty because no entities match the query. That may seem contrived, but it
+   * is an excellent way for testing whether an entity's list cache tags are
+   * working as expected.
+   *
+   * @param string $entity_type_id
+   *   The type of the entity being listed.
+   *
+   * @return array
+   *   A renderable array.
+   */
+  public function listEntitiesEmpty($entity_type_id) {
+    return [
+      '#theme' => 'item_list',
+      '#items' => [],
+      '#cache' => [
+        'tags' => $this->entityManager()->getDefinition($entity_type_id)->getListCacheTags(),
+      ],
+    ];
+  }
+
 }
diff --git a/core/modules/toolbar/toolbar.module b/core/modules/toolbar/toolbar.module
index 815d1cf11928c9c3965bf771284b3744265fcc31..2fbf64dc6436a9ba258edfc5b88c463a18b99e82 100644
--- a/core/modules/toolbar/toolbar.module
+++ b/core/modules/toolbar/toolbar.module
@@ -13,8 +13,7 @@
 use Drupal\Component\Datetime\DateTimePlus;
 use Drupal\Component\Utility\Crypt;
 use Drupal\Component\Utility\String;
-use Drupal\user\RoleInterface;
-use Drupal\user\UserInterface;
+use Drupal\user\Entity\Role;
 
 /**
  * Implements hook_help().
@@ -371,7 +370,8 @@ function _toolbar_get_subtrees_hash($langcode) {
     // caches later, based on the user's ID regardless of language.
     // Clear the cache when the 'locale' tag is deleted. This ensures a fresh
     // subtrees rendering when string translations are made.
-    \Drupal::cache('toolbar')->set($cid, $hash, Cache::PERMANENT, array('user:' . $uid, 'locale', 'menu:admin', 'user_roles'));
+    $role_list_cache_tags = \Drupal::entityManager()->getDefinition('user_role')->getListCacheTags();
+    \Drupal::cache('toolbar')->set($cid, $hash, Cache::PERMANENT, Cache::mergeTags(array('user:' . $uid, 'locale', 'menu:admin'), $role_list_cache_tags));
   }
   return $hash;
 }
diff --git a/core/modules/user/src/Tests/UserCacheTagsTest.php b/core/modules/user/src/Tests/UserCacheTagsTest.php
index d3ac2948af8925756e15da0739f206967b0a6d6f..4085de4f3f52dd13a2f3a8b19ab84507afe93bc2 100644
--- a/core/modules/user/src/Tests/UserCacheTagsTest.php
+++ b/core/modules/user/src/Tests/UserCacheTagsTest.php
@@ -48,4 +48,11 @@ protected function createEntity() {
     return $user;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function getAdditionalCacheTagsForEntityListing() {
+    return ['user:0', 'user:1'];
+  }
+
 }
diff --git a/core/modules/user/src/Tests/UserPictureTest.php b/core/modules/user/src/Tests/UserPictureTest.php
index d20ade7361ff895ab0e8ac3c85054c4d92c2ffcf..639c38269acec5e895eebfd839113319eaebdc46 100644
--- a/core/modules/user/src/Tests/UserPictureTest.php
+++ b/core/modules/user/src/Tests/UserPictureTest.php
@@ -7,6 +7,7 @@
 
 namespace Drupal\user\Tests;
 
+use Drupal\Core\Cache\Cache;
 use Drupal\simpletest\WebTestBase;
 
 /**
@@ -102,6 +103,9 @@ function testPictureOnNodeComment() {
       ->set('features.comment_user_picture', TRUE)
       ->save();
 
+    // @todo Remove when https://www.drupal.org/node/2040135 lands.
+    Cache::invalidateTags(['rendered']);
+
     $edit = array(
       'comment_body[0][value]' => $this->randomString(),
     );
@@ -113,7 +117,9 @@ function testPictureOnNodeComment() {
       ->set('features.node_user_picture', FALSE)
       ->set('features.comment_user_picture', FALSE)
       ->save();
-    \Drupal::entityManager()->getViewBuilder('comment')->resetCache();
+
+    // @todo Remove when https://www.drupal.org/node/2040135 lands.
+    Cache::invalidateTags(['rendered']);
 
     $this->drupalGet('node/' . $node->id());
     $this->assertNoRaw(file_uri_target($file->getFileUri()), 'User picture not found on node and comment.');
diff --git a/core/modules/views/src/Plugin/views/cache/CachePluginBase.php b/core/modules/views/src/Plugin/views/cache/CachePluginBase.php
index a4626412529197a672a7725ca9021edb2c89a2ef..068197c7681dae30b7eac1c067722b08896b98ca 100644
--- a/core/modules/views/src/Plugin/views/cache/CachePluginBase.php
+++ b/core/modules/views/src/Plugin/views/cache/CachePluginBase.php
@@ -358,9 +358,9 @@ protected function getCacheTags() {
     $entity_information = $this->view->query->getEntityTableInfo();
 
     if (!empty($entity_information)) {
-      // Add an ENTITY_TYPE_list tag for each entity type used by this view.
+      // Add the list cache tags for each entity type used by this view.
       foreach (array_keys($entity_information) as $entity_type) {
-        $tags[] = $entity_type . '_list';
+        $tags = Cache::mergeTags($tags, \Drupal::entityManager()->getDefinition($entity_type)->getListCacheTags());
       }
     }
 
diff --git a/core/modules/views_ui/src/ViewUI.php b/core/modules/views_ui/src/ViewUI.php
index 86d535b7914ad5760d853f5e9c40bc4e591befaf..260a71e843805f1d639ea71bb176e2685bc1f44d 100644
--- a/core/modules/views_ui/src/ViewUI.php
+++ b/core/modules/views_ui/src/ViewUI.php
@@ -1127,13 +1127,6 @@ public function getCacheTag() {
     $this->storage->getCacheTag();
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function getListCacheTags() {
-    $this->storage->getListCacheTags();
-  }
-
   /**
    * {@inheritdoc}
    */
diff --git a/core/tests/Drupal/Tests/Core/Config/Entity/ConfigEntityStorageTest.php b/core/tests/Drupal/Tests/Core/Config/Entity/ConfigEntityStorageTest.php
index 6bf678fd0033939700ac9f925dcd020d78b27a80..4193f42fada47182e25c5fe42aa8628c574afb38 100644
--- a/core/tests/Drupal/Tests/Core/Config/Entity/ConfigEntityStorageTest.php
+++ b/core/tests/Drupal/Tests/Core/Config/Entity/ConfigEntityStorageTest.php
@@ -121,7 +121,9 @@ protected function setUp() {
     $this->entityType->expects($this->any())
       ->method('getClass')
       ->will($this->returnValue(get_class($this->getMockEntity())));
-
+    $this->entityType->expects($this->any())
+      ->method('getListCacheTags')
+      ->willReturn(array('test_entity_type_list'));
 
     $this->moduleHandler = $this->getMock('Drupal\Core\Extension\ModuleHandlerInterface');
 
@@ -241,7 +243,7 @@ public function testSaveInsert(EntityInterface $entity) {
     $this->cacheBackend->expects($this->once())
       ->method('invalidateTags')
       ->with(array(
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     $this->configFactory->expects($this->exactly(2))
@@ -301,7 +303,7 @@ public function testSaveUpdate(EntityInterface $entity) {
       ->method('invalidateTags')
       ->with(array(
         $this->entityTypeId . ':foo', // Own cache tag.
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     $this->configFactory->expects($this->exactly(2))
@@ -361,7 +363,7 @@ public function testSaveRename(ConfigEntityInterface $entity) {
       ->method('invalidateTags')
       ->with(array(
         $this->entityTypeId .':bar', // Own cache tag.
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     $this->configFactory->expects($this->once())
@@ -493,7 +495,7 @@ public function testSaveNoMismatch() {
     $this->cacheBackend->expects($this->once())
       ->method('invalidateTags')
       ->with(array(
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     $this->configFactory->expects($this->once())
@@ -728,7 +730,7 @@ public function testDelete() {
       ->with(array(
         $this->entityTypeId . ':bar', // Own cache tag.
         $this->entityTypeId . ':foo', // Own cache tag.
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     $this->configFactory->expects($this->exactly(2))
@@ -790,7 +792,6 @@ public function testDeleteNothing() {
    * @return \Drupal\Core\Entity\EntityInterface|\PHPUnit_Framework_MockObject_MockObject
    */
   public function getMockEntity(array $values = array(), $methods = array()) {
-    $methods[] = 'onSaveOrDelete';
     $methods[] = 'onUpdateBundleEntity';
     return $this->getMockForAbstractClass('Drupal\Core\Config\Entity\ConfigEntityBase', array($values, 'test_entity_type'), '', TRUE, TRUE, TRUE, $methods);
   }
diff --git a/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php b/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php
index 1fc8d7b1c9ab48930e35d2d9e4fd70a4ba8988d8..a2034e78cfa2a42b111213ad79f8d56ae960088a 100644
--- a/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php
+++ b/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php
@@ -98,6 +98,9 @@ protected function setUp() {
     $this->entityTypeId = $this->randomMachineName();
 
     $this->entityType = $this->getMock('\Drupal\Core\Entity\EntityTypeInterface');
+    $this->entityType->expects($this->any())
+      ->method('getListCacheTags')
+      ->willReturn(array($this->entityTypeId . '_list'));
 
     $this->entityManager = $this->getMock('\Drupal\Core\Entity\EntityManagerInterface');
     $this->entityManager->expects($this->any())
@@ -248,6 +251,7 @@ function setupTestLoad() {
       ->disableOriginalConstructor()
       ->setMethods($methods)
       ->getMock();
+
   }
 
   /**
@@ -392,13 +396,13 @@ public function testPostSave() {
     $this->cacheBackend->expects($this->at(0))
       ->method('invalidateTags')
       ->with(array(
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
     $this->cacheBackend->expects($this->at(1))
       ->method('invalidateTags')
       ->with(array(
         $this->entityTypeId . ':' . $this->values['id'], // Own cache tag.
-        $this->entityTypeId . 's', // List cache tag.
+        $this->entityTypeId . '_list', // List cache tag.
       ));
 
     // This method is internal, so check for errors on calling it only.
@@ -450,18 +454,14 @@ public function testPostDelete() {
       ->method('invalidateTags')
       ->with(array(
         $this->entityTypeId . ':' . $this->values['id'],
-        $this->entityTypeId . 's',
+        $this->entityTypeId . '_list',
       ));
     $storage = $this->getMock('\Drupal\Core\Entity\EntityStorageInterface');
+    $storage->expects($this->once())
+      ->method('getEntityType')
+      ->willReturn($this->entityType);
 
-    $entity = $this->getMockBuilder('\Drupal\Core\Entity\Entity')
-      ->setConstructorArgs(array($this->values, $this->entityTypeId))
-      ->setMethods(array('onSaveOrDelete'))
-      ->getMock();
-    $entity->expects($this->once())
-      ->method('onSaveOrDelete');
-
-    $entities = array($this->values['id'] => $entity);
+    $entities = array($this->values['id'] => $this->entity);
     $this->entity->postDelete($storage, $entities);
   }
 
diff --git a/core/tests/Drupal/Tests/Core/Entity/KeyValueStore/KeyValueEntityStorageTest.php b/core/tests/Drupal/Tests/Core/Entity/KeyValueStore/KeyValueEntityStorageTest.php
index 5c69511d371d33c931a9bcf020a0c4a42b958d49..adfefd59d1afa705b3e3c910486c211bb9a5f463 100644
--- a/core/tests/Drupal/Tests/Core/Entity/KeyValueStore/KeyValueEntityStorageTest.php
+++ b/core/tests/Drupal/Tests/Core/Entity/KeyValueStore/KeyValueEntityStorageTest.php
@@ -100,6 +100,9 @@ protected function setUpKeyValueEntityStorage($uuid_key = 'uuid') {
     $this->entityType->expects($this->atLeastOnce())
       ->method('id')
       ->will($this->returnValue('test_entity_type'));
+    $this->entityType->expects($this->any())
+      ->method('getListCacheTags')
+      ->willReturn(array('test_entity_type_list'));
 
     $this->entityManager = $this->getMock('Drupal\Core\Entity\EntityManagerInterface');
     $this->entityManager->expects($this->any())
@@ -426,7 +429,6 @@ public function testSaveContentEntity() {
     $this->keyValueStore->expects($this->never())
       ->method('delete');
     $entity = $this->getMockEntity('Drupal\Core\Entity\ContentEntityBase', array(), array(
-      'onSaveOrDelete',
       'toArray',
       'id',
     ));