diff --git a/core/core.services.yml b/core/core.services.yml
index f8c428f2d5cfe504fa233eed667b603e3ea4f48e..8f19eb47742ed252a05648b2ad96c4ac12f560fc 100644
--- a/core/core.services.yml
+++ b/core/core.services.yml
@@ -196,6 +196,11 @@ services:
     factory_method: get
     factory_service: logger.factory
     arguments: ['system']
+  logger.channel.php:
+    class: Drupal\Core\Logger\LoggerChannel
+    factory_method: get
+    factory_service: logger.factory
+    arguments: ['php']
   logger.channel.image:
     class: Drupal\Core\Logger\LoggerChannel
     factory_method: get
@@ -736,16 +741,44 @@ services:
     tags:
       - { name: event_subscriber }
     arguments: ['@config.manager', '@config.storage', '@config.storage.snapshot']
-  exception_controller:
-    class: Drupal\Core\Controller\ExceptionController
-    arguments: ['@content_negotiation', '@title_resolver', '@html_page_renderer', '@html_fragment_renderer', '@string_translation', '@url_generator', '@logger.factory']
-    calls:
-      - [setContainer, ['@service_container']]
-  exception_listener:
-    class: Drupal\Core\EventSubscriber\ExceptionListener
+  exception.default_json:
+    class: Drupal\Core\EventSubscriber\ExceptionJsonSubscriber
+    tags:
+      - { name: event_subscriber }
+  exception.default_html:
+    class: Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@html_fragment_renderer', '@html_page_renderer']
+  exception.default:
+    class: Drupal\Core\EventSubscriber\DefaultExceptionSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@html_fragment_renderer', '@html_page_renderer', '@config.factory']
+  exception.logger:
+    class: Drupal\Core\EventSubscriber\ExceptionLoggingSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@logger.factory']
+  exception.custom_page_json:
+    class: Drupal\Core\EventSubscriber\ExceptionJsonSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@config.factory', '@path.alias_manager', '@http_kernel']
+  exception.custom_page_html:
+    class: Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@config.factory', '@path.alias_manager', '@http_kernel', '@logger.channel.php']
+  exception.fast_404_html:
+    class: Drupal\Core\EventSubscriber\Fast404ExceptionHtmlSubscriber
+    tags:
+      - { name: event_subscriber }
+    arguments: ['@config.factory', '@http_kernel']
+  exception.test_site:
+    class: Drupal\Core\EventSubscriber\ExceptionTestSiteSubscriber
     tags:
       - { name: event_subscriber }
-    arguments: [['@exception_controller', execute]]
   route_processor_manager:
     class: Drupal\Core\RouteProcessor\RouteProcessorManager
     tags:
