Index: modules/field/field.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/field/field.module,v
retrieving revision 1.81
diff -u -p -r1.81 field.module
--- modules/field/field.module	14 Aug 2010 03:10:04 -0000	1.81
+++ modules/field/field.module	18 Aug 2010 22:55:43 -0000
@@ -219,7 +219,7 @@ function field_modules_disabled($modules
     ->condition('storage_module', $modules, 'IN')
     ->execute();
 
-  field_cache_clear(TRUE);
+  field_cache_clear();
 }
 
 /**
Index: modules/field/modules/text/text.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/field/modules/text/text.module,v
retrieving revision 1.61
diff -u -p -r1.61 text.module
--- modules/field/modules/text/text.module	17 Aug 2010 18:25:31 -0000	1.61
+++ modules/field/modules/text/text.module	18 Aug 2010 22:55:43 -0000
@@ -633,6 +633,13 @@ function text_field_widget_error($elemen
 }
 
 /**
+ * Implements hook_filter_format_update().
+ */
+function text_filter_format_update($format) {
+  field_cache_clear();
+}
+
+/**
  * Implements hook_field_prepare_translation().
  */
 function text_field_prepare_translation($entity_type, $entity, $field, $instance, $langcode, &$items, $source_entity, $source_langcode) {
Index: modules/filter/filter.admin.inc
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.admin.inc,v
retrieving revision 1.64
diff -u -p -r1.64 filter.admin.inc
--- modules/filter/filter.admin.inc	17 Aug 2010 13:50:52 -0000	1.64
+++ modules/filter/filter.admin.inc	18 Aug 2010 22:55:43 -0000
@@ -17,8 +17,16 @@ function filter_admin_overview($form) {
   $formats = filter_formats();
   $fallback_format = filter_fallback_format();
 
+  // Output a warning if there are formats containing missing filters.
+  $broken_filters = _filter_get_broken_filters();
+  if ($broken_filters) {
+    drupal_set_message(format_plural(count($broken_filters), 'The highlighted text format needs to be re-configured, because it is using filters that no longer exist.', 'The highlighted text formats need to be re-configured, because they are using filters that no longer exist.'), 'error');
+  }
+
   $form['#tree'] = TRUE;
   foreach ($formats as $id => $format) {
+    // Indicate broken formats.
+    $form['formats'][$id]['#is_broken'] = isset($broken_filters[$id]);
     // Check whether this is the fallback text format. This format is available
     // to all roles and cannot be deleted via the admin interface.
     $form['formats'][$id]['#is_fallback'] = ($id == $fallback_format);
@@ -78,7 +86,7 @@ function theme_filter_admin_overview($va
         drupal_render($form['formats'][$id]['configure']),
         drupal_render($form['formats'][$id]['delete']),
       ),
-      'class' => array('draggable'),
+      'class' => array('draggable', ($form['formats'][$id]['#is_broken'] ? 'error' : '')),
     );
   }
   $header = array(t('Name'), t('Roles'), t('Weight'), array('data' => t('Operations'), 'colspan' => 2));
Index: modules/filter/filter.install
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.install,v
retrieving revision 1.42
diff -u -p -r1.42 filter.install
--- modules/filter/filter.install	1 Jul 2010 15:14:27 -0000	1.42
+++ modules/filter/filter.install	18 Aug 2010 22:55:43 -0000
@@ -7,6 +7,27 @@
  */
 
 /**
+ * Implements hook_requirements().
+ */
+function filter_requirements($phase) {
+  $requirements = array();
+
+  if ($phase == 'runtime') {
+    // Check for text formats containing missing/vanished filters.
+    if (_filter_get_broken_filters()) {
+      $requirements['filter'] = array(
+        'title' => t('Text formats'),
+        'value' => t('Missing filters'),
+        'severity' => REQUIREMENT_ERROR,
+        'description' => t('At least one text format contains a filter that was provided by a module that has been disabled. All content using one of these text formats will be hidden until <a href="@text-format-url">text formats have been re-configured</a>.', array('@text-format-url' => url('admin/config/content/formats'))),
+      );
+    }
+  }
+
+  return $requirements;
+}
+
+/**
  * Implements hook_schema().
  */
 function filter_schema() {
Index: modules/filter/filter.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.module,v
retrieving revision 1.338
diff -u -p -r1.338 filter.module
--- modules/filter/filter.module	17 Aug 2010 13:50:52 -0000	1.338
+++ modules/filter/filter.module	18 Aug 2010 22:58:15 -0000
@@ -187,6 +187,10 @@ function filter_format_load($format_id) 
 function filter_format_save(&$format) {
   $format->name = trim($format->name);
   $format->cache = _filter_format_is_cacheable($format);
+  // Programmatic saves may not contain any filters.
+  if (!isset($format->filters)) {
+    $format->filters = array();
+  }
 
   // Add a new text format.
   if (empty($format->format)) {
@@ -194,13 +198,20 @@ function filter_format_save(&$format) {
   }
   else {
     $return = drupal_write_record('filter_format', $format, 'format');
+
+    // When a text format is updated, we need to assume that its filter
+    // configuration has been verified and we are safe to remove any pointers
+    // to vanished filters.
+    if ($broken_filters = _filter_get_broken_filters($format->format, NULL, FALSE)) {
+      db_delete('filter')
+        ->condition('format', $format->format)
+        ->condition('name', $broken_filters, 'IN')
+        ->condition('name', array_keys($format->filters), 'NOT IN')
+        ->execute();
+    }
   }
 
   $filter_info = filter_get_filters();
-  // Programmatic saves may not contain any filters.
-  if (!isset($format->filters)) {
-    $format->filters = array();
-  }
   foreach ($filter_info as $name => $filter) {
     // Add new filters without weight to the bottom.
     if (!isset($format->filters[$name]['weight'])) {
@@ -280,6 +291,7 @@ function filter_format_delete($format, $
   $fallback = filter_format_load($fallback_id);
   module_invoke_all('filter_format_delete', $format, $fallback);
 
+  // Clear the filter cache whenever a text format is deleted.
   filter_formats_reset();
   cache_clear_all($format->format . ':', 'cache_filter', TRUE);
 }
@@ -353,11 +365,38 @@ function filter_modules_enabled($modules
 
 /**
  * Implements hook_modules_disabled().
+ *
+ * If a module is disabled that provides a filter, all text formats using the
+ * filter have to be manually updated. This cannot be automated, because that
+ * would always break the intended filter processing logic.
+ *
+ * @see filter_formats()
+ * @see check_markup()
  */
 function filter_modules_disabled($modules) {
   // Reset the static cache of module-provided filters, in case any of the
   // newly disabled modules defined or altered any filters.
   drupal_static_reset('filter_get_filters');
+
+  if (_filter_get_broken_filters(NULL, $modules)) {
+    drupal_set_message(t('At least one text format contains a filter that was provided by a module that has been disabled. All content using one of these text formats will be hidden until <a href="@text-format-url">text formats have been re-configured</a>.', array('@text-format-url' => url('admin/config/content/formats'))), 'error');
+  }
+}
+
+/**
+ * Implements hook_modules_uninstalled().
+ *
+ * If a module is uninstalled that provides a filter, all text formats using the
+ * filter have to be manually updated. This cannot be automated, because that
+ * would always break the intended filter processing logic.
+ *
+ * @see filter_formats()
+ * @see check_markup()
+ */
+function filter_modules_uninstalled($modules) {
+  if (_filter_get_broken_filters(NULL, $modules)) {
+    drupal_set_message(t('At least one text format contains a filter that was provided by a module that has been uninstalled. All content using one of these text formats will be hidden until <a href="@text-format-url">text formats have been re-configured</a>.', array('@text-format-url' => url('admin/config/content/formats'))), 'error');
+  }
 }
 
 /**
@@ -642,9 +681,14 @@ function filter_list_format($format_id) 
         if (isset($filter_info[$name]['default settings'])) {
           $filter->settings += $filter_info[$name]['default settings'];
         }
-
-        $format_filters[$name] = $filter;
       }
+      else {
+        // The filter is assigned to the text format, but the module providing
+        // the filter no longer exists. We assign a negative status, so
+        // check_markup() can bail out.
+        $filter->status = -1;
+      }
+      $format_filters[$name] = $filter;
     }
     $filters[$format_id] = $format_filters;
   }
@@ -653,6 +697,58 @@ function filter_list_format($format_id) 
 }
 
 /**
+ * Helper function to return a list of vanished filters keyed by text format.
+ *
+ * @param $format_id
+ *   (optional) A text format id to limit the query to.
+ * @param $modules
+ *   (optional) A list of module names to limit the query to.
+ * @param $enabled_only
+ *   (optional) Boolean whether to search only for enabled filters. Defaults to
+ *   TRUE.
+ *
+ * @return
+ *   A list of assigned filters that no longer exist, keyed by text format. If
+ *   $format_id was passed, only filters for the given text format are
+ *   returned.
+ *
+ * @see filter_list_format()
+ */
+function _filter_get_broken_filters($format_id = NULL, $modules = NULL, $enabled_only = TRUE) {
+  // Retrieve all filters exposed by modules.
+  $filter_info = filter_get_filters();
+
+  // Query all filters not contained in the list of available filters.
+  $query = db_select('filter', 'f')
+    ->fields('f', array('format', 'name'))
+    ->condition('name', array_keys($filter_info), 'NOT IN');
+  // Optionally limit to enabled filters.
+  if ($enabled_only) {
+    $query->condition('status', 1);
+  }
+  // Optionally limit to text format.
+  if (isset($format_id)) {
+    $query->condition('format', $format_id);
+  }
+  // Optionally limit to a list of modules.
+  if (isset($modules)) {
+    $query->condition('module', $modules, 'IN');
+  }
+  $result = $query->execute()->fetchAll();
+
+  // Compile a list of vanished filters.
+  $broken_filters = array();
+  foreach ($result as $filter) {
+    $broken_filters[$filter->format][] = $filter->name;
+  }
+
+  if (isset($format_id)) {
+    return isset($broken_filters[$format_id]) ? $broken_filters[$format_id] : array();
+  }
+  return $broken_filters;
+}
+
+/**
  * Run all the enabled filters on a piece of text.
  *
  * Note: Because filters can inject JavaScript or execute PHP code, security is
@@ -681,7 +777,13 @@ function check_markup($text, $format_id 
   if (empty($format_id)) {
     $format_id = filter_fallback_format();
   }
-  $format = filter_format_load($format_id);
+  // If the requested text format does not exist, we return an empty text for
+  // security reasons. The text stays empty until the text format has been
+  // re-configured.
+  if (!$format = filter_format_load($format_id)) {
+    watchdog('filter', 'Missing text format: %format.', array('%format' => $format_id), WATCHDOG_ALERT);
+    return '';
+  }
 
   // Check for a cached version of this piece of text.
   $cache = $cache && !empty($format->cache);
@@ -703,7 +805,13 @@ function check_markup($text, $format_id 
 
   // Give filters the chance to escape HTML-like data such as code or formulas.
   foreach ($filters as $name => $filter) {
-    if ($filter->status && isset($filter_info[$name]['prepare callback']) && function_exists($filter_info[$name]['prepare callback'])) {
+    if ($filter->status == -1) {
+      // If a filter ought to run, but no longer exists, return an empty text.
+      // The text stays empty until the text format has been re-configured.
+      watchdog('filter', 'Missing filter %name in text format %format.', array('%name' => $name, '%format' => $format->name), WATCHDOG_ALERT);
+      return '';
+    }
+    elseif ($filter->status && isset($filter_info[$name]['prepare callback']) && function_exists($filter_info[$name]['prepare callback'])) {
       $function = $filter_info[$name]['prepare callback'];
       $text = $function($text, $filter, $format, $langcode, $cache, $cache_id);
     }
Index: modules/filter/filter.test
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.test,v
retrieving revision 1.71
diff -u -p -r1.71 filter.test
--- modules/filter/filter.test	5 Aug 2010 23:53:38 -0000	1.71
+++ modules/filter/filter.test	18 Aug 2010 22:59:59 -0000
@@ -640,6 +640,83 @@ class FilterNoFormatTestCase extends Dru
 }
 
 /**
+ * Security tests for missing/vanished text formats or filters.
+ */
+class FilterBrokenTestCase extends DrupalWebTestCase {
+  public static function getInfo() {
+    return array(
+      'name' => 'Vanishing filters and text formats',
+      'description' => 'Test the behavior of check_markup() when a filter or text format vanishes.',
+      'group' => 'Filter',
+    );
+  }
+
+  function setUp() {
+    parent::setUp('php', 'filter_test');
+    $this->admin_user = $this->drupalCreateUser(array('administer modules', 'administer filters', 'administer site configuration'));
+    $this->drupalLogin($this->admin_user);
+  }
+
+  /**
+   * Test that filtered content is emptied when an actively used filter module is disabled.
+   */
+  function testDisableFilterModule() {
+    // Create a new node.
+    $node = $this->drupalCreateNode(array('promote' => 1));
+    $body_raw = $node->body[LANGUAGE_NONE][0]['value'];
+    $format_id = $node->body[LANGUAGE_NONE][0]['format'];
+    $this->drupalGet('node/' . $node->nid);
+    $this->assertText($body_raw, t('Node body found.'));
+
+    // Enable the filter_test_replace filter.
+    $edit = array(
+      'filters[filter_test_replace][status]' => 1,
+    );
+    $this->drupalPost('admin/config/content/formats/' . $format_id, $edit, t('Save configuration'));
+
+    // Verify that filter_test_replace filter replaced the content.
+    $this->drupalGet('node/' . $node->nid);
+    $this->assertNoText($body_raw, t('Node body not found.'));
+    $this->assertText('Filter: Testing filter', t('Testing filter output found.'));
+
+    // Force disable the filter_test module, without disabling dependent modules.
+    module_disable(array('filter_test'), FALSE);
+
+    // Verify that the content is now empty, because the text format contains
+    // a filter that no longer exists.
+    $this->drupalGet('node/' . $node->nid);
+    $this->assertNoText($body_raw, t('Node body not found.'));
+    $this->assertNoText('Filter: Testing filter', t('Testing filter output not found.'));
+
+    // Ensure we get an error message on the text format overview page.
+    $this->drupalGet('admin/config/content/formats');
+    $this->assertText(t('The highlighted text format needs to be re-configured, because it is using filters that no longer exist.'), t('Error message about broken text formats found.'));
+
+    // Also ensure we get a requirements error.
+    $this->drupalGet('admin/reports/status');
+    $this->assertRaw(t('At least one text format contains a filter that was provided by a module that has been disabled. All content using one of these text formats will be hidden until <a href="@text-format-url">text formats have been re-configured</a>.', array('@text-format-url' => url('admin/config/content/formats'))), t('Requirements error found.'));
+
+    // Update text format.
+    $this->drupalPost('admin/config/content/formats/' . $format_id, array(), t('Save configuration'));
+
+    // Verify no error message.
+    $this->drupalGet('admin/config/content/formats');
+    $this->assertNoText(t('The highlighted text format needs to be re-configured, because it is using filters that no longer exist.'), t('Error message about broken text formats not found.'));
+
+    // Verify node content is displayed.
+    $this->drupalGet('node/' . $node->nid);
+    $this->assertText($body_raw, t('Node body found.'));
+
+    // Delete the text format entirely.
+    $this->drupalPost('admin/config/content/formats/' . $format_id . '/delete', array(), t('Delete'));
+
+    // Verify that the content is empty, because the text format does not exist.
+    $this->drupalGet('node/' . $node->nid);
+    $this->assertNoText($body_raw, t('Node body not found.'));
+  }
+}
+
+/**
  * Unit tests for core filters.
  */
 class FilterUnitTestCase extends DrupalUnitTestCase {
Index: modules/php/php.install
===================================================================
RCS file: /cvs/drupal/drupal/modules/php/php.install,v
retrieving revision 1.17
diff -u -p -r1.17 php.install
--- modules/php/php.install	9 Jan 2010 23:03:21 -0000	1.17
+++ modules/php/php.install	18 Aug 2010 22:55:43 -0000
@@ -36,10 +36,3 @@ function php_enable() {
     drupal_set_message(t('A !php-code text format has been created.', array('!php-code' => l('PHP code', 'admin/config/content/formats/' . $php_format->format))));
   }
 }
-
-/**
- * Implements hook_disable().
- */
-function php_disable() {
-  drupal_set_message(t('The PHP module has been disabled. Any existing content that was using the PHP filter will now be visible in plain text. This might pose a security risk by exposing sensitive information, if any, used in the PHP code.'));
-}
Index: modules/simpletest/tests/filter_test.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/simpletest/tests/filter_test.module,v
retrieving revision 1.4
diff -u -p -r1.4 filter_test.module
--- modules/simpletest/tests/filter_test.module	4 Dec 2009 16:49:47 -0000	1.4
+++ modules/simpletest/tests/filter_test.module	18 Aug 2010 22:55:43 -0000
@@ -36,6 +36,28 @@ function filter_test_filter_info() {
     'description' => 'Does nothing, but makes a text format uncacheable.',
     'cache' => FALSE,
   );
+  $filters['filter_test_replace'] = array(
+    'title' => 'Testing filter',
+    'description' => 'Replaces all content with filter and text format information.',
+    'process callback' => 'filter_test_replace',
+  );
   return $filters;
 }
 
+/**
+ * Process handler for filter_test_replace filter.
+ *
+ * Replaces all text with filter and text format information.
+ */
+function filter_test_replace($text, $filter, $format, $langcode, $cache, $cache_id) {
+  $text = array();
+  $text[] = 'Filter: ' . $filter->title . ' (' . $filter->name . ')';
+  $text[] = 'Format: ' . $format->name . ' (' . $format->format . ')';
+  $text[] = 'Language: ' . $langcode;
+  $text[] = 'Cache: ' . ($cache ? 'Enabled' : 'Disabled');
+  if ($cache_id) {
+    $text[] = 'Cache ID: ' . $cache_id;
+  }
+  return implode("<br />\n", $text);
+}
+
Index: modules/system/system-messages.css
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system-messages.css,v
retrieving revision 1.2
diff -u -p -r1.2 system-messages.css
--- modules/system/system-messages.css	26 May 2010 19:51:01 -0000	1.2
+++ modules/system/system-messages.css	18 Aug 2010 22:55:43 -0000
@@ -12,7 +12,7 @@ div.messages {
 }
 
 div.error,
-tr.error {
+table tr.error {
   background-color: #fcc;
 }
 
Index: modules/system/system.admin.inc
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system.admin.inc,v
retrieving revision 1.296
diff -u -p -r1.296 system.admin.inc
--- modules/system/system.admin.inc	18 Aug 2010 18:41:30 -0000	1.296
+++ modules/system/system.admin.inc	18 Aug 2010 23:03:56 -0000
@@ -1243,16 +1243,12 @@ function system_modules_submit($form, &$
     drupal_set_message(t('The configuration options have been saved.'));
   }
 
-  // Clear all caches.
-  registry_rebuild();
-  system_rebuild_theme_data();
-  drupal_theme_rebuild();
-  node_types_rebuild();
-  menu_rebuild();
-  cache_clear_all('schema', 'cache');
+  // Clear all caches. We need to invoke drupal_flush_all_caches() to ensure
+  // that also dependent caches are flushed, e.g. the filter cache and field
+  // cache, and also registered themes are rebuilt, since modules can also
+  // register themes.
+  drupal_flush_all_caches();
   entity_info_cache_clear();
-  drupal_clear_css_cache();
-  drupal_clear_js_cache();
 
   $form_state['redirect'] = 'admin/modules';
 
@@ -1263,8 +1259,6 @@ function system_modules_submit($form, &$
 
   // Synchronize to catch any actions that were added or removed.
   actions_synchronize();
-
-  return;
 }
 
 /**
