Index: modules/field/field.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/field/field.module,v
retrieving revision 1.67
diff -u -p -r1.67 field.module
--- modules/field/field.module	3 Mar 2010 07:40:58 -0000	1.67
+++ modules/field/field.module	6 Mar 2010 16:48:38 -0000
@@ -241,7 +241,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.48
diff -u -p -r1.48 text.module
--- modules/field/modules/text/text.module	11 Feb 2010 17:44:47 -0000	1.48
+++ modules/field/modules/text/text.module	6 Mar 2010 16:49:46 -0000
@@ -596,3 +596,17 @@ function text_field_widget_formatted_tex
     return $edit;
   }
 }
+
+/**
+ * Implements hook_filter_format_update().
+ */
+function text_filter_format_update($format) {
+  field_cache_clear();
+}
+
+/**
+ * Implements hook_filter_format_delete().
+ */
+function text_filter_format_delete($format) {
+  field_cache_clear();
+}
Index: modules/filter/filter.admin.inc
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.admin.inc,v
retrieving revision 1.57
diff -u -p -r1.57 filter.admin.inc
--- modules/filter/filter.admin.inc	5 Feb 2010 21:44:35 -0000	1.57
+++ modules/filter/filter.admin.inc	6 Mar 2010 16:47:14 -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);
@@ -74,7 +82,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.34
diff -u -p -r1.34 filter.install
--- modules/filter/filter.install	5 Mar 2010 13:32:09 -0000	1.34
+++ modules/filter/filter.install	6 Mar 2010 16:47:14 -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.319
diff -u -p -r1.319 filter.module
--- modules/filter/filter.module	6 Mar 2010 06:39:00 -0000	1.319
+++ modules/filter/filter.module	6 Mar 2010 16:47:15 -0000
@@ -170,6 +170,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)) {
@@ -177,13 +181,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'])) {
@@ -256,6 +267,7 @@ function filter_format_delete($format) {
   $fallback = filter_format_load(filter_fallback_format());
   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);
 }
@@ -619,9 +631,14 @@ function filter_list_format($format_id) 
         $filter->title = $filter_info[$name]['title'];
         // Unpack stored filter settings.
         $filter->settings = (isset($filter->settings) ? unserialize($filter->settings) : array());
-
-        $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;
   }
@@ -630,6 +647,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
@@ -656,7 +725,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);
@@ -678,9 +753,17 @@ 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'])) {
-      $function = $filter_info[$name]['prepare callback'];
-      $text = $function($text, $filter, $format, $langcode, $cache, $cache_id);
+    if ($filter->status >= 0) {
+      if ($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);
+      }
+    }
+    // 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.
+    else {
+      watchdog('filter', 'Missing filter %name in text format %format.', array('%name' => $name, '%format' => $format->name), WATCHDOG_ALERT);
+      return '';
     }
   }
 
@@ -1309,3 +1392,38 @@ function _filter_html_escape_tips($filte
 /**
  * @} End of "Standard filters".
  */
+
+/**
+ * Implements hook_modules_disabled().
+ *
+ * In case a module providing a filter is disabled, the site administrator needs
+ * to manually update the text formats that have it applied. We cannot
+ * automatically disable or remove filters, because that would likely break the
+ * intended filtering logic in a text format.
+ *
+ * @see filter_formats()
+ * @see check_markup()
+ */
+function filter_modules_disabled($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 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().
+ *
+ * In case a module providing a filter is uninstalled, the site administrator
+ * needs to manually update the text formats that have it applied. We cannot
+ * automatically disable or remove filters, because that would likely break the
+ * intended filtering logic in a text format.
+ *
+ * @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');
+  }
+}
+
Index: modules/filter/filter.test
===================================================================
RCS file: /cvs/drupal/drupal/modules/filter/filter.test,v
retrieving revision 1.60
diff -u -p -r1.60 filter.test
--- modules/filter/filter.test	6 Mar 2010 06:39:00 -0000	1.60
+++ modules/filter/filter.test	6 Mar 2010 16:47:15 -0000
@@ -568,6 +568,88 @@ 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.'));
+
+    // Disable the filter_test module.
+    $edit = array(
+      'modules[Testing][filter_test][enable]' => FALSE,
+    );
+    $this->drupalPost('admin/modules', $edit, t('Save configuration'));
+    $this->assertText(t('The configuration options have been saved.'), t('Module list has changed.'));
+    $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('Error message about disabled filter module displayed.'));
+
+    // 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/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	6 Mar 2010 16:47:15 -0000
@@ -28,6 +28,17 @@ function filter_test_filter_format_delet
 }
 
 /**
+ * Implements hook_system_info_alter().
+ */
+function filter_test_system_info_alter(&$info, $file) {
+  // Make this testing module visible on the module configuration page, so tests
+  // can disable it.
+  if ($file->name == 'filter_test') {
+    $info['hidden'] = FALSE;
+  }
+}
+
+/**
  * Implements hook_filter_info().
  */
 function filter_test_filter_info() {
@@ -36,6 +47,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.admin.inc
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system.admin.inc,v
retrieving revision 1.260
diff -u -p -r1.260 system.admin.inc
--- modules/system/system.admin.inc	3 Mar 2010 19:46:26 -0000	1.260
+++ modules/system/system.admin.inc	6 Mar 2010 16:47:15 -0000
@@ -1236,15 +1236,12 @@ function system_modules_submit($form, &$
     drupal_set_message(t('The configuration options have been saved.'));
   }
 
-  // Clear all caches.
-  registry_rebuild();
-  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';
 
@@ -1255,8 +1252,6 @@ function system_modules_submit($form, &$
 
   // Synchronize to catch any actions that were added or removed.
   actions_synchronize();
-
-  return;
 }
 
 /**
Index: modules/system/system.css
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system.css,v
retrieving revision 1.73
diff -u -p -r1.73 system.css
--- modules/system/system.css	18 Jan 2010 17:29:28 -0000	1.73
+++ modules/system/system.css	6 Mar 2010 16:47:15 -0000
@@ -70,7 +70,7 @@ thead th {
 div.error {
   border: 1px solid #d77;
 }
-div.error, tr.error {
+div.error, table tr.error {
   background: #fcc;
   color: #200;
   padding: 2px;