diff --git a/core/includes/menu.inc b/core/includes/menu.inc
index 12d2a239faf97a61a5a443beb82872153e01b21b..a731750aa581acf346f616e3a6a81d007e26e1b2 100644
--- a/core/includes/menu.inc
+++ b/core/includes/menu.inc
@@ -573,7 +573,7 @@ function menu_local_tasks($level = 0) {
     $data['actions'] = array();
 
     $route_name = \Drupal::routeMatch()->getRouteName();
-    if (!empty($route_name)) {
+    if (!\Drupal::request()->attributes->has('exception') && !empty($route_name)) {
       $manager = \Drupal::service('plugin.manager.menu.local_task');
       $local_tasks = $manager->getTasksBuild($route_name);
       foreach ($local_tasks as $level => $items) {
diff --git a/core/lib/Drupal/Core/ContentNegotiation.php b/core/lib/Drupal/Core/ContentNegotiation.php
index 500200cf75135c828d20e3eadad2379b389a67bc..857f9d59f643d58965b1f1d86e89bb77665559c2 100644
--- a/core/lib/Drupal/Core/ContentNegotiation.php
+++ b/core/lib/Drupal/Core/ContentNegotiation.php
@@ -23,7 +23,7 @@ class ContentNegotiation {
    * The normalized type is a short, lowercase version of the format, such as
    * 'html', 'json' or 'atom'.
    *
-   * @param Symfony\Component\HttpFoundation\Request $request
+   * @param \Symfony\Component\HttpFoundation\Request $request
    *   The request object from which to extract the content type.
    *
    * @return string
diff --git a/core/lib/Drupal/Core/Controller/ExceptionController.php b/core/lib/Drupal/Core/Controller/ExceptionController.php
deleted file mode 100644
index 69a477b4ccc04014448ba35392cd08790c50bb64..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/Controller/ExceptionController.php
+++ /dev/null
@@ -1,452 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Controller\ExceptionController.
- */
-
-namespace Drupal\Core\Controller;
-
-use Drupal\Core\Logger\LoggerChannelFactoryInterface;
-use Drupal\Core\Page\DefaultHtmlPageRenderer;
-use Drupal\Core\Page\HtmlFragmentRendererInterface;
-use Drupal\Core\Page\HtmlPageRendererInterface;
-use Drupal\Core\Routing\UrlGeneratorInterface;
-use Symfony\Component\DependencyInjection\ContainerAwareInterface;
-use Symfony\Component\DependencyInjection\ContainerInterface;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\HttpFoundation\Response;
-use Symfony\Component\HttpFoundation\JsonResponse;
-use Symfony\Component\HttpKernel\HttpKernelInterface;
-use Drupal\Component\Utility\SafeMarkup;
-use Drupal\Component\Utility\String;
-use Symfony\Component\Debug\Exception\FlattenException;
-use Drupal\Core\ContentNegotiation;
-use Drupal\Core\Utility\Error;
-use Drupal\Core\StringTranslation\StringTranslationTrait;
-use Drupal\Core\StringTranslation\TranslationInterface;
-
-/**
- * This controller handles HTTP errors generated by the routing system.
- */
-class ExceptionController extends HtmlControllerBase implements ContainerAwareInterface {
-  use StringTranslationTrait;
-
-  /**
-   * The content negotiation library.
-   *
-   * @var \Drupal\Core\ContentNegotiation
-   */
-  protected $negotiation;
-
-  /**
-   * The service container.
-   *
-   * @var \Symfony\Component\DependencyInjection\ContainerInterface
-   */
-  protected $container;
-
-  /**
-   * The page rendering service.
-   *
-   * @var \Drupal\Core\Page\HtmlPageRendererInterface
-   */
-  protected $htmlPageRenderer;
-
-  /**
-   * The fragment rendering service.
-   *
-   * @var \Drupal\Core\Page\HtmlFragmentRendererInterface
-   */
-  protected $fragmentRenderer;
-
-  /**
-   * The logger factory service.
-   *
-   * @var \Drupal\Core\Logger\LoggerChannelFactoryInterface
-   */
-  protected $loggerFactory;
-
-  /**
-   * Constructor.
-   *
-   * @param \Drupal\Core\ContentNegotiation $negotiation
-   *   The content negotiation library to use to determine the correct response
-   *   format.
-   * @param \Drupal\Core\Controller\TitleResolverInterface $title_resolver
-   *   The title resolver.
-   * @param \Drupal\Core\Page\HtmlPageRendererInterface $renderer
-   *   The page renderer.
-   * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer
-   *   The fragment rendering service.
-   * @param \Drupal\Core\StringTranslation\TranslationInterface $string_translation
-   *   The url generator.
-   * @param \Drupal\Core\Routing\UrlGeneratorInterface $url_generator
-   *   The URL generator.
-   * @param \Drupal\Core\Logger\LoggerChannelFactoryInterface $logger_factory
-   *   The logger factory.
-   */
-  public function __construct(ContentNegotiation $negotiation, TitleResolverInterface $title_resolver, HtmlPageRendererInterface $renderer, HtmlFragmentRendererInterface $fragment_renderer, TranslationInterface $string_translation, UrlGeneratorInterface $url_generator, LoggerChannelFactoryInterface $logger_factory) {
-    parent::__construct($title_resolver, $url_generator);
-    $this->negotiation = $negotiation;
-    $this->htmlPageRenderer = $renderer;
-    $this->fragmentRenderer = $fragment_renderer;
-    $this->stringTranslation = $string_translation;
-    $this->loggerFactory = $logger_factory;
-  }
-
-  /**
-   * Sets the Container associated with this Controller.
-   *
-   * @param \Symfony\Component\DependencyInjection\ContainerInterface $container
-   *   A ContainerInterface instance.
-   *
-   * @api
-   */
-  public function setContainer(ContainerInterface $container = NULL) {
-    $this->container = $container;
-  }
-
-  /**
-   * Handles an exception on a request.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request that generated the exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\Response
-   *   A response object.
-   */
-  public function execute(FlattenException $exception, Request $request) {
-    $method = 'on' . $exception->getStatusCode() . $this->negotiation->getContentType($request);
-
-    if (method_exists($this, $method)) {
-      return $this->$method($exception, $request);
-    }
-
-    return new Response('A fatal error occurred: ' . $exception->getMessage(), $exception->getStatusCode(), $exception->getHeaders());
-  }
-
-  /**
-   * Processes a MethodNotAllowed exception into an HTTP 405 response.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\Response
-   *   A response object.
-   */
-  public function on405Html(FlattenException $exception, Request $request) {
-    return new Response('Method Not Allowed', 405);
-  }
-
-  /**
-   * Processes an AccessDenied exception into an HTTP 403 response.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\Response
-   *   A response object.
-   */
-  public function on403Html(FlattenException $exception, Request $request) {
-    // @todo Remove dependency on the internal _system_path attribute:
-    //   https://www.drupal.org/node/2293523.
-    $system_path = $request->attributes->get('_system_path');
-    $this->loggerFactory->get('access denied')->warning($system_path);
-
-    $system_config = $this->container->get('config.factory')->get('system.site');
-    $path = $this->container->get('path.alias_manager')->getPathByAlias($system_config->get('page.403'));
-    if ($path && $path != $system_path) {
-      if ($request->getMethod() === 'POST') {
-        $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'POST', array('destination' => $system_path, '_exception_statuscode' => 403) + $request->request->all(), $request->cookies->all(), array(), $request->server->all());
-      }
-      else {
-        $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'GET', array('destination' => $system_path, '_exception_statuscode' => 403), $request->cookies->all(), array(), $request->server->all());
-      }
-
-      $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST);
-      $response->setStatusCode(403, 'Access denied');
-    }
-    else {
-      $page_content = array(
-        '#markup' => $this->t('You are not authorized to access this page.'),
-        '#title' => $this->t('Access denied'),
-      );
-
-      $fragment = $this->createHtmlFragment($page_content, $request);
-      $page = $this->fragmentRenderer->render($fragment, 403);
-      $response = new Response($this->htmlPageRenderer->render($page), $page->getStatusCode());
-      return $response;
-    }
-
-    return $response;
-  }
-
-  /**
-   * Processes a NotFound exception into an HTTP 404 response.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\Response
-   *   A response object.
-   */
-  public function on404Html(FlattenException $exception, Request $request) {
-    $this->loggerFactory->get('page not found')->warning(String::checkPlain($request->attributes->get('_system_path')));
-
-    // Check for and return a fast 404 page if configured.
-    $config = \Drupal::config('system.performance');
-
-    $exclude_paths = $config->get('fast_404.exclude_paths');
-    if ($config->get('fast_404.enabled') && $exclude_paths && !preg_match($exclude_paths, $request->getPathInfo())) {
-      $fast_paths = $config->get('fast_404.paths');
-      if ($fast_paths && preg_match($fast_paths, $request->getPathInfo())) {
-        $fast_404_html = $config->get('fast_404.html');
-        $fast_404_html = strtr($fast_404_html, array('@path' => String::checkPlain($request->getUri())));
-        return new Response($fast_404_html, 404);
-      }
-    }
-
-    // @todo Remove dependency on the internal _system_path attribute:
-    //   https://www.drupal.org/node/2293523.
-    $system_path = $request->attributes->get('_system_path');
-
-    $path = $this->container->get('path.alias_manager')->getPathByAlias(\Drupal::config('system.site')->get('page.404'));
-    if ($path && $path != $system_path) {
-      // @todo Um, how do I specify an override URL again? Totally not clear. Do
-      //   that and sub-call the kernel rather than using meah().
-      // @todo The create() method expects a slash-prefixed path, but we store a
-      //   normal system path in the site_404 variable.
-      if ($request->getMethod() === 'POST') {
-        $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'POST', array('destination' => $system_path, '_exception_statuscode' => 404) + $request->request->all(), $request->cookies->all(), array(), $request->server->all());
-      }
-      else {
-        $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'GET', array('destination' => $system_path, '_exception_statuscode' => 404), $request->cookies->all(), array(), $request->server->all());
-      }
-
-      $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST);
-      $response->setStatusCode(404, 'Not Found');
-    }
-    else {
-      $page_content = array(
-        '#markup' => $this->t('The requested page "@path" could not be found.', array('@path' => $request->getPathInfo())),
-        '#title' => $this->t('Page not found'),
-      );
-
-      $fragment = $this->createHtmlFragment($page_content, $request);
-      $page = $this->fragmentRenderer->render($fragment, 404);
-      $response = new Response($this->htmlPageRenderer->render($page), $page->getStatusCode());
-      return $response;
-    }
-
-    return $response;
-  }
-
-  /**
-   * Processes a generic exception into an HTTP 500 response.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   Metadata about the exception that was thrown.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\Response
-   *   A response object.
-   */
-  public function on500Html(FlattenException $exception, Request $request) {
-    $error = $this->decodeException($exception);
-
-    // Because the kernel doesn't run until full bootstrap, we know that
-    // most subsystems are already initialized.
-
-    $headers = array();
-
-    // When running inside the testing framework, we relay the errors
-    // to the tested site by the way of HTTP headers.
-    if (DRUPAL_TEST_IN_CHILD_SITE && !headers_sent() && (!defined('SIMPLETEST_COLLECT_ERRORS') || SIMPLETEST_COLLECT_ERRORS)) {
-      // $number does not use drupal_static as it should not be reset
-      // as it uniquely identifies each PHP error.
-      static $number = 0;
-      $assertion = array(
-        $error['!message'],
-        $error['%type'],
-        array(
-          'function' => $error['%function'],
-          'file' => $error['%file'],
-          'line' => $error['%line'],
-        ),
-      );
-      $headers['X-Drupal-Assertion-' . $number] = rawurlencode(serialize($assertion));
-      $number++;
-    }
-
-    $this->loggerFactory->get('php')->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error);
-
-    // Display the message if the current error reporting level allows this type
-    // of message to be displayed, and unconditionnaly in update.php.
-    if (error_displayable($error)) {
-      $class = 'error';
-
-      // If error type is 'User notice' then treat it as debug information
-      // instead of an error message.
-      // @see debug()
-      if ($error['%type'] == 'User notice') {
-        $error['%type'] = 'Debug';
-        $class = 'status';
-      }
-
-      // Attempt to reduce verbosity by removing DRUPAL_ROOT from the file path
-      // in the message. This does not happen for (false) security.
-      $root_length = strlen(DRUPAL_ROOT);
-      if (substr($error['%file'], 0, $root_length) == DRUPAL_ROOT) {
-        $error['%file'] = substr($error['%file'], $root_length + 1);
-      }
-      // Should not translate the string to avoid errors producing more errors.
-      $message = String::format('%type: !message in %function (line %line of %file).', $error);
-
-      // Check if verbose error reporting is on.
-      $error_level = $this->container->get('config.factory')->get('system.logging')->get('error_level');
-
-      if ($error_level == ERROR_REPORTING_DISPLAY_VERBOSE) {
-        $backtrace_exception = $exception;
-        while ($backtrace_exception->getPrevious()) {
-          $backtrace_exception = $backtrace_exception->getPrevious();
-        }
-        $backtrace = $backtrace_exception->getTrace();
-        // First trace is the error itself, already contained in the message.
-        // While the second trace is the error source and also contained in the
-        // message, the message doesn't contain argument values, so we output it
-        // once more in the backtrace.
-        array_shift($backtrace);
-        // Generate a backtrace containing only scalar argument values.
-        $message .= '<pre class="backtrace">' . Error::formatFlattenedBacktrace($backtrace) . '</pre>';
-      }
-      drupal_set_message(SafeMarkup::set($message), $class, TRUE);
-    }
-
-    $content = $this->t('The website has encountered an error. Please try again later.');
-    $output = DefaultHtmlPageRenderer::renderPage($content, $this->t('Error'));
-    $response = new Response($output);
-    $response->setStatusCode(500, '500 Service unavailable (with message)');
-
-    return $response;
-  }
-
-  /**
-   * Processes an AccessDenied exception that occurred on a JSON request.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\JsonResponse
-   *   A JSON response object.
-   */
-  public function on403Json(FlattenException $exception, Request $request) {
-    $response = new JsonResponse();
-    $response->setStatusCode(403, 'Access Denied');
-    return $response;
-  }
-
-  /**
-   * Processes a NotFound exception that occurred on a JSON request.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\JsonResponse
-   *   A JSON response object.
-   */
-  public function on404Json(FlattenException $exception, Request $request) {
-    $response = new JsonResponse();
-    $response->setStatusCode(404, 'Not Found');
-    return $response;
-  }
-
-  /**
-   * Processes a MethodNotAllowed exception that occurred on a JSON request.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *   The flattened exception.
-   * @param \Symfony\Component\HttpFoundation\Request $request
-   *   The request object that triggered this exception.
-   *
-   * @return \Symfony\Component\HttpFoundation\JsonResponse
-   *   A JSON response object.
-   */
-  public function on405Json(FlattenException $exception, Request $request) {
-    $response = new JsonResponse();
-    $response->setStatusCode(405, 'Method Not Allowed');
-    return $response;
-  }
-
-
-  /**
-   * This method is a temporary port of _drupal_decode_exception().
-   *
-   * @todo This should get refactored. FlattenException could use some
-   *   improvement as well.
-   *
-   * @param \Symfony\Component\Debug\Exception\FlattenException $exception
-   *  The flattened exception.
-   *
-   * @return array
-   *   An array of string-substitution tokens for formatting a message about the
-   *   exception.
-   */
-  protected function decodeException(FlattenException $exception) {
-    $message = $exception->getMessage();
-
-    $backtrace = $exception->getTrace();
-
-    // This value is missing from the stack for some reason in the
-    // FlattenException version of the backtrace.
-    $backtrace[0]['line'] = $exception->getLine();
-
-    // For database errors, we try to return the initial caller,
-    // skipping internal functions of the database layer.
-    if (strpos($exception->getClass(), 'DatabaseExceptionWrapper') !== FALSE) {
-      // A DatabaseExceptionWrapper exception is actually just a courier for
-      // the original PDOException.  It's the stack trace from that exception
-      // that we care about.
-      $backtrace = $exception->getPrevious()->getTrace();
-      $backtrace[0]['line'] = $exception->getLine();
-
-      // The first element in the stack is the call, the second element gives us the caller.
-      // We skip calls that occurred in one of the classes of the database layer
-      // or in one of its global functions.
-      $db_functions = array('db_query',  'db_query_range');
-      while (!empty($backtrace[1]) && ($caller = $backtrace[1]) &&
-          ((strpos($caller['namespace'], 'Drupal\Core\Database') !== FALSE || strpos($caller['class'], 'PDO') !== FALSE)) ||
-          in_array($caller['function'], $db_functions)) {
-        // We remove that call.
-        array_shift($backtrace);
-      }
-    }
-
-    $caller = Error::getLastCaller($backtrace);
-
-    return array(
-      '%type' => $exception->getClass(),
-      // The standard PHP exception handler considers that the exception message
-      // is plain-text. We mimick this behavior here.
-      '!message' => String::checkPlain($message),
-      '%function' => $caller['function'],
-      '%file' => $caller['file'],
-      '%line' => $caller['line'],
-      'severity_level' => WATCHDOG_ERROR,
-    );
-  }
-
-}
diff --git a/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php
index 87a3cfe1ae495a29f831308b7470fbf36f63c2c8..80f77962f67f6f0921a93629090fbe471ea79186 100644
--- a/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php
+++ b/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php
@@ -70,8 +70,8 @@ public function onException(GetResponseForExceptionEvent $event) {
    * Cookie provider to send all relevant session data to the user.
    */
   public static function getSubscribedEvents() {
-    $events[KernelEvents::RESPONSE][] = array('onRespond', 0);
-    $events[KernelEvents::EXCEPTION][] = array('onException', 0);
+    $events[KernelEvents::RESPONSE][] = ['onRespond', 0];
+    $events[KernelEvents::EXCEPTION][] = ['onException', 75];
     return $events;
   }
 }
