diff --git a/core/modules/image/image.install b/core/modules/image/image.install
index 75ffd2a8f8264974f5c60b5cd9e1aecd5e7571d0..88f05a121c91f0ae5e14738d67c3be5cc7e8c501 100644
--- a/core/modules/image/image.install
+++ b/core/modules/image/image.install
@@ -5,6 +5,8 @@
  * Install, update and uninstall functions for the image module.
  */
 
+use Drupal\Component\Uuid\Uuid;
+
 /**
  * Implements hook_install().
  */
@@ -103,3 +105,65 @@ function image_requirements($phase) {
 
   return $requirements;
 }
+
+/**
+ * Loads all effects for an image style.
+ *
+ * @param array $style
+ *   The image style (array) to retrieve effects for.
+ *
+ * @return array
+ *   An array of effects keyed by UUIDs.
+ *
+ * @see image_update_8000()
+ */
+function _image_update_get_style_with_effects(array $style) {
+  // Retrieve image effects.
+  $effects = array();
+  $result = db_select('image_effects', NULL, array('fetch' => PDO::FETCH_ASSOC))
+    ->fields('image_effects')
+    ->condition('isid', $style['isid'])
+    ->execute();
+  foreach ($result as $effect) {
+    unset($effect['isid']);
+    $effect['data'] = unserialize($effect['data']);
+
+    // Generate a unique image effect ID for the effect.
+    $uuid = new Uuid();
+    $effect['ieid'] = $uuid->generate();
+
+    $effects[$effect['ieid']] = $effect;
+  }
+  return $effects;
+}
+
+/**
+ * Convert existing image styles to the new config system.
+ */
+function image_update_8000() {
+  $styles = array();
+  $result = db_select('image_styles', NULL, array('fetch' => PDO::FETCH_ASSOC))
+    ->fields('image_styles')
+    ->execute()
+    ->fetchAllAssoc('name', PDO::FETCH_ASSOC);
+  foreach ($result as $style_name => $style) {
+    $style['effects'] = _image_update_get_style_with_effects($style);
+    $styles[$style_name] = $style;
+  }
+
+  // Convert each style into a configuration object.
+  foreach ($styles as $name => $style) {
+    $config = config('image.style.' . $name);
+    $config->set('name', $name);
+    $config->set('effects', $style['effects']);
+    $config->save();
+  }
+}
+
+/**
+ * Remove the {image_styles} and {image_effects} tables.
+ */
+function image_update_8001() {
+  db_drop_table('image_styles');
+  db_drop_table('image_effects');
+}
diff --git a/core/modules/system/lib/Drupal/system/Tests/Upgrade/ImageUpgradePathTest.php b/core/modules/system/lib/Drupal/system/Tests/Upgrade/ImageUpgradePathTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..fc248133f3845e59ff7fc91d6d3c96c728ab5d03
--- /dev/null
+++ b/core/modules/system/lib/Drupal/system/Tests/Upgrade/ImageUpgradePathTest.php
@@ -0,0 +1,110 @@
+<?php
+
+/**
+ * @file
+ * Definition of Drupal\system\Tests\Upgrade\ImageUpgradePathTest.
+ */
+
+namespace Drupal\system\Tests\Upgrade;
+
+/**
+ * Test upgrade of overridden and custom image styles.
+ */
+class ImageUpgradePathTest extends UpgradePathTestBase {
+  public static function getInfo() {
+    return array(
+      'name' => 'Image upgrade test',
+      'description' => 'Upgrade tests for overridden and custom image styles.',
+      'group' => 'Upgrade path',
+    );
+  }
+
+  public function setUp() {
+    $this->databaseDumpFiles = array(
+      drupal_get_path('module', 'system') . '/tests/upgrade/drupal-7.bare.standard_all.database.php.gz',
+      drupal_get_path('module', 'system') . '/tests/upgrade/drupal-7.image.database.php',
+    );
+    parent::setUp();
+  }
+
+  /**
+   * Tests that custom and overridden image styles have been upgraded.
+   */
+  public function testImageStyleUpgrade() {
+    $this->assertTrue($this->performUpgrade(), 'The upgrade was completed successfully.');
+
+    // Verify that image styles were properly upgraded.
+    $expected_styles['test-custom'] = array(
+      'name' => 'test-custom',
+      'effects' => array(
+        'image_rotate' => array(
+          'name' => 'image_rotate',
+          'data' => array(
+            'degrees' => '90',
+            'bgcolor' => '#FFFFFF',
+            'random' => '1',
+          ),
+          'weight' => '1',
+        ),
+        'image_desaturate' => array(
+          'name' => 'image_desaturate',
+          'data' => array(),
+          'weight' => '2',
+        ),
+      ),
+    );
+    $expected_styles['thumbnail'] = array(
+      'name' => 'thumbnail',
+      'effects' => array (
+        'image_scale' => array(
+          'name' => 'image_scale',
+          'data' => array (
+            'width' => '177',
+            'height' => '177',
+            'upscale' => '0',
+          ),
+          'weight' => '0',
+        ),
+      ),
+    );
+    foreach ($expected_styles as $name => $style) {
+      $config = config('image.style.' . $name);
+      // Replace placeholder with image effect name keys with UUID's generated
+      // during by the image style upgrade functions.
+      foreach ($config->get('effects') as $uuid => $effect) {
+        // Copy placeholder data.
+        $style['effects'][$uuid] = $style['effects'][$effect['name']];
+        // Set the missing ieid key as this is unknown because it is a UUID.
+        $style['effects'][$uuid]['ieid'] = $uuid;
+        // Remove the placeholder data.
+        unset($style['effects'][$effect['name']]);
+      }
+      $this->assertEqual($this->sortByKey($style), $config->get(), format_string('@first is equal to @second.', array(
+        '@first' => var_export($this->sortByKey($style), TRUE),
+        '@second' => var_export($config->get(), TRUE),
+      )));
+    }
+  }
+  /**
+   * Sorts all keys in configuration data.
+   *
+   * Since we can not be sure of the order of the UUID's generated by
+   * _image_update_get_style_with_effects() we need to sort the data in order
+   * to compare it with data saved in the config system.
+   *
+   * @param array $data
+   *   An associative array to sort recursively by key name.
+   *
+   * @return array
+   *   A sorted array.
+   */
+  public function sortByKey(array $data) {
+    ksort($data);
+    foreach ($data as &$value) {
+      if (is_array($value)) {
+        $this->sortByKey($value);
+      }
+    }
+    return $data;
+  }
+}
diff --git a/core/modules/system/tests/upgrade/drupal-7.image.database.php b/core/modules/system/tests/upgrade/drupal-7.image.database.php
new file mode 100644
index 0000000000000000000000000000000000000000..3033858d2fd0558ece50d7bf473877433eae9e20
--- /dev/null
+++ b/core/modules/system/tests/upgrade/drupal-7.image.database.php
@@ -0,0 +1,59 @@
+<?php
+
+/**
+ * @file
+ * Database additions for Drupal\system\Tests\Upgrade\ImageUpgradePathTest.
+ *
+ * This dump only contains data and schema components relevant for image
+ * functionality. The drupal-7.filled.database.php file is imported before
+ * this dump, so the two form the database structure expected in tests
+ * altogether.
+ */
+
+// Add image styles.
+db_insert('image_styles')->fields(array(
+  'isid',
+  'name',
+))
+// Override thumbnail style.
+->values(array(
+  'isid' => '1',
+  'name' => 'thumbnail',
+))
+// Custom style.
+->values(array(
+  'isid' => '2',
+  'name' => 'test-custom',
+))
+->execute();
+
+// Add image effects.
+db_insert('image_effects')->fields(array(
+  'ieid',
+  'isid',
+  'weight',
+  'name',
+  'data',
+))
+->values(array(
+  'ieid' => '1',
+  'isid' => '1',
+  'weight' => '0',
+  'name' => 'image_scale',
+  'data' => 'a:3:{s:5:"width";s:3:"177";s:6:"height";s:3:"177";s:7:"upscale";i:0;}',
+))
+->values(array(
+  'ieid' => '3',
+  'isid' => '2',
+  'weight' => '1',
+  'name' => 'image_rotate',
+  'data' => 'a:3:{s:7:"degrees";s:2:"90";s:7:"bgcolor";s:7:"#FFFFFF";s:6:"random";i:1;}',
+))
+->values(array(
+  'ieid' => '4',
+  'isid' => '2',
+  'weight' => '2',
+  'name' => 'image_desaturate',
+  'data' => 'a:0:{}',
+))
+->execute();