diff --git a/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..ae3d9447632e1281c506f59bf4ae08966d0de02d
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php
@@ -0,0 +1,149 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\Config\ConfigFactoryInterface;
+use Drupal\Core\Path\AliasManagerInterface;
+use Drupal\Core\Utility\Error;
+use Psr\Log\LoggerInterface;
+use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\HttpKernelInterface;
+
+/**
+ * Exception subscriber for handling core custom error pages.
+ */
+class CustomPageExceptionHtmlSubscriber extends HttpExceptionSubscriberBase {
+
+  /**
+   * The configuration factory.
+   *
+   * @var \Drupal\Core\Config\ConfigFactoryInterface
+   */
+  protected $configFactory;
+
+  /**
+   * The page alias manager.
+   *
+   * @var \Drupal\Core\Path\AliasManagerInterface
+   */
+  protected $aliasManager;
+
+  /**
+   * The HTTP kernel.
+   *
+   * @var \Symfony\Component\HttpKernel\HttpKernelInterface
+   */
+  protected $httpKernel;
+
+  /**
+   * The logger instance.
+   *
+   * @var \Psr\Log\LoggerInterface
+   */
+  protected $logger;
+
+  /**
+   * Constructs a new CustomPageExceptionHtmlSubscriber.
+   *
+   * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
+   *   The configuration factory.
+   * @param \Drupal\Core\Path\AliasManagerInterface $alias_manager
+   *   The alias manager service.
+   * @param \Symfony\Component\HttpKernel\HttpKernelInterface $http_kernel
+   *   The HTTP Kernel service.
+   * @param \Psr\Log\LoggerInterface $logger
+   *   The logger service.
+   */
+  public function __construct(ConfigFactoryInterface $config_factory, AliasManagerInterface $alias_manager, HttpKernelInterface $http_kernel, LoggerInterface $logger) {
+    $this->configFactory = $config_factory;
+    $this->aliasManager = $alias_manager;
+    $this->httpKernel = $http_kernel;
+    $this->logger = $logger;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static function getPriority() {
+    return -50;
+  }
+
+  /**
+   * {@inheritDoc}
+   */
+  protected function getHandledFormats() {
+    return ['html'];
+  }
+
+  /**
+   * Handles a 403 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on403(GetResponseForExceptionEvent $event) {
+    $path = $this->aliasManager->getPathByAlias($this->configFactory->get('system.site')->get('page.403'));
+    $this->makeSubrequest($event, $path, Response::HTTP_FORBIDDEN);
+  }
+
+  /**
+   * Handles a 404 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on404(GetResponseForExceptionEvent $event) {
+    $path = $this->aliasManager->getPathByAlias($this->configFactory->get('system.site')->get('page.404'));
+    $this->makeSubrequest($event, $path, Response::HTTP_NOT_FOUND);
+  }
+
+  /**
+   * Makes a subrequest to retrieve a custom error page.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process
+   * @param string $path
+   *   The path to which to make a subrequest for this error message.
+   * @param int $status_code
+   *   The status code for the error being handled.
+   */
+  protected function makeSubrequest(GetResponseForExceptionEvent $event, $path, $status_code) {
+    $request = $event->getRequest();
+
+    // @todo Remove dependency on the internal _system_path attribute:
+    //   https://www.drupal.org/node/2293523.
+    $system_path = $request->attributes->get('_system_path');
+
+    if ($path && $path != $system_path) {
+      // @todo The create() method expects a slash-prefixed path, but we store a
+      //   normal system path in the site_404 variable.
+      if ($request->getMethod() === 'POST') {
+        $sub_request = Request::create($request->getBaseUrl() . '/' . $path, 'POST', ['destination' => $system_path, '_exception_statuscode' => $status_code] + $request->request->all(), $request->cookies->all(), [], $request->server->all());
+      }
+      else {
+        $sub_request = Request::create($request->getBaseUrl() . '/' . $path, 'GET', $request->query->all() + ['destination' => $system_path, '_exception_statuscode' => $status_code], $request->cookies->all(), [], $request->server->all());
+      }
+
+      try {
+        $response = $this->httpKernel->handle($sub_request, HttpKernelInterface::SUB_REQUEST);
+        $response->setStatusCode($status_code);
+        $event->setResponse($response);
+      }
+      catch (\Exception $e) {
+        // If an error happened in the subrequest we can't do much else.
+        // Instead, just log it.  The DefaultExceptionHandler will catch the
+        // original exception and handle it normally.
+        $error = Error::decodeException($e);
+        $this->logger->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error);
+      }
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..0df6a3073a1c1c663e89e86fa899e11902f6b18b
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php
@@ -0,0 +1,121 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\Page\HtmlFragment;
+use Drupal\Core\Page\HtmlFragmentRendererInterface;
+use Drupal\Core\Page\HtmlPageRendererInterface;
+use Drupal\Core\StringTranslation\StringTranslationTrait;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+
+/**
+ * Handle most HTTP errors for HTML.
+ */
+class DefaultExceptionHtmlSubscriber extends HttpExceptionSubscriberBase {
+  use StringTranslationTrait;
+
+  /**
+   * The HTML fragment renderer.
+   *
+   * @var \Drupal\Core\Page\HtmlFragmentRendererInterface
+   */
+  protected $fragmentRenderer;
+
+  /**
+   * The HTML page renderer.
+   *
+   * @var \Drupal\Core\Page\HtmlPageRendererInterface
+   */
+  protected $htmlPageRenderer;
+
+  /**
+   * Constructs a new DefaultExceptionHtmlSubscriber.
+   *
+   * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer
+   *   The fragment renderer.
+   * @param \Drupal\Core\Page\HtmlPageRendererInterface $page_renderer
+   *   The page renderer.
+   */
+  public function __construct(HtmlFragmentRendererInterface $fragment_renderer, HtmlPageRendererInterface $page_renderer) {
+    $this->fragmentRenderer = $fragment_renderer;
+    $this->htmlPageRenderer = $page_renderer;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static function getPriority() {
+    // A very low priority so that custom handlers are almost certain to fire
+    // before it, even if someone forgets to set a priority.
+    return -128;
+  }
+
+  /**
+   * {@inheritDoc}
+   */
+  protected function getHandledFormats() {
+    return ['html'];
+  }
+
+  /**
+   * Handles a 403 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on403(GetResponseForExceptionEvent $event) {
+    $response = $this->createResponse($this->t('Access denied'), $this->t('You are not authorized to access this page.'), Response::HTTP_FORBIDDEN);
+    $response->headers->set('Content-type', 'text/html');
+    $event->setResponse($response);
+  }
+
+  /**
+   * Handles a 404 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on404(GetResponseForExceptionEvent $event) {
+    $path = $event->getRequest()->getPathInfo();
+    $response = $this->createResponse($this->t('Page not found'), $this->t('The requested page "@path" could not be found.', ['@path' => $path]), Response::HTTP_NOT_FOUND);
+    $response->headers->set('Content-type', 'text/html');
+    $event->setResponse($response);
+  }
+
+  /**
+   * Handles a 405 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on405(GetResponseForExceptionEvent $event) {
+    $response = new Response('Method Not Allowed', Response::HTTP_METHOD_NOT_ALLOWED);
+    $response->headers->set('Content-type', 'text/html');
+    $event->setResponse($response);
+  }
+
+  /**
+   * @param $title
+   *   The page title of the response.
+   * @param $body
+   *   The body of the error page.
+   * @param $response_code
+   *   The HTTP response code of the response.
+   * @return Response
+   *   An error Response object ready to return to the browser.
+   */
+  protected function createResponse($title, $body, $response_code) {
+    $fragment = new HtmlFragment($body);
+    $fragment->setTitle($title);
+
+    $page = $this->fragmentRenderer->render($fragment, $response_code);
+    return new Response($this->htmlPageRenderer->render($page), $page->getStatusCode());
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..b8dc94e85c6f44520aef1392debf5682c3499d2a
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php
@@ -0,0 +1,276 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\DefaultExceptionSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Component\Utility\SafeMarkup;
+use Drupal\Component\Utility\String;
+use Drupal\Core\Config\ConfigFactoryInterface;
+use Drupal\Core\ContentNegotiation;
+use Drupal\Core\Page\DefaultHtmlPageRenderer;
+use Drupal\Core\Page\HtmlFragment;
+use Drupal\Core\Page\HtmlFragmentRendererInterface;
+use Drupal\Core\Page\HtmlPageRendererInterface;
+use Drupal\Core\StringTranslation\StringTranslationTrait;
+use Drupal\Core\Utility\Error;
+use Symfony\Component\Debug\Exception\FlattenException;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
+use Symfony\Component\HttpKernel\KernelEvents;
+
+/**
+ * Last-chance handler for exceptions.
+ *
+ * This handler will catch any exceptions not caught elsewhere and report
+ * them as an error page.
+ */
+class DefaultExceptionSubscriber implements EventSubscriberInterface {
+  use StringTranslationTrait;
+
+  /**
+   * The fragment renderer.
+   *
+   * @var \Drupal\Core\Page\HtmlFragmentRendererInterface
+   */
+  protected $fragmentRenderer;
+
+  /**
+   * The page renderer.
+   *
+   * @var \Drupal\Core\Page\HtmlPageRendererInterface
+   */
+  protected $htmlPageRenderer;
+
+  /**
+   * @var string
+   *
+   * One of the error level constants defined in bootstrap.inc.
+   */
+  protected $errorLevel;
+
+  /**
+   * The config factory.
+   *
+   * @var \Drupal\Core\Config\ConfigFactoryInterface
+   */
+  protected $configFactory;
+
+  /**
+   * Constructs a new DefaultExceptionHtmlSubscriber.
+   *
+   * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer
+   *   The fragment renderer.
+   * @param \Drupal\Core\Page\HtmlPageRendererInterface $page_renderer
+   *   The page renderer.
+   * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
+   *   The configuration factory.
+   */
+  public function __construct(HtmlFragmentRendererInterface $fragment_renderer, HtmlPageRendererInterface $page_renderer, ConfigFactoryInterface $config_factory) {
+    $this->fragmentRenderer = $fragment_renderer;
+    $this->htmlPageRenderer = $page_renderer;
+    $this->configFactory = $config_factory;
+  }
+
+  /**
+   * Gets the configured error level.
+   *
+   * @return string
+   */
+  protected function getErrorLevel() {
+    if (!isset($this->errorLevel)) {
+      $this->errorLevel = $this->configFactory->get('system.logging')->get('error_level');
+    }
+    return $this->errorLevel;
+  }
+
+  /**
+   * Handles any exception as a generic error page for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  protected function onHtml(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+    $error = Error::decodeException($exception);
+    $flatten_exception = FlattenException::create($exception, 500);
+
+    // Display the message if the current error reporting level allows this type
+    // of message to be displayed, and unconditionally in update.php.
+    if (error_displayable($error)) {
+      $class = 'error';
+
+      // If error type is 'User notice' then treat it as debug information
+      // instead of an error message.
+      // @see debug()
+      if ($error['%type'] == 'User notice') {
+        $error['%type'] = 'Debug';
+        $class = 'status';
+      }
+
+      // Attempt to reduce verbosity by removing DRUPAL_ROOT from the file path
+      // in the message. This does not happen for (false) security.
+      $root_length = strlen(DRUPAL_ROOT);
+      if (substr($error['%file'], 0, $root_length) == DRUPAL_ROOT) {
+        $error['%file'] = substr($error['%file'], $root_length + 1);
+      }
+      // Do not translate the string to avoid errors producing more errors.
+      unset($error['backtrace']);
+      $message = String::format('%type: !message in %function (line %line of %file).', $error);
+
+      // Check if verbose error reporting is on.
+      if ($this->getErrorLevel() == ERROR_REPORTING_DISPLAY_VERBOSE) {
+        $backtrace_exception = $flatten_exception;
+        while ($backtrace_exception->getPrevious()) {
+          $backtrace_exception = $backtrace_exception->getPrevious();
+        }
+        $backtrace = $backtrace_exception->getTrace();
+        // First trace is the error itself, already contained in the message.
+        // While the second trace is the error source and also contained in the
+        // message, the message doesn't contain argument values, so we output it
+        // once more in the backtrace.
+        array_shift($backtrace);
+
+        // Generate a backtrace containing only scalar argument values.
+        $message .= '<pre class="backtrace">' . Error::formatFlattenedBacktrace($backtrace) . '</pre>';
+      }
+      drupal_set_message(SafeMarkup::set($message), $class, TRUE);
+    }
+
+    $content = $this->t('The website has encountered an error. Please try again later.');
+    $output = DefaultHtmlPageRenderer::renderPage($content, $this->t('Error'));
+    $response = new Response($output);
+
+    if ($exception instanceof HttpExceptionInterface) {
+      $response->setStatusCode($exception->getStatusCode());
+    }
+    else {
+      $response->setStatusCode(Response::HTTP_INTERNAL_SERVER_ERROR, '500 Service unavailable (with message)');
+    }
+
+    $event->setResponse($response);
+  }
+
+  /**
+   * Handles any exception as a generic error page for JSON.
+   *
+   * @todo This should probably check the error reporting level.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  protected function onJson(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+    $error = Error::decodeException($exception);
+
+    // Display the message if the current error reporting level allows this type
+    // of message to be displayed,
+    $message = error_displayable($error) ? 'A fatal error occurred: ' . $exception->getMessage() : '';
+
+    // @todo We would prefer to use JsonResponse here, but existing code and
+    // tests are not prepared for parsing a JSON response when there are quotes
+    // or other values that would cause escaping issues. Instead, return a
+    // plain string and mark it as such.
+    $response = new Response($message, Response::HTTP_INTERNAL_SERVER_ERROR, [
+      'Content-type' => 'text/plain'
+    ]);
+    if ($exception instanceof HttpExceptionInterface) {
+      $response->setStatusCode($exception->getStatusCode());
+    }
+
+    $event->setResponse($response);
+  }
+
+  /**
+   * Creates an Html response for the provided criteria.
+   *
+   * @param $title
+   *   The page title of the response.
+   * @param $body
+   *   The body of the error page.
+   * @param $response_code
+   *   The HTTP response code of the response.
+   * @return \Symfony\Component\HttpFoundation\Response
+   *   An error Response object ready to return to the browser.
+   */
+  protected function createHtmlResponse($title, $body, $response_code) {
+    $fragment = new HtmlFragment($body);
+    $fragment->setTitle($title);
+
+    $page = $this->fragmentRenderer->render($fragment, $response_code);
+    return new Response($this->htmlPageRenderer->render($page), $page->getStatusCode());
+  }
+
+  /**
+   * Handles errors for this subscriber.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function onException(GetResponseForExceptionEvent $event) {
+    $format = $this->getFormat($event->getRequest());
+
+    // If it's an unrecognized format, assume HTML.
+    $method = 'on' . $format;
+    if (!method_exists($this, $method)) {
+      $method = 'onHtml';
+    }
+    $this->$method($event);
+  }
+
+  /**
+   * Gets the error-relevant format from the request.
+   *
+   * @param \Symfony\Component\HttpFoundation\Request $request
+   *   The request object.
+   *
+   * @return string
+   *   The format as which to treat the exception.
+   */
+  protected function getFormat(Request $request) {
+    // @todo We are trying to switch to a more robust content negotiation
+    // library in https://www.drupal.org/node/1505080 that will make
+    // $request->getRequestFormat() reliable as a better alternative
+    // to this code. We therefore use this style for now on the expectation
+    // that it will get replaced with better code later. This approach makes
+    // that change easier when we get to it.
+    $conneg = new ContentNegotiation();
+    $format = $conneg->getContentType($request);
+
+    // These are all JSON errors for our purposes. Any special handling for
+    // them can/should happen in earlier listeners if desired.
+    if (in_array($format, ['drupal_modal', 'drupal_dialog', 'drupal_ajax'])) {
+      $format = 'json';
+    }
+
+    // Make an educated guess that any Accept header type that includes "json"
+    // can probably handle a generic JSON response for errors. As above, for
+    // any format this doesn't catch or that wants custom handling should
+    // register its own exception listener.
+    foreach ($request->getAcceptableContentTypes() as $mime) {
+      if (strpos($mime, 'html') === FALSE && strpos($mime, 'json') !== FALSE) {
+        $format = 'json';
+      }
+    }
+
+    return $format;
+  }
+
+  /**
+   * Registers the methods in this class that should be listeners.
+   *
+   * @return array
+   *   An array of event listener definitions.
+   */
+  public static function getSubscribedEvents() {
+    $events[KernelEvents::EXCEPTION][] = ['onException', -256];
+    return $events;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..7befb00bec27c1094bf27e031c88c621e97a27e2
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php
@@ -0,0 +1,68 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\ExceptionJsonSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Symfony\Component\HttpFoundation\JsonResponse;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+
+/**
+ * Default handling for JSON errors.
+ */
+class ExceptionJsonSubscriber extends HttpExceptionSubscriberBase {
+
+  /**
+   * {@inheritDoc}
+   */
+  protected function getHandledFormats() {
+    return ['json'];
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static function getPriority() {
+    // This will fire after the most common HTML handler, since HTML requests
+    // are still more common than JSON requests.
+    return -75;
+  }
+
+  /**
+   * Handles a 403 error for JSON.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on403(GetResponseForExceptionEvent $event) {
+    $response = new JsonResponse(NULL, Response::HTTP_FORBIDDEN);
+    $event->setResponse($response);
+  }
+
+  /**
+   * Handles a 404 error for JSON.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on404(GetResponseForExceptionEvent $event) {
+    $response = new JsonResponse(NULL, Response::HTTP_NOT_FOUND);
+    $event->setResponse($response);
+  }
+
+  /**
+   * Handles a 405 error for JSON.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on405(GetResponseForExceptionEvent $event) {
+    $response = new JsonResponse(NULL, Response::HTTP_METHOD_NOT_ALLOWED);
+    $event->setResponse($response);
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php
deleted file mode 100644
index 09447036a90502125365b4da2064388c073ea99d..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php
+++ /dev/null
@@ -1,36 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\EventSubscriber\ExceptionListener.
- */
-
-namespace Drupal\Core\EventSubscriber;
-
-use Symfony\Component\Debug\Exception\FlattenException;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\HttpKernel\EventListener\ExceptionListener as ExceptionListenerBase;
-use Symfony\Component\HttpKernel\Log\DebugLoggerInterface;
-
-/**
- * Extends the symfony exception listener to support POST subrequests.
- */
-class ExceptionListener extends ExceptionListenerBase {
-
-  /**
-   * {@inheritdoc}
-   *
-   * In contrast to the symfony base class, do not override POST requests to GET
-   * requests.
-   */
-  protected function duplicateRequest(\Exception $exception, Request $request) {
-    $attributes = array(
-      '_controller' => $this->controller,
-      'exception' => FlattenException::create($exception),
-      'logger' => $this->logger instanceof DebugLoggerInterface ? $this->logger : NULL,
-      'format' => $request->getRequestFormat(),
-    );
-    return $request->duplicate(NULL, NULL, $attributes);
-  }
-
-}
diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..3d421763aac032aa7ab585b206804f6bb9f059be
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php
@@ -0,0 +1,113 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\ExceptionLoggingSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Component\Utility\String;
+use Drupal\Core\Logger\LoggerChannelFactoryInterface;
+use Drupal\Core\Utility\Error;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
+use Symfony\Component\HttpKernel\KernelEvents;
+
+/**
+ * Log exceptions without further handling.
+ */
+class ExceptionLoggingSubscriber implements EventSubscriberInterface {
+
+  /**
+   * The logger channel factory.
+   *
+   * @var \Drupal\Core\Logger\LoggerChannelFactoryInterface
+   */
+  protected $logger;
+
+  /**
+   * Constructs a new ExceptionLoggingSubscriber.
+   *
+   * @param \Drupal\Core\Logger\LoggerChannelFactoryInterface $logger
+   *   The logger channel factory.
+   */
+  public function __construct(LoggerChannelFactoryInterface $logger) {
+    $this->logger = $logger;
+  }
+
+  /**
+   * Log 403 errors.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on403(GetResponseForExceptionEvent $event) {
+    $request = $event->getRequest();
+    // @todo Remove dependency on the internal _system_path attribute:
+    //   https://www.drupal.org/node/2293523.
+    $this->logger->get('access denied')->warning(String::checkPlain($request->attributes->get('_system_path')));
+  }
+
+  /**
+   * Log 404 errors.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on404(GetResponseForExceptionEvent $event) {
+    $request = $event->getRequest();
+    // @todo Remove dependency on the internal _system_path attribute:
+    //   https://www.drupal.org/node/2293523.
+    $this->logger->get('page not found')->warning(String::checkPlain($request->attributes->get('_system_path')));
+  }
+
+  /**
+   * Log not-otherwise-specified errors, including HTTP 500.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function onError(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+    $error = Error::decodeException($exception);
+    $this->logger->get('php')->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error);
+
+    $is_critical = !$exception instanceof HttpExceptionInterface || $exception->getStatusCode() >= 500;
+    if ($is_critical) {
+      error_log(sprintf('Uncaught PHP Exception %s: "%s" at %s line %s', get_class($exception), $exception->getMessage(), $exception->getFile(), $exception->getLine()));
+    }
+  }
+
+  /**
+   * Log all exceptions.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function onException(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+
+    $method = 'onError';
+
+    // Treat any non-HTTP exception as if it were one, so we log it the same.
+    if ($exception instanceof HttpExceptionInterface) {
+      $possible_method = 'on' . $exception->getStatusCode();
+      if (method_exists($this, $possible_method)) {
+        $method = $possible_method;
+      }
+    }
+
+    $this->$method($event);
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getSubscribedEvents() {
+    $events[KernelEvents::EXCEPTION][] = ['onException', 50];
+    return $events;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..d8f1a61339a534b66783950662b822b8c3240b90
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php
@@ -0,0 +1,67 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\ExceptionTestSiteSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\Utility\Error;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+
+/**
+ * Custom handling of errors when in a system-under-test.
+ */
+class ExceptionTestSiteSubscriber extends HttpExceptionSubscriberBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static function getPriority() {
+    return 3;
+  }
+
+  /**
+   * {@inheritDoc}
+   */
+  protected function getHandledFormats() {
+    return ['html'];
+  }
+
+  /**
+   * Checks for special handling of errors inside Simpletest.
+   *
+   * @todo The $headers array appears to not actually get used at all in the
+   *   original code. It's quite possible that this entire method is now
+   *   vestigial and can be removed.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   */
+  public function on500(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+    $error = Error::decodeException($exception);
+
+    $headers = array();
+
+    // When running inside the testing framework, we relay the errors
+    // to the tested site by the way of HTTP headers.
+    if (DRUPAL_TEST_IN_CHILD_SITE && !headers_sent() && (!defined('SIMPLETEST_COLLECT_ERRORS') || SIMPLETEST_COLLECT_ERRORS)) {
+      // $number does not use drupal_static as it should not be reset
+      // as it uniquely identifies each PHP error.
+      static $number = 0;
+      $assertion = array(
+        $error['!message'],
+        $error['%type'],
+        array(
+          'function' => $error['%function'],
+          'file' => $error['%file'],
+          'line' => $error['%line'],
+        ),
+      );
+      $headers['X-Drupal-Assertion-' . $number] = rawurlencode(serialize($assertion));
+      $number++;
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php
new file mode 100644
index 0000000000000000000000000000000000000000..c5d18dbeb2f465939693863950574e7d016eeaac
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php
@@ -0,0 +1,89 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\Fast404ExceptionHtmlSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\Config\ConfigFactoryInterface;
+use Drupal\Component\Utility\String;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\HttpKernelInterface;
+
+/**
+ * High-performance 404 exception subscriber.
+ *
+ * This subscriber will return a minimalist 404 response for HTML requests
+ * without running a full page theming operation.
+ */
+class Fast404ExceptionHtmlSubscriber extends HttpExceptionSubscriberBase {
+
+  /**
+   * The HTTP kernel.
+   *
+   * @var \Symfony\Component\HttpKernel\HttpKernelInterface
+   */
+  protected $httpKernel;
+
+  /**
+   * The config factory.
+   *
+   * @var \Drupal\Core\Config\ConfigFactoryInterface
+   */
+  protected $configFactory;
+
+  /**
+   * Constructs a new CustomPageExceptionHtmlSubscriber.
+   *
+   * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
+   *   The configuration factory.
+   * @param \Symfony\Component\HttpKernel\HttpKernelInterface $http_kernel
+   *   The HTTP Kernel service.
+   */
+  public function __construct(ConfigFactoryInterface $config_factory, HttpKernelInterface $http_kernel) {
+    $this->configFactory = $config_factory;
+    $this->httpKernel = $http_kernel;
+  }
+
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static function getPriority() {
+    // A very high priority so that it can take precedent over anything else,
+    // and thus be fast.
+    return 200;
+  }
+
+  /**
+   * {@inheritDoc}
+   */
+  protected function getHandledFormats() {
+    return ['html'];
+  }
+
+  /**
+   * Handles a 404 error for HTML.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function on404(GetResponseForExceptionEvent $event) {
+    $request = $event->getRequest();
+
+    $config = $this->configFactory->get('system.performance');
+    $exclude_paths = $config->get('fast_404.exclude_paths');
+    if ($config->get('fast_404.enabled') && $exclude_paths && !preg_match($exclude_paths, $request->getPathInfo())) {
+      $fast_paths = $config->get('fast_404.paths');
+      if ($fast_paths && preg_match($fast_paths, $request->getPathInfo())) {
+        $fast_404_html = strtr($config->get('fast_404.html'), ['@path' => String::checkPlain($request->getUri())]);
+        $response = new Response($fast_404_html, Response::HTTP_NOT_FOUND);
+        $event->setResponse($response);
+      }
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php b/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php
new file mode 100644
index 0000000000000000000000000000000000000000..832505834746aca068fa48178ba80ad5611c63ba
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php
@@ -0,0 +1,129 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\HttpExceptionSubscriberBase.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\ContentNegotiation;
+use Symfony\Component\Debug\Exception\FlattenException;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
+use Symfony\Component\HttpKernel\KernelEvents;
+
+/**
+ * Utility base class for exception subscribers.
+ *
+ * A subscriber may extend this class and implement getHandledFormats() to
+ * indicate which request formats it will respond to. Then implement an on*()
+ * method for any error code (HTTP response code) that should be handled. For
+ * example, to handle 404 Not Found messages add a method:
+ *
+ * @code
+ * public function on404(GetResponseForExceptionEvent $event) {}
+ * @endcode
+ *
+ * That method should then call $event->setResponse() to set the response object
+ * for the exception. Alternatively, it may opt not to do so and then other
+ * listeners will have the opportunity to handle the exception.
+ *
+ * Note: Core provides several important exception listeners by default. In most
+ * cases, setting the priority of a contrib listener to the default of 0 will
+ * do what you expect and handle the exceptions you'd expect it to handle.
+ * If a custom priority is set, be aware of the following core-registered
+ * listeners.
+ *
+ * - Fast404ExceptionHtmlSubscriber: 200. This subscriber will return a
+ *     minimalist, high-performance 404 page for HTML requests. It is not
+ *     recommended to have a priority higher than this one as it will only slow
+ *     down that use case.
+ * - ExceptionLoggingSubscriber: 50.  This subscriber logs all exceptions but
+ *     does not handle them. Do not register a listener with a higher priority
+ *     unless you want exceptions to not get logged, which makes debugging more
+ *     difficult.
+ * - DefaultExceptionSubscriber: -256. The subscriber of last resort, this will
+ *     provide generic handling for any exception. A listener with a lower
+ *     priority will never get called.
+ *
+ * All other core-provided exception handlers have negative priorities so most
+ * module-provided listeners will naturally take precedence over them.
+ */
+abstract class HttpExceptionSubscriberBase implements EventSubscriberInterface {
+
+  /**
+   * Specifies the request formats this subscriber will respond to.
+   *
+   * @return array
+   *   An indexed array of the format machine names that this subscriber will
+   *   attempt ot process,such as "html" or "json". Returning an empty array
+   *   will apply to all formats.
+   *
+   * @see \Symfony\Component\HttpFoundation\Request
+   */
+  abstract protected function getHandledFormats();
+
+  /**
+   * Specifies the priority of all listeners in this class.
+   *
+   * The default priority is 1, which is very low. To have listeners that have
+   * a "first attempt" at handling exceptions return a higher priority.
+   *
+   * @return int
+   *   The event priority of this subscriber.
+   */
+  protected static function getPriority() {
+    return 0;
+  }
+
+  /**
+   * Handles errors for this subscriber.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event
+   *   The event to process.
+   */
+  public function onException(GetResponseForExceptionEvent $event) {
+    $exception = $event->getException();
+
+    // Make the exception available for example when rendering a block.
+    $event->getRequest()->attributes->set('exception', FlattenException::create($exception));
+
+    $handled_formats = $this->getHandledFormats();
+
+    // @todo Injecting this service would force all implementing classes to also
+    // handle its injection. However, we are trying to switch to a more robust
+    // content negotiation library in https://www.drupal.org/node/1505080 that
+    // will make $request->getRequestFormat() reliable as a better alternative
+    // to this code. We therefore use this style for now on the expectation
+    // that it will get replaced with better code later. That change will NOT
+    // be an API change for any implementing classes.  (Whereas if we injected
+    // this class it would be an API change.  That's why we're not doing it.)
+    $conneg = new ContentNegotiation();
+    $format = $conneg->getContentType($event->getRequest());
+
+    if ($exception instanceof HttpExceptionInterface && (empty($handled_formats) || in_array($format, $handled_formats))) {
+      $method = 'on' . $exception->getStatusCode();
+      // We want to allow the method to be called and still not set a response
+      // if it has additional filtering logic to determine when it will apply.
+      // It is therefore the method's responsibility to set the response on the
+      // event if appropriate.
+      if (method_exists($this, $method)) {
+        $this->$method($event);
+      }
+    }
+  }
+
+  /**
+   * Registers the methods in this class that should be listeners.
+   *
+   * @return array
+   *   An array of event listener definitions.
+   */
+  public static function getSubscribedEvents() {
+    $events[KernelEvents::EXCEPTION][] = ['onException', static::getPriority()];
+    return $events;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php
index 8556f9b7923401a704e104761bdb4f2333f9e94d..f09e3e318d2daee03aa030a713e0cf14087c08e0 100644
--- a/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php
+++ b/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php
@@ -130,6 +130,7 @@ protected function drupalSetMessage($message = NULL, $type = 'status', $repeat =
    */
   public static function getSubscribedEvents() {
     $events[KernelEvents::REQUEST][] = array('onKernelRequestMaintenance', 30);
+    $events[KernelEvents::EXCEPTION][] = array('onKernelRequestMaintenance');
     return $events;
   }
 
diff --git a/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php b/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php
index 6785b95c0772e34a96b59fdb779199ffd4c83f06..a8c115935820c0f772217f4ea80e3034368c001c 100644
--- a/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php
+++ b/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php
@@ -86,7 +86,7 @@ public function onException(GetResponseForExceptionEvent $event) {
    * {@inheritdoc}
    */
   public static function getSubscribedEvents() {
-    $events[KernelEvents::EXCEPTION][] = array('onException', 0);
+    $events[KernelEvents::EXCEPTION][] = array('onException', 75);
     return $events;
   }
 
diff --git a/core/lib/Drupal/Core/Utility/Error.php b/core/lib/Drupal/Core/Utility/Error.php
index 2bfee623e8f4fe9be5e29910be83e951b17ad096..e3bca6dbbd45561c53a30601763b6d428871bfbc 100644
--- a/core/lib/Drupal/Core/Utility/Error.php
+++ b/core/lib/Drupal/Core/Utility/Error.php
@@ -10,6 +10,7 @@
 use Drupal\Component\Utility\String;
 use Drupal\Component\Utility\Xss;
 use Drupal\Component\Utility\SafeMarkup;
+use Drupal\Core\Database\DatabaseExceptionWrapper;
 
 /**
  * Drupal error utility class.
@@ -48,7 +49,7 @@ public static function decodeException(\Exception $exception) {
 
     // For PDOException errors, we try to return the initial caller,
     // skipping internal functions of the database layer.
-    if ($exception instanceof \PDOException) {
+    if ($exception instanceof \PDOException || $exception instanceof DatabaseExceptionWrapper) {
       // The first element in the stack is the call, the second element gives us
       // the caller. We skip calls that occurred in one of the classes of the
       // database layer or in one of its global functions.
diff --git a/core/modules/shortcut/shortcut.module b/core/modules/shortcut/shortcut.module
index 9dc7d5a667d7edc2f95d956ad2ba57d8456ccd9e..13c827bc754376e887ea489fe386a7995dc229f7 100644
--- a/core/modules/shortcut/shortcut.module
+++ b/core/modules/shortcut/shortcut.module
@@ -320,15 +320,9 @@ function shortcut_preprocess_page(&$variables) {
   // shortcuts and if the page's actual content is being shown (for example,
   // we do not want to display it on "access denied" or "page not found"
   // pages).
-  $item = array();
-  if (\Drupal::routeMatch()->getRouteObject()) {
-    // @todo What should be done on a 404/403 page?
-    $item['access'] = TRUE;
-  }
-
-  if (shortcut_set_edit_access() && !empty($item['access'])) {
+  if (shortcut_set_edit_access() && !\Drupal::request()->attributes->has('exception')) {
     $link = current_path();
-    if (!($url = Url::createFromPath($link))) {
+    if (!($url = \Drupal::pathValidator()->getUrlIfValid($link))) {
       // Bail out early if we couldn't find a matching route.
       return;
     }
diff --git a/core/modules/simpletest/src/WebTestBase.php b/core/modules/simpletest/src/WebTestBase.php
index dc3771361cd8ca3b7c8750b3cc0d850752ffaa72..5a6bba80ff1cb5cba06d6cf56a652be469f3af44 100644
--- a/core/modules/simpletest/src/WebTestBase.php
+++ b/core/modules/simpletest/src/WebTestBase.php
@@ -1827,6 +1827,10 @@ protected function drupalProcessAjaxResponse($content, array $ajax_response, arr
     // XPath allows for finding wrapper nodes better than DOM does.
     $xpath = new \DOMXPath($dom);
     foreach ($ajax_response as $command) {
+      // Error messages might be not commands.
+      if (!is_array($command)) {
+        continue;
+      }
       switch ($command['command']) {
         case 'settings':
           $drupal_settings = drupal_merge_js_settings(array($drupal_settings, $command['settings']));
diff --git a/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php b/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..b6ed513ac9087f28e6a8effd1d8bf999eaf0d62c
--- /dev/null
+++ b/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php
@@ -0,0 +1,120 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\system\Tests\Routing\ExceptionHandlingTest.
+ */
+
+namespace Drupal\system\Tests\Routing;
+
+use Drupal\simpletest\KernelTestBase;
+use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+
+/**
+ * Tests the exception handling for various cases.
+ *
+ * @group Routing
+ */
+class ExceptionHandlingTest extends KernelTestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static $modules = ['system', 'router_test'];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    parent::setUp();
+
+    $this->installSchema('system', ['router']);
+    \Drupal::service('router.builder')->rebuild();
+  }
+
+  /**
+   * Tests the exception handling for json and 403 status code.
+   */
+  public function testJson403() {
+    $request = Request::create('/router_test/test15');
+    $request->headers->set('Accept', 'application/json');
+    $request->setFormat('json', ['application/json']);
+
+    /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */
+    $kernel = \Drupal::getContainer()->get('http_kernel');
+    $response = $kernel->handle($request);
+
+    $this->assertEqual($response->getStatusCode(), Response::HTTP_FORBIDDEN);
+    $this->assertEqual($response->headers->get('Content-type'), 'application/json');
+    $this->assertEqual('{}', $response->getContent());
+  }
+
+  /**
+   * Tests the exception handling for json and 404 status code.
+   */
+  public function testJson404() {
+    $request = Request::create('/not-found');
+    $request->headers->set('Accept', 'application/json');
+    $request->setFormat('json', ['application/json']);
+
+    /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */
+    $kernel = \Drupal::getContainer()->get('http_kernel');
+    $response = $kernel->handle($request);
+
+    $this->assertEqual($response->getStatusCode(), Response::HTTP_NOT_FOUND);
+    $this->assertEqual($response->headers->get('Content-type'), 'application/json');
+    $this->assertEqual('{}', $response->getContent());
+  }
+
+  /**
+   * Tests the exception handling for HTML and 403 status code.
+   */
+  public function testHtml403() {
+    $request = Request::create('/router_test/test15');
+    $request->headers->set('Accept', 'text/html');
+    $request->setFormat('html', ['text/html']);
+
+    /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */
+    $kernel = \Drupal::getContainer()->get('http_kernel');
+    $response = $kernel->handle($request);
+
+    $this->assertEqual($response->getStatusCode(), Response::HTTP_FORBIDDEN);
+    $this->assertEqual($response->headers->get('Content-type'), 'text/html');
+  }
+
+  /**
+   * Tests the exception handling for HTML and 403 status code.
+   */
+  public function testHtml404() {
+    $request = Request::create('/not-found');
+    $request->headers->set('Accept', 'text/html');
+    $request->setFormat('html', ['text/html']);
+
+    /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */
+    $kernel = \Drupal::getContainer()->get('http_kernel');
+    $response = $kernel->handle($request);
+
+    $this->assertEqual($response->getStatusCode(), Response::HTTP_NOT_FOUND);
+    $this->assertEqual($response->headers->get('Content-type'), 'text/html');
+  }
+
+  /**
+   * Tests the exception handling for HTML and 405 status code.
+   */
+  public function testHtml405() {
+    $request = Request::create('/admin', 'NOT_EXISTING');
+    $request->headers->set('Accept', 'text/html');
+    $request->setFormat('html', ['text/html']);
+
+    /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */
+    $kernel = \Drupal::getContainer()->get('http_kernel');
+    $response = $kernel->handle($request);
+
+    $this->assertEqual($response->getStatusCode(), Response::HTTP_METHOD_NOT_ALLOWED);
+    $this->assertEqual($response->headers->get('Content-type'), 'text/html');
+    $this->assertEqual($response->getContent(), 'Method Not Allowed');
+  }
+
+}
+
diff --git a/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php b/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php
deleted file mode 100644
index 3a796ec15609c74580e693e7758342a1ac096327..0000000000000000000000000000000000000000
--- a/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php
+++ /dev/null
@@ -1,58 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Tests\Core\Controller\ExceptionControllerTest
- */
-
-namespace Drupal\Tests\Core\Controller {
-
-use Drupal\Core\Controller\ExceptionController;
-use Drupal\Tests\UnitTestCase;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\Debug\Exception\FlattenException;
-
-/**
- * @coversDefaultClass \Drupal\Core\Controller\ExceptionController
- * @group Controller
- */
-class ExceptionControllerTest extends UnitTestCase {
-
-  /**
-   * Ensure the execute() method returns a valid response on 405 exceptions.
-   */
-  public function test405HTML() {
-    $exception = new \Exception('Test exception');
-    $flat_exception = FlattenException::create($exception, 405);
-    $html_page_renderer = $this->getMock('Drupal\Core\Page\HtmlPageRendererInterface');
-    $html_fragment_renderer = $this->getMock('Drupal\Core\Page\HtmlFragmentRendererInterface');
-    $title_resolver = $this->getMock('Drupal\Core\Controller\TitleResolverInterface');
-    $translation = $this->getMock('Drupal\Core\StringTranslation\TranslationInterface');
-    $url_generator = $this->getMock('Drupal\Core\Routing\UrlGeneratorInterface');
-    $logger_factory = $this->getMock('Drupal\Core\Logger\LoggerChannelFactoryInterface');
-
-    $content_negotiation = $this->getMock('Drupal\Core\ContentNegotiation');
-    $content_negotiation->expects($this->any())
-      ->method('getContentType')
-      ->will($this->returnValue('html'));
-
-    $exception_controller = new ExceptionController($content_negotiation, $title_resolver, $html_page_renderer, $html_fragment_renderer, $translation, $url_generator, $logger_factory);
-    $response = $exception_controller->execute($flat_exception, new Request());
-    $this->assertEquals($response->getStatusCode(), 405, 'HTTP status of response is correct.');
-    $this->assertEquals($response->getContent(), 'Method Not Allowed', 'HTTP response body is correct.');
-  }
-
-}
-
-}
-
-namespace {
-  use Drupal\Core\Language\Language;
-
-  if (!function_exists('language_default')) {
-    function language_default() {
-      $language = new Language(array('langcode' => 'en'));
-      return $language;
-    }
-  }
-}
diff --git a/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php b/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..1de4249cf7037389b54c6e189959537f7ec538d4
--- /dev/null
+++ b/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php
@@ -0,0 +1,141 @@
+<?php
+
+/**
+ * @file
+ * Contains
+ *   \Drupal\Tests\Core\EventSubscriber\CustomPageExceptionHtmlSubscriberTest.
+ */
+
+namespace Drupal\Tests\Core\EventSubscriber;
+
+use Drupal\Component\Utility\UrlHelper;
+use Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber;
+use Drupal\Tests\UnitTestCase;
+use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
+
+/**
+ * @coversDefaultClass \Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber
+ * @group EventSubscriber
+ */
+class CustomPageExceptionHtmlSubscriberTest extends UnitTestCase {
+
+  /**
+   * The mocked HTTP kernel.
+   *
+   * @var \Symfony\Component\HttpKernel\HttpKernelInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $kernel;
+
+  /**
+   * The mocked config factory
+   *
+   * @var  \Drupal\Core\Config\ConfigFactoryInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $configFactory;
+
+  /**
+   * The mocked alias manager.
+   *
+   * @var \Drupal\Core\Path\AliasManagerInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $aliasManager;
+
+  /**
+   * The mocked logger.
+   *
+   * @var \Psr\Log\LoggerInterface
+   */
+  protected $logger;
+
+  /**
+   * The PHP error log settings before the test.
+   *
+   * @var string
+   */
+  protected $errorLog;
+
+  /**
+   * The tested custom page exception subscriber.
+   *
+   * @var \Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber
+   */
+  protected $customPageSubscriber;
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    $this->configFactory = $this->getConfigFactoryStub(['system.site' => ['page.403' => 'access-denied-page', 'page.404' => 'not-found-page']]);
+
+    $this->aliasManager = $this->getMock('Drupal\Core\Path\AliasManagerInterface');
+    $this->kernel = $this->getMock('Symfony\Component\HttpKernel\HttpKernelInterface');
+    $this->logger = $this->getMock('Psr\Log\LoggerInterface');
+    $this->customPageSubscriber = new CustomPageExceptionHtmlSubscriber($this->configFactory, $this->aliasManager, $this->kernel, $this->logger);
+
+    // You can't create an exception in PHP without throwing it. Store the
+    // current error_log, and disable it temporarily.
+    $this->errorLog = ini_set('error_log', file_exists('/dev/null') ? '/dev/null' : 'nul');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function tearDown() {
+    ini_set('error_log', $this->errorLog);
+  }
+
+  /**
+   * Sets up an alias manager that does nothing.
+   */
+  protected function setupStubAliasManager() {
+    $this->aliasManager->expects($this->any())
+      ->method('getPathByAlias')
+      ->willReturnArgument(0);
+  }
+
+  /**
+   * Tests onHandleException with a POST request.
+   */
+  public function testHandleWithPostRequest() {
+    $this->setupStubAliasManager();
+
+    $request = Request::create('/test', 'POST', array('name' => 'druplicon', 'pass' => '12345'));
+
+    $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) {
+      return new Response($request->getMethod());
+    }));
+
+    $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new NotFoundHttpException('foo'));
+
+    $this->customPageSubscriber->onException($event);
+
+    $response = $event->getResponse();
+    $result = $response->getContent() . " " . UrlHelper::buildQuery($request->request->all());
+    $this->assertEquals('POST name=druplicon&pass=12345', $result);
+  }
+
+  /**
+   * Tests onHandleException with a GET request.
+   */
+  public function testHandleWithGetRequest() {
+    $this->setupStubAliasManager();
+
+    $request = Request::create('/test', 'GET', array('name' => 'druplicon', 'pass' => '12345'));
+    $request->attributes->set('_system_path', 'test');
+
+    $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) {
+      return new Response($request->getMethod() . ' ' . UrlHelper::buildQuery($request->query->all()));
+    }));
+
+    $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new NotFoundHttpException('foo'));
+    $this->customPageSubscriber->onException($event);
+
+    $response = $event->getResponse();
+    $result = $response->getContent() . " " . UrlHelper::buildQuery($request->request->all());
+    $this->assertEquals('GET name=druplicon&pass=12345&destination=test&_exception_statuscode=404 ', $result);
+  }
+
+}
diff --git a/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php b/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php
deleted file mode 100644
index 5815ee167f9239a4cbeb3f24d10e4e8bcfc753fd..0000000000000000000000000000000000000000
--- a/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php
+++ /dev/null
@@ -1,98 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Tests\Core\EventSubscriber\ExceptionListenerTest.
- */
-
-namespace Drupal\Tests\Core\EventSubscriber;
-
-use Drupal\Component\Utility\UrlHelper;
-use Drupal\Core\EventSubscriber\ExceptionListener;
-use Drupal\Tests\UnitTestCase;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\HttpFoundation\Response;
-use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
-
-/**
- * @coversDefaultClass \Drupal\Core\EventSubscriber\ExceptionListener
- * @group EventSubscriber
- */
-class ExceptionListenerTest extends UnitTestCase {
-
-  /**
-   * The tested exception listener.
-   *
-   * @var \Drupal\Core\EventSubscriber\ExceptionListener
-   */
-  protected $exceptionListener;
-
-  /**
-   * The mocked HTTP kernel.
-   *
-   * @var \Symfony\Component\HttpKernel\HttpKernelInterface|\PHPUnit_Framework_MockObject_MockObject
-   */
-  protected $kernel;
-
-  /**
-   * The PHP error log settings before the test.
-   *
-   * @var string
-   */
-  protected $errorLog;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp() {
-    $this->exceptionListener = new ExceptionListener('example');
-    $this->kernel = $this->getMock('Symfony\Component\HttpKernel\HttpKernelInterface');
-
-    // You can't create an exception in PHP without throwing it. Store the
-    // current error_log, and disable it temporarily.
-    $this->errorLog = ini_set('error_log', file_exists('/dev/null') ? '/dev/null' : 'nul');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function tearDown() {
-    ini_set('error_log', $this->errorLog);
-  }
-
-  /**
-   * Tests onHandleException with a POST request.
-   */
-  public function testHandleWithPostRequest() {
-    $request = Request::create('/test', 'POST', array('name' => 'druplicon', 'pass' => '12345'));
-
-    $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) {
-      return new Response($request->getMethod());
-    }));
-
-    $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new \Exception('foo'));
-
-    $this->exceptionListener->onKernelException($event);
-
-    $response = $event->getResponse();
-    $this->assertEquals('POST name=druplicon&pass=12345', $response->getContent() . " " . UrlHelper::buildQuery($request->request->all()));
-  }
-
-  /**
-   * Tests onHandleException with a GET request.
-   */
-  public function testHandleWithGetRequest() {
-    $request = Request::create('/test', 'GET', array('name' => 'druplicon', 'pass' => '12345'));
-
-    $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) {
-      return new Response($request->getMethod() . ' ' . UrlHelper::buildQuery($request->query->all()));
-    }));
-
-    $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new \Exception('foo'));
-    $this->exceptionListener->onKernelException($event);
-
-    $response = $event->getResponse();
-    $this->assertEquals('GET name=druplicon&pass=12345 ', $response->getContent() . " " . UrlHelper::buildQuery($request->request->all()));
-  }
-
-}