OSDN Git Service

Daily bump.
[pf3gnuchains/gcc-fork.git] / gcc / df-problems.c
index f45c6d2..25bbba6 100644 (file)
@@ -1,7 +1,7 @@
 /* Standard problems for dataflow support routines.
 /* Standard problems for dataflow support routines.
-   Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007
-   Free Software Foundation, Inc.
-   Originally contributed by Michael P. Hayes 
+   Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007,
+   2008, 2009, 2010 Free Software Foundation, Inc.
+   Originally contributed by Michael P. Hayes
              (m.hayes@elec.canterbury.ac.nz, mhayes@redhat.com)
    Major rewrite contributed by Danny Berlin (dberlin@dberlin.org)
              and Kenneth Zadeck (zadeck@naturalbridge.com).
              (m.hayes@elec.canterbury.ac.nz, mhayes@redhat.com)
    Major rewrite contributed by Danny Berlin (dberlin@dberlin.org)
              and Kenneth Zadeck (zadeck@naturalbridge.com).
@@ -47,7 +47,7 @@ along with GCC; see the file COPYING3.  If not see
 
 /* Note that turning REG_DEAD_DEBUGGING on will cause
    gcc.c-torture/unsorted/dump-noaddr.c to fail because it prints
 
 /* Note that turning REG_DEAD_DEBUGGING on will cause
    gcc.c-torture/unsorted/dump-noaddr.c to fail because it prints
-   addresses in the dumps.  */  
+   addresses in the dumps.  */
 #if 0
 #define REG_DEAD_DEBUGGING
 #endif
 #if 0
 #define REG_DEAD_DEBUGGING
 #endif
@@ -73,7 +73,7 @@ df_get_live_out (basic_block bb)
 
   if (df_live)
     return DF_LIVE_OUT (bb);
 
   if (df_live)
     return DF_LIVE_OUT (bb);
-  else 
+  else
     return DF_LR_OUT (bb);
 }
 
     return DF_LR_OUT (bb);
 }
 
@@ -89,7 +89,7 @@ df_get_live_in (basic_block bb)
 
   if (df_live)
     return DF_LIVE_IN (bb);
 
   if (df_live)
     return DF_LIVE_IN (bb);
-  else 
+  else
     return DF_LR_IN (bb);
 }
 
     return DF_LR_IN (bb);
 }
 
@@ -109,8 +109,7 @@ df_grow_bb_info (struct dataflow *dflow)
   if (dflow->block_info_size < new_size)
     {
       new_size += new_size / 4;
   if (dflow->block_info_size < new_size)
     {
       new_size += new_size / 4;
-      dflow->block_info = xrealloc (dflow->block_info, 
-                                   new_size *sizeof (void*));
+      dflow->block_info = XRESIZEVEC (void *, dflow->block_info, new_size);
       memset (dflow->block_info + dflow->block_info_size, 0,
              (new_size - dflow->block_info_size) *sizeof (void *));
       dflow->block_info_size = new_size;
       memset (dflow->block_info + dflow->block_info_size, 0,
              (new_size - dflow->block_info_size) *sizeof (void *));
       dflow->block_info_size = new_size;
@@ -129,7 +128,7 @@ df_chain_dump (struct df_link *link, FILE *file)
               DF_REF_REG_DEF_P (link->ref) ? 'd' : 'u',
               DF_REF_ID (link->ref),
               DF_REF_BBNO (link->ref),
               DF_REF_REG_DEF_P (link->ref) ? 'd' : 'u',
               DF_REF_ID (link->ref),
               DF_REF_BBNO (link->ref),
-              DF_REF_INSN (link->ref) ? DF_REF_INSN_UID (link->ref) : -1);
+              DF_REF_IS_ARTIFICIAL (link->ref) ? -1 : DF_REF_INSN_UID (link->ref));
     }
   fprintf (file, "}");
 }
     }
   fprintf (file, "}");
 }
@@ -137,7 +136,7 @@ df_chain_dump (struct df_link *link, FILE *file)
 
 /* Print some basic block info as part of df_dump.  */
 
 
 /* Print some basic block info as part of df_dump.  */
 
-void 
+void
 df_print_bb_index (basic_block bb, FILE *file)
 {
   edge e;
 df_print_bb_index (basic_block bb, FILE *file)
 {
   edge e;
@@ -148,37 +147,16 @@ df_print_bb_index (basic_block bb, FILE *file)
     {
       basic_block pred = e->src;
       fprintf (file, "%d%s ", pred->index, e->flags & EDGE_EH ? "(EH)" : "");
     {
       basic_block pred = e->src;
       fprintf (file, "%d%s ", pred->index, e->flags & EDGE_EH ? "(EH)" : "");
-    } 
+    }
   fprintf (file, ")->[%d]->( ", bb->index);
   FOR_EACH_EDGE (e, ei, bb->succs)
     {
       basic_block succ = e->dest;
       fprintf (file, "%d%s ", succ->index, e->flags & EDGE_EH ? "(EH)" : "");
   fprintf (file, ")->[%d]->( ", bb->index);
   FOR_EACH_EDGE (e, ei, bb->succs)
     {
       basic_block succ = e->dest;
       fprintf (file, "%d%s ", succ->index, e->flags & EDGE_EH ? "(EH)" : "");
-    } 
+    }
   fprintf (file, ")\n");
 }
 
   fprintf (file, ")\n");
 }
 
-
-
-/* Make sure that the seen_in_insn and seen_in_block sbitmaps are set
-   up correctly. */
-
-static void
-df_set_seen (void)
-{
-  seen_in_block = BITMAP_ALLOC (&df_bitmap_obstack);
-  seen_in_insn = BITMAP_ALLOC (&df_bitmap_obstack);
-}
-
-
-static void
-df_unset_seen (void)
-{
-  BITMAP_FREE (seen_in_block);
-  BITMAP_FREE (seen_in_insn);
-}
-
-
 \f
 /*----------------------------------------------------------------------------
    REACHING DEFINITIONS
 \f
 /*----------------------------------------------------------------------------
    REACHING DEFINITIONS
@@ -190,12 +168,12 @@ df_unset_seen (void)
    ----------------------------------------------------------------------------*/
 
 /* This problem plays a large number of games for the sake of
    ----------------------------------------------------------------------------*/
 
 /* This problem plays a large number of games for the sake of
-   efficiency.  
-   
+   efficiency.
+
    1) The order of the bits in the bitvectors.  After the scanning
    phase, all of the defs are sorted.  All of the defs for the reg 0
    are first, followed by all defs for reg 1 and so on.
    1) The order of the bits in the bitvectors.  After the scanning
    phase, all of the defs are sorted.  All of the defs for the reg 0
    are first, followed by all defs for reg 1 and so on.
-   
+
    2) There are two kill sets, one if the number of defs is less or
    equal to DF_SPARSE_THRESHOLD and another if the number of defs is
    greater.
    2) There are two kill sets, one if the number of defs is less or
    equal to DF_SPARSE_THRESHOLD and another if the number of defs is
    greater.
@@ -217,8 +195,8 @@ df_unset_seen (void)
 struct df_rd_problem_data
 {
   /* The set of defs to regs invalidated by call.  */
 struct df_rd_problem_data
 {
   /* The set of defs to regs invalidated by call.  */
-  bitmap sparse_invalidated_by_call;  
-  /* The set of defs to regs invalidate by call for rd.  */  
+  bitmap sparse_invalidated_by_call;
+  /* The set of defs to regs invalidate by call for rd.  */
   bitmap dense_invalidated_by_call;
   /* An obstack for the bitmaps we need for this problem.  */
   bitmap_obstack rd_bitmaps;
   bitmap dense_invalidated_by_call;
   /* An obstack for the bitmaps we need for this problem.  */
   bitmap_obstack rd_bitmaps;
@@ -227,7 +205,7 @@ struct df_rd_problem_data
 /* Set basic block info.  */
 
 static void
 /* Set basic block info.  */
 
 static void
-df_rd_set_bb_info (unsigned int index, 
+df_rd_set_bb_info (unsigned int index,
                   struct df_rd_bb_info *bb_info)
 {
   gcc_assert (df_rd);
                   struct df_rd_bb_info *bb_info)
 {
   gcc_assert (df_rd);
@@ -239,14 +217,12 @@ df_rd_set_bb_info (unsigned int index,
 /* Free basic block info.  */
 
 static void
 /* Free basic block info.  */
 
 static void
-df_rd_free_bb_info (basic_block bb ATTRIBUTE_UNUSED, 
+df_rd_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
                    void *vbb_info)
 {
   struct df_rd_bb_info *bb_info = (struct df_rd_bb_info *) vbb_info;
   if (bb_info)
     {
                    void *vbb_info)
 {
   struct df_rd_bb_info *bb_info = (struct df_rd_bb_info *) vbb_info;
   if (bb_info)
     {
-      if (bb_info->expanded_lr_out)
-       BITMAP_FREE (bb_info->expanded_lr_out);
       BITMAP_FREE (bb_info->kill);
       BITMAP_FREE (bb_info->sparse_kill);
       BITMAP_FREE (bb_info->gen);
       BITMAP_FREE (bb_info->kill);
       BITMAP_FREE (bb_info->sparse_kill);
       BITMAP_FREE (bb_info->gen);
@@ -260,7 +236,7 @@ df_rd_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
 /* Allocate or reset bitmaps for DF_RD blocks. The solution bits are
    not touched unless the block is new.  */
 
 /* Allocate or reset bitmaps for DF_RD blocks. The solution bits are
    not touched unless the block is new.  */
 
-static void 
+static void
 df_rd_alloc (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_rd_alloc (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -268,7 +244,7 @@ df_rd_alloc (bitmap all_blocks)
   struct df_rd_problem_data *problem_data;
 
   if (!df_rd->block_pool)
   struct df_rd_problem_data *problem_data;
 
   if (!df_rd->block_pool)
-    df_rd->block_pool = create_alloc_pool ("df_rd_block pool", 
+    df_rd->block_pool = create_alloc_pool ("df_rd_block pool",
                                           sizeof (struct df_rd_bb_info), 50);
 
   if (df_rd->problem_data)
                                           sizeof (struct df_rd_bb_info), 50);
 
   if (df_rd->problem_data)
@@ -277,7 +253,7 @@ df_rd_alloc (bitmap all_blocks)
       bitmap_clear (problem_data->sparse_invalidated_by_call);
       bitmap_clear (problem_data->dense_invalidated_by_call);
     }
       bitmap_clear (problem_data->sparse_invalidated_by_call);
       bitmap_clear (problem_data->dense_invalidated_by_call);
     }
-  else 
+  else
     {
       problem_data = XNEW (struct df_rd_problem_data);
       df_rd->problem_data = problem_data;
     {
       problem_data = XNEW (struct df_rd_problem_data);
       df_rd->problem_data = problem_data;
@@ -299,21 +275,15 @@ df_rd_alloc (bitmap all_blocks)
     {
       struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
       if (bb_info)
     {
       struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
       if (bb_info)
-       { 
-         if (bb_info->expanded_lr_out)
-           bitmap_clear (bb_info->expanded_lr_out);
+       {
          bitmap_clear (bb_info->kill);
          bitmap_clear (bb_info->sparse_kill);
          bitmap_clear (bb_info->gen);
        }
       else
          bitmap_clear (bb_info->kill);
          bitmap_clear (bb_info->sparse_kill);
          bitmap_clear (bb_info->gen);
        }
       else
-       { 
+       {
          bb_info = (struct df_rd_bb_info *) pool_alloc (df_rd->block_pool);
          df_rd_set_bb_info (bb_index, bb_info);
          bb_info = (struct df_rd_bb_info *) pool_alloc (df_rd->block_pool);
          df_rd_set_bb_info (bb_index, bb_info);
-         if (df->changeable_flags & DF_RD_NO_TRIM)
-           bb_info->expanded_lr_out = NULL;
-         else
-           bb_info->expanded_lr_out = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->kill = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->sparse_kill = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->gen = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->kill = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->sparse_kill = BITMAP_ALLOC (&problem_data->rd_bitmaps);
          bb_info->gen = BITMAP_ALLOC (&problem_data->rd_bitmaps);
@@ -325,56 +295,113 @@ df_rd_alloc (bitmap all_blocks)
 }
 
 
 }
 
 
-/* Process a list of DEFs for df_rd_bb_local_compute.  */
+/* Add the effect of the top artificial defs of BB to the reaching definitions
+   bitmap LOCAL_RD.  */
 
 
-static void
-df_rd_bb_local_compute_process_def (struct df_rd_bb_info *bb_info,
-                                   struct df_ref **def_rec,
-                                   enum df_ref_flags top_flag)
+void
+df_rd_simulate_artificial_defs_at_top (basic_block bb, bitmap local_rd)
 {
 {
-  for (; *def_rec; def_rec++)
+  int bb_index = bb->index;
+  df_ref *def_rec;
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
     {
-      struct df_ref *def = *def_rec;
-      unsigned int regno = DF_REF_REGNO (def);
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
+           bitmap_clear_range (local_rd,
+                               DF_DEFS_BEGIN (dregno),
+                               DF_DEFS_COUNT (dregno));
+         bitmap_set_bit (local_rd, DF_REF_ID (def));
+       }
+    }
+}
 
 
-      /* This makes sure we do the artificial defs in the right order
-        since they are all in the same list.  */
-      if (top_flag != (DF_REF_FLAGS (def) & DF_REF_AT_TOP))
-       continue;
+/* Add the effect of the defs of INSN to the reaching definitions bitmap
+   LOCAL_RD.  */
 
 
-      /* Skip over the hard regs if we do not care about them.  */
-      if ((df->changeable_flags & DF_NO_HARD_REGS) && 
-         (regno < FIRST_PSEUDO_REGISTER))
-       continue;
+void
+df_rd_simulate_one_insn (basic_block bb ATTRIBUTE_UNUSED, rtx insn,
+                        bitmap local_rd)
+{
+  unsigned uid = INSN_UID (insn);
+  df_ref *def_rec;
 
 
-      /* Only the last def(s) for a regno in the block has any
-        effect.  */ 
-      if (bitmap_bit_p (seen_in_block, regno))
-       continue;
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      unsigned int dregno = DF_REF_REGNO (def);
+      if ((!(df->changeable_flags & DF_NO_HARD_REGS))
+          || (dregno >= FIRST_PSEUDO_REGISTER))
+        {
+          if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
+           bitmap_clear_range (local_rd,
+                               DF_DEFS_BEGIN (dregno),
+                               DF_DEFS_COUNT (dregno));
+         if (!(DF_REF_FLAGS (def)
+               & (DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER)))
+           bitmap_set_bit (local_rd, DF_REF_ID (def));
+       }
+    }
+}
 
 
-      /* The first def for regno in insn gets to knock out the
-        defs from other instructions.  */
-      if ((!bitmap_bit_p (seen_in_insn, regno))
-         /* If the def is to only part of the reg, it does
-            not kill the other defs that reach here.  */
-         && (!(DF_REF_FLAGS (def) & 
-               (DF_REF_PARTIAL | DF_REF_CONDITIONAL | DF_REF_MAY_CLOBBER))))
+/* Process a list of DEFs for df_rd_bb_local_compute.  This is a bit
+   more complicated than just simulating, because we must produce the
+   gen and kill sets and hence deal with the two possible representations
+   of kill sets.   */
+
+static void
+df_rd_bb_local_compute_process_def (struct df_rd_bb_info *bb_info,
+                                   df_ref *def_rec,
+                                   int top_flag)
+{
+  while (*def_rec)
+    {
+      df_ref def = *def_rec;
+      if (top_flag == (DF_REF_FLAGS (def) & DF_REF_AT_TOP))
        {
        {
+         unsigned int regno = DF_REF_REGNO (def);
          unsigned int begin = DF_DEFS_BEGIN (regno);
          unsigned int n_defs = DF_DEFS_COUNT (regno);
          unsigned int begin = DF_DEFS_BEGIN (regno);
          unsigned int n_defs = DF_DEFS_COUNT (regno);
-         if (n_defs > DF_SPARSE_THRESHOLD)
-           bitmap_set_bit (bb_info->sparse_kill, regno);
-         else
-           bitmap_set_range (bb_info->kill, begin, n_defs);
-         bitmap_clear_range(bb_info->gen, begin, n_defs);
+
+         if ((!(df->changeable_flags & DF_NO_HARD_REGS))
+             || (regno >= FIRST_PSEUDO_REGISTER))
+           {
+             /* Only the last def(s) for a regno in the block has any
+                effect.  */
+             if (!bitmap_bit_p (seen_in_block, regno))
+               {
+                 /* The first def for regno in insn gets to knock out the
+                    defs from other instructions.  */
+                 if ((!bitmap_bit_p (seen_in_insn, regno))
+                     /* If the def is to only part of the reg, it does
+                        not kill the other defs that reach here.  */
+                     && (!(DF_REF_FLAGS (def) &
+                           (DF_REF_PARTIAL | DF_REF_CONDITIONAL | DF_REF_MAY_CLOBBER))))
+                   {
+                     if (n_defs > DF_SPARSE_THRESHOLD)
+                       {
+                         bitmap_set_bit (bb_info->sparse_kill, regno);
+                         bitmap_clear_range(bb_info->gen, begin, n_defs);
+                       }
+                     else
+                       {
+                         bitmap_set_range (bb_info->kill, begin, n_defs);
+                         bitmap_clear_range (bb_info->gen, begin, n_defs);
+                       }
+                   }
+
+                 bitmap_set_bit (seen_in_insn, regno);
+                 /* All defs for regno in the instruction may be put into
+                    the gen set.  */
+                 if (!(DF_REF_FLAGS (def)
+                       & (DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER)))
+                   bitmap_set_bit (bb_info->gen, DF_REF_ID (def));
+               }
+           }
        }
        }
-      
-      bitmap_set_bit (seen_in_insn, regno);
-      /* All defs for regno in the instruction may be put into
-        the gen set.  */
-      if (!(DF_REF_FLAGS (def) 
-           & (DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER)))
-       bitmap_set_bit (bb_info->gen, DF_REF_ID (def));
+      def_rec++;
     }
 }
 
     }
 }
 
@@ -385,25 +412,11 @@ df_rd_bb_local_compute (unsigned int bb_index)
 {
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
 {
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
-  struct df_lr_bb_info *lr_bb_info = df_lr_get_bb_info (bb_index);
   rtx insn;
 
   bitmap_clear (seen_in_block);
   bitmap_clear (seen_in_insn);
 
   rtx insn;
 
   bitmap_clear (seen_in_block);
   bitmap_clear (seen_in_insn);
 
-  if (!(df->changeable_flags & DF_RD_NO_TRIM))
-    {
-      unsigned int regno;
-      bitmap_iterator bi;
-      int first_reg = (df->changeable_flags & DF_NO_HARD_REGS) ? FIRST_PSEUDO_REGISTER : 0;
-      EXECUTE_IF_SET_IN_BITMAP (lr_bb_info->out, first_reg, regno, bi)
-       {
-         unsigned int begin = DF_DEFS_BEGIN (regno);
-         unsigned int n_defs = DF_DEFS_COUNT (regno);
-         bitmap_set_range (bb_info->expanded_lr_out, begin, n_defs);
-       }
-    }
-
   /* Artificials are only hard regs.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_rd_bb_local_compute_process_def (bb_info,
   /* Artificials are only hard regs.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_rd_bb_local_compute_process_def (bb_info,
@@ -417,7 +430,7 @@ df_rd_bb_local_compute (unsigned int bb_index)
       if (!INSN_P (insn))
        continue;
 
       if (!INSN_P (insn))
        continue;
 
-      df_rd_bb_local_compute_process_def (bb_info, 
+      df_rd_bb_local_compute_process_def (bb_info,
                                          DF_INSN_UID_DEFS (uid), 0);
 
       /* This complex dance with the two bitmaps is required because
                                          DF_INSN_UID_DEFS (uid), 0);
 
       /* This complex dance with the two bitmaps is required because
@@ -434,7 +447,7 @@ df_rd_bb_local_compute (unsigned int bb_index)
      are going backwards through the block and these are logically at
      the start.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
      are going backwards through the block and these are logically at
      the start.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
-    df_rd_bb_local_compute_process_def (bb_info, 
+    df_rd_bb_local_compute_process_def (bb_info,
                                        df_get_artificial_defs (bb_index),
                                        DF_REF_AT_TOP);
 }
                                        df_get_artificial_defs (bb_index),
                                        DF_REF_AT_TOP);
 }
@@ -453,7 +466,8 @@ df_rd_local_compute (bitmap all_blocks)
   bitmap sparse_invalidated = problem_data->sparse_invalidated_by_call;
   bitmap dense_invalidated = problem_data->dense_invalidated_by_call;
 
   bitmap sparse_invalidated = problem_data->sparse_invalidated_by_call;
   bitmap dense_invalidated = problem_data->dense_invalidated_by_call;
 
-  df_set_seen ();
+  seen_in_block = BITMAP_ALLOC (&df_bitmap_obstack);
+  seen_in_insn = BITMAP_ALLOC (&df_bitmap_obstack);
 
   df_maybe_reorganize_def_refs (DF_REF_ORDER_BY_REG);
 
 
   df_maybe_reorganize_def_refs (DF_REF_ORDER_BY_REG);
 
@@ -461,24 +475,26 @@ df_rd_local_compute (bitmap all_blocks)
     {
       df_rd_bb_local_compute (bb_index);
     }
     {
       df_rd_bb_local_compute (bb_index);
     }
-  
+
   /* Set up the knockout bit vectors to be applied across EH_EDGES.  */
   /* Set up the knockout bit vectors to be applied across EH_EDGES.  */
-  EXECUTE_IF_SET_IN_BITMAP (df_invalidated_by_call, 0, regno, bi)
+  EXECUTE_IF_SET_IN_BITMAP (regs_invalidated_by_call_regset, 0, regno, bi)
     {
       if (DF_DEFS_COUNT (regno) > DF_SPARSE_THRESHOLD)
        bitmap_set_bit (sparse_invalidated, regno);
       else
     {
       if (DF_DEFS_COUNT (regno) > DF_SPARSE_THRESHOLD)
        bitmap_set_bit (sparse_invalidated, regno);
       else
-       bitmap_set_range (dense_invalidated, 
-                         DF_DEFS_BEGIN (regno), 
+       bitmap_set_range (dense_invalidated,
+                         DF_DEFS_BEGIN (regno),
                          DF_DEFS_COUNT (regno));
     }
                          DF_DEFS_COUNT (regno));
     }
-  df_unset_seen ();
+
+  BITMAP_FREE (seen_in_block);
+  BITMAP_FREE (seen_in_insn);
 }
 
 
 /* Initialize the solution bit vectors for problem.  */
 
 }
 
 
 /* Initialize the solution bit vectors for problem.  */
 
-static void 
+static void
 df_rd_init_solution (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_rd_init_solution (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -487,7 +503,7 @@ df_rd_init_solution (bitmap all_blocks)
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
       struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
       struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
-      
+
       bitmap_copy (bb_info->out, bb_info->gen);
       bitmap_clear (bb_info->in);
     }
       bitmap_copy (bb_info->out, bb_info->gen);
       bitmap_clear (bb_info->in);
     }
@@ -501,13 +517,10 @@ df_rd_confluence_n (edge e)
   bitmap op1 = df_rd_get_bb_info (e->dest->index)->in;
   bitmap op2 = df_rd_get_bb_info (e->src->index)->out;
 
   bitmap op1 = df_rd_get_bb_info (e->dest->index)->in;
   bitmap op2 = df_rd_get_bb_info (e->src->index)->out;
 
-  if (e->flags & EDGE_FAKE) 
+  if (e->flags & EDGE_FAKE)
     return;
 
     return;
 
-  /* If we are trimming the solution, the invalidated_by_call code in
-     the lr problem makes this unnecessary.  However, if we do not
-     trim, we must take this into account.  */
-  if ((df->changeable_flags & DF_RD_NO_TRIM) && e->flags & EDGE_EH)
+  if (e->flags & EDGE_EH)
     {
       struct df_rd_problem_data *problem_data
        = (struct df_rd_problem_data *) df_rd->problem_data;
     {
       struct df_rd_problem_data *problem_data
        = (struct df_rd_problem_data *) df_rd->problem_data;
@@ -515,15 +528,15 @@ df_rd_confluence_n (edge e)
       bitmap dense_invalidated = problem_data->dense_invalidated_by_call;
       bitmap_iterator bi;
       unsigned int regno;
       bitmap dense_invalidated = problem_data->dense_invalidated_by_call;
       bitmap_iterator bi;
       unsigned int regno;
-      bitmap tmp = BITMAP_ALLOC (&problem_data->rd_bitmaps);
+      bitmap tmp = BITMAP_ALLOC (&df_bitmap_obstack);
 
       bitmap_copy (tmp, op2);
       bitmap_and_compl_into (tmp, dense_invalidated);
 
       EXECUTE_IF_SET_IN_BITMAP (sparse_invalidated, 0, regno, bi)
        {
 
       bitmap_copy (tmp, op2);
       bitmap_and_compl_into (tmp, dense_invalidated);
 
       EXECUTE_IF_SET_IN_BITMAP (sparse_invalidated, 0, regno, bi)
        {
-         bitmap_clear_range (tmp, 
-                             DF_DEFS_BEGIN (regno), 
+         bitmap_clear_range (tmp,
+                             DF_DEFS_BEGIN (regno),
                              DF_DEFS_COUNT (regno));
        }
       bitmap_ior_into (op1, tmp);
                              DF_DEFS_COUNT (regno));
        }
       bitmap_ior_into (op1, tmp);
@@ -547,13 +560,13 @@ df_rd_transfer_function (int bb_index)
   bitmap gen = bb_info->gen;
   bitmap kill = bb_info->kill;
   bitmap sparse_kill = bb_info->sparse_kill;
   bitmap gen = bb_info->gen;
   bitmap kill = bb_info->kill;
   bitmap sparse_kill = bb_info->sparse_kill;
-  bool changed = false;
 
 
-  if ((df->changeable_flags & DF_RD_NO_TRIM) && bitmap_empty_p (sparse_kill))
-    changed = bitmap_ior_and_compl (out, gen, in, kill);
-  else 
+  if (bitmap_empty_p (sparse_kill))
+    return  bitmap_ior_and_compl (out, gen, in, kill);
+  else
     {
       struct df_rd_problem_data *problem_data;
     {
       struct df_rd_problem_data *problem_data;
+      bool changed = false;
       bitmap tmp;
 
       /* Note that TMP is _not_ a temporary bitmap if we end up replacing
       bitmap tmp;
 
       /* Note that TMP is _not_ a temporary bitmap if we end up replacing
@@ -564,25 +577,22 @@ df_rd_transfer_function (int bb_index)
       bitmap_copy (tmp, in);
       EXECUTE_IF_SET_IN_BITMAP (sparse_kill, 0, regno, bi)
        {
       bitmap_copy (tmp, in);
       EXECUTE_IF_SET_IN_BITMAP (sparse_kill, 0, regno, bi)
        {
-         bitmap_clear_range (tmp, 
-                             DF_DEFS_BEGIN (regno), 
+         bitmap_clear_range (tmp,
+                             DF_DEFS_BEGIN (regno),
                              DF_DEFS_COUNT (regno));
        }
       bitmap_and_compl_into (tmp, kill);
       bitmap_ior_into (tmp, gen);
                              DF_DEFS_COUNT (regno));
        }
       bitmap_and_compl_into (tmp, kill);
       bitmap_ior_into (tmp, gen);
-      if (!(df->changeable_flags & DF_RD_NO_TRIM))
-       bitmap_and_into (tmp, bb_info->expanded_lr_out);
       changed = !bitmap_equal_p (tmp, out);
       if (changed)
        {
          BITMAP_FREE (out);
          bb_info->out = tmp;
        }
       changed = !bitmap_equal_p (tmp, out);
       if (changed)
        {
          BITMAP_FREE (out);
          bb_info->out = tmp;
        }
-      else 
-       BITMAP_FREE (tmp);
+      else
+         BITMAP_FREE (tmp);
+      return changed;
     }
     }
-
-  return changed;
 }
 
 
 }
 
 
@@ -591,32 +601,14 @@ df_rd_transfer_function (int bb_index)
 static void
 df_rd_free (void)
 {
 static void
 df_rd_free (void)
 {
-  unsigned int i;
   struct df_rd_problem_data *problem_data
     = (struct df_rd_problem_data *) df_rd->problem_data;
 
   if (problem_data)
     {
   struct df_rd_problem_data *problem_data
     = (struct df_rd_problem_data *) df_rd->problem_data;
 
   if (problem_data)
     {
-      for (i = 0; i < df_rd->block_info_size; i++)
-       {
-         struct df_rd_bb_info *bb_info = df_rd_get_bb_info (i);
-         if (bb_info)
-           {
-             if (bb_info->expanded_lr_out)
-               BITMAP_FREE (bb_info->expanded_lr_out);
-             BITMAP_FREE (bb_info->kill);
-             BITMAP_FREE (bb_info->sparse_kill);
-             BITMAP_FREE (bb_info->gen);
-             BITMAP_FREE (bb_info->in);
-             BITMAP_FREE (bb_info->out);
-           }
-       }
-      
       free_alloc_pool (df_rd->block_pool);
       free_alloc_pool (df_rd->block_pool);
-      BITMAP_FREE (problem_data->sparse_invalidated_by_call);
-      BITMAP_FREE (problem_data->dense_invalidated_by_call);
       bitmap_obstack_release (&problem_data->rd_bitmaps);
       bitmap_obstack_release (&problem_data->rd_bitmaps);
-      
+
       df_rd->block_info_size = 0;
       free (df_rd->block_info);
       free (df_rd->problem_data);
       df_rd->block_info_size = 0;
       free (df_rd->block_info);
       free (df_rd->problem_data);
@@ -634,8 +626,8 @@ df_rd_start_dump (FILE *file)
     = (struct df_rd_problem_data *) df_rd->problem_data;
   unsigned int m = DF_REG_SIZE(df);
   unsigned int regno;
     = (struct df_rd_problem_data *) df_rd->problem_data;
   unsigned int m = DF_REG_SIZE(df);
   unsigned int regno;
-  
-  if (!df_rd->block_info) 
+
+  if (!df_rd->block_info)
     return;
 
   fprintf (file, ";; Reaching defs:\n\n");
     return;
 
   fprintf (file, ";; Reaching defs:\n\n");
@@ -647,8 +639,8 @@ df_rd_start_dump (FILE *file)
 
   for (regno = 0; regno < m; regno++)
     if (DF_DEFS_COUNT (regno))
 
   for (regno = 0; regno < m; regno++)
     if (DF_DEFS_COUNT (regno))
-      fprintf (file, "%d[%d,%d] ", regno, 
-              DF_DEFS_BEGIN (regno), 
+      fprintf (file, "%d[%d,%d] ", regno,
+              DF_DEFS_BEGIN (regno),
               DF_DEFS_COUNT (regno));
   fprintf (file, "\n");
 
               DF_DEFS_COUNT (regno));
   fprintf (file, "\n");
 
@@ -663,7 +655,7 @@ df_rd_top_dump (basic_block bb, FILE *file)
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb->index);
   if (!bb_info || !bb_info->in)
     return;
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb->index);
   if (!bb_info || !bb_info->in)
     return;
-  
+
   fprintf (file, ";; rd  in  \t(%d)\n", (int) bitmap_count_bits (bb_info->in));
   dump_bitmap (file, bb_info->in);
   fprintf (file, ";; rd  gen \t(%d)\n", (int) bitmap_count_bits (bb_info->gen));
   fprintf (file, ";; rd  in  \t(%d)\n", (int) bitmap_count_bits (bb_info->in));
   dump_bitmap (file, bb_info->in);
   fprintf (file, ";; rd  gen \t(%d)\n", (int) bitmap_count_bits (bb_info->gen));
@@ -681,7 +673,7 @@ df_rd_bottom_dump (basic_block bb, FILE *file)
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb->index);
   if (!bb_info || !bb_info->out)
     return;
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb->index);
   if (!bb_info || !bb_info->out)
     return;
-  
+
   fprintf (file, ";; rd  out \t(%d)\n", (int) bitmap_count_bits (bb_info->out));
   dump_bitmap (file, bb_info->out);
 }
   fprintf (file, ";; rd  out \t(%d)\n", (int) bitmap_count_bits (bb_info->out));
   dump_bitmap (file, bb_info->out);
 }
@@ -698,8 +690,8 @@ static struct df_problem problem_RD =
   df_rd_local_compute,        /* Local compute function.  */
   df_rd_init_solution,        /* Init the solution specific data.  */
   df_worklist_dataflow,       /* Worklist solver.  */
   df_rd_local_compute,        /* Local compute function.  */
   df_rd_init_solution,        /* Init the solution specific data.  */
   df_worklist_dataflow,       /* Worklist solver.  */
-  NULL,                       /* Confluence operator 0.  */ 
-  df_rd_confluence_n,         /* Confluence operator n.  */ 
+  NULL,                       /* Confluence operator 0.  */
+  df_rd_confluence_n,         /* Confluence operator n.  */
   df_rd_transfer_function,    /* Transfer function.  */
   NULL,                       /* Finalize function.  */
   df_rd_free,                 /* Free all of the problem information.  */
   df_rd_transfer_function,    /* Transfer function.  */
   NULL,                       /* Finalize function.  */
   df_rd_free,                 /* Free all of the problem information.  */
@@ -710,15 +702,14 @@ static struct df_problem problem_RD =
   NULL,                       /* Incremental solution verify start.  */
   NULL,                       /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
   NULL,                       /* Incremental solution verify start.  */
   NULL,                       /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
-  TV_DF_RD,                   /* Timing variable.  */ 
+  TV_DF_RD,                   /* Timing variable.  */
   true                        /* Reset blocks on dropping out of blocks_to_analyze.  */
 };
 
 
 
   true                        /* Reset blocks on dropping out of blocks_to_analyze.  */
 };
 
 
 
-/* Create a new DATAFLOW instance and add it to an existing instance
-   of DF.  The returned structure is what is used to get at the
-   solution.  */
+/* Create a new RD instance and add it to the existing instance
+   of DF.  */
 
 void
 df_rd_add_problem (void)
 
 void
 df_rd_add_problem (void)
@@ -733,7 +724,7 @@ df_rd_add_problem (void)
 
    Find the locations in the function where any use of a pseudo can
    reach in the backwards direction.  In and out bitvectors are built
 
    Find the locations in the function where any use of a pseudo can
    reach in the backwards direction.  In and out bitvectors are built
-   for each basic block.  The regnum is used to index into these sets.
+   for each basic block.  The regno is used to index into these sets.
    See df.h for details.
    ----------------------------------------------------------------------------*/
 
    See df.h for details.
    ----------------------------------------------------------------------------*/
 
@@ -748,7 +739,7 @@ struct df_lr_problem_data
 /* Set basic block info.  */
 
 static void
 /* Set basic block info.  */
 
 static void
-df_lr_set_bb_info (unsigned int index, 
+df_lr_set_bb_info (unsigned int index,
                   struct df_lr_bb_info *bb_info)
 {
   gcc_assert (df_lr);
                   struct df_lr_bb_info *bb_info)
 {
   gcc_assert (df_lr);
@@ -756,11 +747,11 @@ df_lr_set_bb_info (unsigned int index,
   df_lr->block_info[index] = bb_info;
 }
 
   df_lr->block_info[index] = bb_info;
 }
 
+
 /* Free basic block info.  */
 
 static void
 /* Free basic block info.  */
 
 static void
-df_lr_free_bb_info (basic_block bb ATTRIBUTE_UNUSED, 
+df_lr_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
                    void *vbb_info)
 {
   struct df_lr_bb_info *bb_info = (struct df_lr_bb_info *) vbb_info;
                    void *vbb_info)
 {
   struct df_lr_bb_info *bb_info = (struct df_lr_bb_info *) vbb_info;
@@ -778,14 +769,14 @@ df_lr_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
 /* Allocate or reset bitmaps for DF_LR blocks. The solution bits are
    not touched unless the block is new.  */
 
 /* Allocate or reset bitmaps for DF_LR blocks. The solution bits are
    not touched unless the block is new.  */
 
-static void 
+static void
 df_lr_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 
   if (!df_lr->block_pool)
 df_lr_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 
   if (!df_lr->block_pool)
-    df_lr->block_pool = create_alloc_pool ("df_lr_block pool", 
+    df_lr->block_pool = create_alloc_pool ("df_lr_block pool",
                                           sizeof (struct df_lr_bb_info), 50);
 
   df_grow_bb_info (df_lr);
                                           sizeof (struct df_lr_bb_info), 50);
 
   df_grow_bb_info (df_lr);
@@ -794,12 +785,12 @@ df_lr_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
     {
       struct df_lr_bb_info *bb_info = df_lr_get_bb_info (bb_index);
       if (bb_info)
     {
       struct df_lr_bb_info *bb_info = df_lr_get_bb_info (bb_index);
       if (bb_info)
-       { 
+       {
          bitmap_clear (bb_info->def);
          bitmap_clear (bb_info->use);
        }
       else
          bitmap_clear (bb_info->def);
          bitmap_clear (bb_info->use);
        }
       else
-       { 
+       {
          bb_info = (struct df_lr_bb_info *) pool_alloc (df_lr->block_pool);
          df_lr_set_bb_info (bb_index, bb_info);
          bb_info->use = BITMAP_ALLOC (NULL);
          bb_info = (struct df_lr_bb_info *) pool_alloc (df_lr->block_pool);
          df_lr_set_bb_info (bb_index, bb_info);
          bb_info->use = BITMAP_ALLOC (NULL);
@@ -815,7 +806,7 @@ df_lr_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 
 /* Reset the global solution for recalculation.  */
 
 
 /* Reset the global solution for recalculation.  */
 
-static void 
+static void
 df_lr_reset (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_lr_reset (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -839,13 +830,13 @@ df_lr_bb_local_compute (unsigned int bb_index)
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_lr_bb_info *bb_info = df_lr_get_bb_info (bb_index);
   rtx insn;
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_lr_bb_info *bb_info = df_lr_get_bb_info (bb_index);
   rtx insn;
-  struct df_ref **def_rec;
-  struct df_ref **use_rec;
+  df_ref *def_rec;
+  df_ref *use_rec;
 
   /* Process the registers set in an exception handler.  */
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
 
   /* Process the registers set in an exception handler.  */
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
-      struct df_ref *def = *def_rec;
+      df_ref def = *def_rec;
       if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
        {
          unsigned int dregno = DF_REF_REGNO (def);
       if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
        {
          unsigned int dregno = DF_REF_REGNO (def);
@@ -857,7 +848,7 @@ df_lr_bb_local_compute (unsigned int bb_index)
   /* Process the hardware registers that are always live.  */
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
   /* Process the hardware registers that are always live.  */
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
-      struct df_ref *use = *use_rec;
+      df_ref use = *use_rec;
       /* Add use to set of uses in this BB.  */
       if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
        bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
       /* Add use to set of uses in this BB.  */
       if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
        bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
@@ -867,12 +858,12 @@ df_lr_bb_local_compute (unsigned int bb_index)
     {
       unsigned int uid = INSN_UID (insn);
 
     {
       unsigned int uid = INSN_UID (insn);
 
-      if (!INSN_P (insn))
-       continue;       
+      if (!NONDEBUG_INSN_P (insn))
+       continue;
 
       for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
        {
 
       for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
        {
-         struct df_ref *def = *def_rec;
+         df_ref def = *def_rec;
          /* If the def is to only part of the reg, it does
             not kill the other defs that reach here.  */
          if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
          /* If the def is to only part of the reg, it does
             not kill the other defs that reach here.  */
          if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
@@ -885,7 +876,7 @@ df_lr_bb_local_compute (unsigned int bb_index)
 
       for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
        {
 
       for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
        {
-         struct df_ref *use = *use_rec;
+         df_ref use = *use_rec;
          /* Add use to set of uses in this BB.  */
          bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
        }
          /* Add use to set of uses in this BB.  */
          bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
        }
@@ -896,7 +887,7 @@ df_lr_bb_local_compute (unsigned int bb_index)
      goto.  */
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
      goto.  */
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
-      struct df_ref *def = *def_rec;
+      df_ref def = *def_rec;
       if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
        {
          unsigned int dregno = DF_REF_REGNO (def);
       if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
        {
          unsigned int dregno = DF_REF_REGNO (def);
@@ -904,12 +895,12 @@ df_lr_bb_local_compute (unsigned int bb_index)
          bitmap_clear_bit (bb_info->use, dregno);
        }
     }
          bitmap_clear_bit (bb_info->use, dregno);
        }
     }
-  
+
 #ifdef EH_USES
   /* Process the uses that are live into an exception handler.  */
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
 #ifdef EH_USES
   /* Process the uses that are live into an exception handler.  */
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
-      struct df_ref *use = *use_rec;
+      df_ref use = *use_rec;
       /* Add use to set of uses in this BB.  */
       if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
        bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
       /* Add use to set of uses in this BB.  */
       if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
        bitmap_set_bit (bb_info->use, DF_REF_REGNO (use));
@@ -932,12 +923,12 @@ df_lr_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 {
   unsigned int bb_index;
   bitmap_iterator bi;
-    
+
   bitmap_clear (df->hardware_regs_used);
   bitmap_clear (df->hardware_regs_used);
-  
+
   /* The all-important stack pointer must always be live.  */
   bitmap_set_bit (df->hardware_regs_used, STACK_POINTER_REGNUM);
   /* The all-important stack pointer must always be live.  */
   bitmap_set_bit (df->hardware_regs_used, STACK_POINTER_REGNUM);
-  
+
   /* Before reload, there are a few registers that must be forced
      live everywhere -- which might not already be the case for
      blocks within infinite loops.  */
   /* Before reload, there are a few registers that must be forced
      live everywhere -- which might not already be the case for
      blocks within infinite loops.  */
@@ -946,21 +937,21 @@ df_lr_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
       /* Any reference to any pseudo before reload is a potential
         reference of the frame pointer.  */
       bitmap_set_bit (df->hardware_regs_used, FRAME_POINTER_REGNUM);
       /* Any reference to any pseudo before reload is a potential
         reference of the frame pointer.  */
       bitmap_set_bit (df->hardware_regs_used, FRAME_POINTER_REGNUM);
-      
+
 #if FRAME_POINTER_REGNUM != ARG_POINTER_REGNUM
       /* Pseudos with argument area equivalences may require
         reloading via the argument pointer.  */
       if (fixed_regs[ARG_POINTER_REGNUM])
        bitmap_set_bit (df->hardware_regs_used, ARG_POINTER_REGNUM);
 #endif
 #if FRAME_POINTER_REGNUM != ARG_POINTER_REGNUM
       /* Pseudos with argument area equivalences may require
         reloading via the argument pointer.  */
       if (fixed_regs[ARG_POINTER_REGNUM])
        bitmap_set_bit (df->hardware_regs_used, ARG_POINTER_REGNUM);
 #endif
-      
+
       /* Any constant, or pseudo with constant equivalences, may
         require reloading from memory using the pic register.  */
       if ((unsigned) PIC_OFFSET_TABLE_REGNUM != INVALID_REGNUM
          && fixed_regs[PIC_OFFSET_TABLE_REGNUM])
        bitmap_set_bit (df->hardware_regs_used, PIC_OFFSET_TABLE_REGNUM);
     }
       /* Any constant, or pseudo with constant equivalences, may
         require reloading from memory using the pic register.  */
       if ((unsigned) PIC_OFFSET_TABLE_REGNUM != INVALID_REGNUM
          && fixed_regs[PIC_OFFSET_TABLE_REGNUM])
        bitmap_set_bit (df->hardware_regs_used, PIC_OFFSET_TABLE_REGNUM);
     }
-  
+
   EXECUTE_IF_SET_IN_BITMAP (df_lr->out_of_date_transfer_functions, 0, bb_index, bi)
     {
       if (bb_index == EXIT_BLOCK)
   EXECUTE_IF_SET_IN_BITMAP (df_lr->out_of_date_transfer_functions, 0, bb_index, bi)
     {
       if (bb_index == EXIT_BLOCK)
@@ -980,7 +971,7 @@ df_lr_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
 
 /* Initialize the solution vectors.  */
 
 
 /* Initialize the solution vectors.  */
 
-static void 
+static void
 df_lr_init (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_lr_init (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -1004,7 +995,7 @@ df_lr_confluence_0 (basic_block bb)
   bitmap op1 = df_lr_get_bb_info (bb->index)->out;
   if (bb != EXIT_BLOCK_PTR)
     bitmap_copy (op1, df->hardware_regs_used);
   bitmap op1 = df_lr_get_bb_info (bb->index)->out;
   if (bb != EXIT_BLOCK_PTR)
     bitmap_copy (op1, df->hardware_regs_used);
-} 
+}
 
 
 /* Confluence function that ignores fake edges.  */
 
 
 /* Confluence function that ignores fake edges.  */
@@ -1014,17 +1005,17 @@ df_lr_confluence_n (edge e)
 {
   bitmap op1 = df_lr_get_bb_info (e->src->index)->out;
   bitmap op2 = df_lr_get_bb_info (e->dest->index)->in;
 {
   bitmap op1 = df_lr_get_bb_info (e->src->index)->out;
   bitmap op2 = df_lr_get_bb_info (e->dest->index)->in;
+
   /* Call-clobbered registers die across exception and call edges.  */
   /* ??? Abnormal call edges ignored for the moment, as this gets
      confused by sibling call edges, which crashes reg-stack.  */
   if (e->flags & EDGE_EH)
   /* Call-clobbered registers die across exception and call edges.  */
   /* ??? Abnormal call edges ignored for the moment, as this gets
      confused by sibling call edges, which crashes reg-stack.  */
   if (e->flags & EDGE_EH)
-    bitmap_ior_and_compl_into (op1, op2, df_invalidated_by_call);
+    bitmap_ior_and_compl_into (op1, op2, regs_invalidated_by_call_regset);
   else
     bitmap_ior_into (op1, op2);
 
   bitmap_ior_into (op1, df->hardware_regs_used);
   else
     bitmap_ior_into (op1, op2);
 
   bitmap_ior_into (op1, df->hardware_regs_used);
-} 
+}
 
 
 /* Transfer function.  */
 
 
 /* Transfer function.  */
@@ -1045,25 +1036,34 @@ df_lr_transfer_function (int bb_index)
 /* Run the fast dce as a side effect of building LR.  */
 
 static void
 /* Run the fast dce as a side effect of building LR.  */
 
 static void
-df_lr_finalize (bitmap all_blocks ATTRIBUTE_UNUSED)
+df_lr_finalize (bitmap all_blocks)
 {
 {
+  df_lr->solutions_dirty = false;
   if (df->changeable_flags & DF_LR_RUN_DCE)
     {
       run_fast_df_dce ();
   if (df->changeable_flags & DF_LR_RUN_DCE)
     {
       run_fast_df_dce ();
-      if (df_lr->problem_data && df_lr->solutions_dirty)
+
+      /* If dce deletes some instructions, we need to recompute the lr
+        solution before proceeding further.  The problem is that fast
+        dce is a pessimestic dataflow algorithm.  In the case where
+        it deletes a statement S inside of a loop, the uses inside of
+        S may not be deleted from the dataflow solution because they
+        were carried around the loop.  While it is conservatively
+        correct to leave these extra bits, the standards of df
+        require that we maintain the best possible (least fixed
+        point) solution.  The only way to do that is to redo the
+        iteration from the beginning.  See PR35805 for an
+        example.  */
+      if (df_lr->solutions_dirty)
        {
        {
-         /* If we are here, then it is because we are both verifying
-         the solution and the dce changed the function.  In that case
-         the verification info built will be wrong.  So we leave the
-         dirty flag true so that the verifier will skip the checking
-         part and just clean up.*/
-         df_lr->solutions_dirty = true;
+         df_clear_flags (DF_LR_RUN_DCE);
+         df_lr_alloc (all_blocks);
+         df_lr_local_compute (all_blocks);
+         df_worklist_dataflow (df_lr, all_blocks, df->postorder, df->n_blocks);
+         df_lr_finalize (all_blocks);
+         df_set_flags (DF_LR_RUN_DCE);
        }
        }
-      else
-       df_lr->solutions_dirty = false;
     }
     }
-  else
-    df_lr->solutions_dirty = false;
 }
 
 
 }
 
 
@@ -1087,7 +1087,7 @@ df_lr_free (void)
            }
        }
       free_alloc_pool (df_lr->block_pool);
            }
        }
       free_alloc_pool (df_lr->block_pool);
-      
+
       df_lr->block_info_size = 0;
       free (df_lr->block_info);
     }
       df_lr->block_info_size = 0;
       free (df_lr->block_info);
     }
@@ -1106,7 +1106,7 @@ df_lr_top_dump (basic_block bb, FILE *file)
   struct df_lr_problem_data *problem_data;
   if (!bb_info || !bb_info->in)
     return;
   struct df_lr_problem_data *problem_data;
   if (!bb_info || !bb_info->in)
     return;
-      
+
   fprintf (file, ";; lr  in  \t");
   df_print_regset (file, bb_info->in);
   if (df_lr->problem_data)
   fprintf (file, ";; lr  in  \t");
   df_print_regset (file, bb_info->in);
   if (df_lr->problem_data)
@@ -1119,7 +1119,7 @@ df_lr_top_dump (basic_block bb, FILE *file)
   df_print_regset (file, bb_info->use);
   fprintf (file, ";; lr  def \t");
   df_print_regset (file, bb_info->def);
   df_print_regset (file, bb_info->use);
   fprintf (file, ";; lr  def \t");
   df_print_regset (file, bb_info->def);
-}  
+}
 
 
 /* Debugging info at bottom of bb.  */
 
 
 /* Debugging info at bottom of bb.  */
@@ -1131,7 +1131,7 @@ df_lr_bottom_dump (basic_block bb, FILE *file)
   struct df_lr_problem_data *problem_data;
   if (!bb_info || !bb_info->out)
     return;
   struct df_lr_problem_data *problem_data;
   if (!bb_info || !bb_info->out)
     return;
-  
+
   fprintf (file, ";; lr  out \t");
   df_print_regset (file, bb_info->out);
   if (df_lr->problem_data)
   fprintf (file, ";; lr  out \t");
   df_print_regset (file, bb_info->out);
   if (df_lr->problem_data)
@@ -1140,7 +1140,7 @@ df_lr_bottom_dump (basic_block bb, FILE *file)
       fprintf (file, ";;  old out  \t");
       df_print_regset (file, problem_data->out[bb->index]);
     }
       fprintf (file, ";;  old out  \t");
       df_print_regset (file, problem_data->out[bb->index]);
     }
-}  
+}
 
 
 /* Build the datastructure to verify that the solution to the dataflow
 
 
 /* Build the datastructure to verify that the solution to the dataflow
@@ -1157,7 +1157,7 @@ df_lr_verify_solution_start (void)
       return;
     }
 
       return;
     }
 
-  /* Set it true so that the solution is recomputed.  */ 
+  /* Set it true so that the solution is recomputed.  */
   df_lr->solutions_dirty = true;
 
   problem_data = XNEW (struct df_lr_problem_data);
   df_lr->solutions_dirty = true;
 
   problem_data = XNEW (struct df_lr_problem_data);
@@ -1231,8 +1231,8 @@ static struct df_problem problem_LR =
   df_lr_local_compute,        /* Local compute function.  */
   df_lr_init,                 /* Init the solution specific data.  */
   df_worklist_dataflow,       /* Worklist solver.  */
   df_lr_local_compute,        /* Local compute function.  */
   df_lr_init,                 /* Init the solution specific data.  */
   df_worklist_dataflow,       /* Worklist solver.  */
-  df_lr_confluence_0,         /* Confluence operator 0.  */ 
-  df_lr_confluence_n,         /* Confluence operator n.  */ 
+  df_lr_confluence_0,         /* Confluence operator 0.  */
+  df_lr_confluence_n,         /* Confluence operator n.  */
   df_lr_transfer_function,    /* Transfer function.  */
   df_lr_finalize,             /* Finalize function.  */
   df_lr_free,                 /* Free all of the problem information.  */
   df_lr_transfer_function,    /* Transfer function.  */
   df_lr_finalize,             /* Finalize function.  */
   df_lr_free,                 /* Free all of the problem information.  */
@@ -1243,7 +1243,7 @@ static struct df_problem problem_LR =
   df_lr_verify_solution_start,/* Incremental solution verify start.  */
   df_lr_verify_solution_end,  /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
   df_lr_verify_solution_start,/* Incremental solution verify start.  */
   df_lr_verify_solution_end,  /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
-  TV_DF_LR,                   /* Timing variable.  */ 
+  TV_DF_LR,                   /* Timing variable.  */
   false                       /* Reset blocks on dropping out of blocks_to_analyze.  */
 };
 
   false                       /* Reset blocks on dropping out of blocks_to_analyze.  */
 };
 
@@ -1294,7 +1294,7 @@ df_lr_verify_transfer_functions (void)
          /* Make a copy of the transfer functions and then compute
             new ones to see if the transfer functions have
             changed.  */
          /* Make a copy of the transfer functions and then compute
             new ones to see if the transfer functions have
             changed.  */
-         if (!bitmap_bit_p (df_lr->out_of_date_transfer_functions, 
+         if (!bitmap_bit_p (df_lr->out_of_date_transfer_functions,
                             bb->index))
            {
              bitmap_copy (saved_def, bb_info->def);
                             bb->index))
            {
              bitmap_copy (saved_def, bb_info->def);
@@ -1312,7 +1312,7 @@ df_lr_verify_transfer_functions (void)
          /* If we do not have basic block info, the block must be in
             the list of dirty blocks or else some one has added a
             block behind our backs. */
          /* If we do not have basic block info, the block must be in
             the list of dirty blocks or else some one has added a
             block behind our backs. */
-         gcc_assert (bitmap_bit_p (df_lr->out_of_date_transfer_functions, 
+         gcc_assert (bitmap_bit_p (df_lr->out_of_date_transfer_functions,
                                    bb->index));
        }
       /* Make sure no one created a block without following
                                    bb->index));
        }
       /* Make sure no one created a block without following
@@ -1321,8 +1321,8 @@ df_lr_verify_transfer_functions (void)
     }
 
   /* Make sure there are no dirty bits in blocks that have been deleted.  */
     }
 
   /* Make sure there are no dirty bits in blocks that have been deleted.  */
-  gcc_assert (!bitmap_intersect_compl_p (df_lr->out_of_date_transfer_functions, 
-                                        all_blocks)); 
+  gcc_assert (!bitmap_intersect_compl_p (df_lr->out_of_date_transfer_functions,
+                                        all_blocks));
 
   BITMAP_FREE (saved_def);
   BITMAP_FREE (saved_use);
 
   BITMAP_FREE (saved_def);
   BITMAP_FREE (saved_use);
@@ -1349,7 +1349,7 @@ df_lr_verify_transfer_functions (void)
 
    Then, the in and out sets for the LIVE problem itself are computed.
    These are the logical AND of the IN and OUT sets from the LR problem
 
    Then, the in and out sets for the LIVE problem itself are computed.
    These are the logical AND of the IN and OUT sets from the LR problem
-   and the must-initialized problem. 
+   and the must-initialized problem.
 ----------------------------------------------------------------------------*/
 
 /* Private data used to verify the solution for this problem.  */
 ----------------------------------------------------------------------------*/
 
 /* Private data used to verify the solution for this problem.  */
@@ -1367,7 +1367,7 @@ static bitmap df_live_scratch;
 /* Set basic block info.  */
 
 static void
 /* Set basic block info.  */
 
 static void
-df_live_set_bb_info (unsigned int index, 
+df_live_set_bb_info (unsigned int index,
                   struct df_live_bb_info *bb_info)
 {
   gcc_assert (df_live);
                   struct df_live_bb_info *bb_info)
 {
   gcc_assert (df_live);
@@ -1379,7 +1379,7 @@ df_live_set_bb_info (unsigned int index,
 /* Free basic block info.  */
 
 static void
 /* Free basic block info.  */
 
 static void
-df_live_free_bb_info (basic_block bb ATTRIBUTE_UNUSED, 
+df_live_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
                    void *vbb_info)
 {
   struct df_live_bb_info *bb_info = (struct df_live_bb_info *) vbb_info;
                    void *vbb_info)
 {
   struct df_live_bb_info *bb_info = (struct df_live_bb_info *) vbb_info;
@@ -1397,14 +1397,14 @@ df_live_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
 /* Allocate or reset bitmaps for DF_LIVE blocks. The solution bits are
    not touched unless the block is new.  */
 
 /* Allocate or reset bitmaps for DF_LIVE blocks. The solution bits are
    not touched unless the block is new.  */
 
-static void 
+static void
 df_live_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 
   if (!df_live->block_pool)
 df_live_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 
   if (!df_live->block_pool)
-    df_live->block_pool = create_alloc_pool ("df_live_block pool", 
+    df_live->block_pool = create_alloc_pool ("df_live_block pool",
                                           sizeof (struct df_live_bb_info), 100);
   if (!df_live_scratch)
     df_live_scratch = BITMAP_ALLOC (NULL);
                                           sizeof (struct df_live_bb_info), 100);
   if (!df_live_scratch)
     df_live_scratch = BITMAP_ALLOC (NULL);
@@ -1415,12 +1415,12 @@ df_live_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
     {
       struct df_live_bb_info *bb_info = df_live_get_bb_info (bb_index);
       if (bb_info)
     {
       struct df_live_bb_info *bb_info = df_live_get_bb_info (bb_index);
       if (bb_info)
-       { 
+       {
          bitmap_clear (bb_info->kill);
          bitmap_clear (bb_info->gen);
        }
       else
          bitmap_clear (bb_info->kill);
          bitmap_clear (bb_info->gen);
        }
       else
-       { 
+       {
          bb_info = (struct df_live_bb_info *) pool_alloc (df_live->block_pool);
          df_live_set_bb_info (bb_index, bb_info);
          bb_info->kill = BITMAP_ALLOC (NULL);
          bb_info = (struct df_live_bb_info *) pool_alloc (df_live->block_pool);
          df_live_set_bb_info (bb_index, bb_info);
          bb_info->kill = BITMAP_ALLOC (NULL);
@@ -1435,7 +1435,7 @@ df_live_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 
 /* Reset the global solution for recalculation.  */
 
 
 /* Reset the global solution for recalculation.  */
 
-static void 
+static void
 df_live_reset (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_live_reset (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -1459,7 +1459,7 @@ df_live_bb_local_compute (unsigned int bb_index)
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_live_bb_info *bb_info = df_live_get_bb_info (bb_index);
   rtx insn;
   basic_block bb = BASIC_BLOCK (bb_index);
   struct df_live_bb_info *bb_info = df_live_get_bb_info (bb_index);
   rtx insn;
-  struct df_ref **def_rec;
+  df_ref *def_rec;
   int luid = 0;
 
   FOR_BB_INSNS (bb, insn)
   int luid = 0;
 
   FOR_BB_INSNS (bb, insn)
@@ -1472,17 +1472,17 @@ df_live_bb_local_compute (unsigned int bb_index)
       if (!insn_info)
        {
          gcc_assert (!INSN_P (insn));
       if (!insn_info)
        {
          gcc_assert (!INSN_P (insn));
-         df_insn_create_insn_record (insn);
+         insn_info = df_insn_create_insn_record (insn);
        }
 
        }
 
-      DF_INSN_LUID (insn) = luid;
+      DF_INSN_INFO_LUID (insn_info) = luid;
       if (!INSN_P (insn))
        continue;
 
       luid++;
       if (!INSN_P (insn))
        continue;
 
       luid++;
-      for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+      for (def_rec = DF_INSN_INFO_DEFS (insn_info); *def_rec; def_rec++)
        {
        {
-         struct df_ref *def = *def_rec;
+         df_ref def = *def_rec;
          unsigned int regno = DF_REF_REGNO (def);
 
          if (DF_REF_FLAGS_IS_SET (def,
          unsigned int regno = DF_REF_REGNO (def);
 
          if (DF_REF_FLAGS_IS_SET (def,
@@ -1501,7 +1501,7 @@ df_live_bb_local_compute (unsigned int bb_index)
 
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
 
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
-      struct df_ref *def = *def_rec;
+      df_ref def = *def_rec;
       bitmap_set_bit (bb_info->gen, DF_REF_REGNO (def));
     }
 }
       bitmap_set_bit (bb_info->gen, DF_REF_REGNO (def));
     }
 }
@@ -1517,7 +1517,7 @@ df_live_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
 
   df_grow_insn_info ();
 
 
   df_grow_insn_info ();
 
-  EXECUTE_IF_SET_IN_BITMAP (df_live->out_of_date_transfer_functions, 
+  EXECUTE_IF_SET_IN_BITMAP (df_live->out_of_date_transfer_functions,
                            0, bb_index, bi)
     {
       df_live_bb_local_compute (bb_index);
                            0, bb_index, bi)
     {
       df_live_bb_local_compute (bb_index);
@@ -1529,7 +1529,7 @@ df_live_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
 
 /* Initialize the solution vectors.  */
 
 
 /* Initialize the solution vectors.  */
 
-static void 
+static void
 df_live_init (bitmap all_blocks)
 {
   unsigned int bb_index;
 df_live_init (bitmap all_blocks)
 {
   unsigned int bb_index;
@@ -1554,12 +1554,12 @@ df_live_confluence_n (edge e)
 {
   bitmap op1 = df_live_get_bb_info (e->dest->index)->in;
   bitmap op2 = df_live_get_bb_info (e->src->index)->out;
 {
   bitmap op1 = df_live_get_bb_info (e->dest->index)->in;
   bitmap op2 = df_live_get_bb_info (e->src->index)->out;
-  if (e->flags & EDGE_FAKE) 
+
+  if (e->flags & EDGE_FAKE)
     return;
 
   bitmap_ior_into (op1, op2);
     return;
 
   bitmap_ior_into (op1, op2);
-} 
+}
 
 
 /* Transfer function for the forwards must-initialized problem.  */
 
 
 /* Transfer function for the forwards must-initialized problem.  */
@@ -1574,10 +1574,9 @@ df_live_transfer_function (int bb_index)
   bitmap gen = bb_info->gen;
   bitmap kill = bb_info->kill;
 
   bitmap gen = bb_info->gen;
   bitmap kill = bb_info->kill;
 
-  /* We need to use a scratch set here so that the value returned from
-     this function invocation properly reflects if the sets changed in
-     a significant way; i.e. not just because the lr set was anded
-     in.  */
+  /* We need to use a scratch set here so that the value returned from this
+     function invocation properly reflects whether the sets changed in a
+     significant way; i.e. not just because the lr set was anded in.  */
   bitmap_and (df_live_scratch, gen, bb_lr_info->out);
   /* No register may reach a location where it is not used.  Thus
      we trim the rr result to the places where it is used.  */
   bitmap_and (df_live_scratch, gen, bb_lr_info->out);
   /* No register may reach a location where it is not used.  Thus
      we trim the rr result to the places where it is used.  */
@@ -1602,13 +1601,13 @@ df_live_finalize (bitmap all_blocks)
        {
          struct df_lr_bb_info *bb_lr_info = df_lr_get_bb_info (bb_index);
          struct df_live_bb_info *bb_live_info = df_live_get_bb_info (bb_index);
        {
          struct df_lr_bb_info *bb_lr_info = df_lr_get_bb_info (bb_index);
          struct df_live_bb_info *bb_live_info = df_live_get_bb_info (bb_index);
-  
+
          /* No register may reach a location where it is not used.  Thus
             we trim the rr result to the places where it is used.  */
          bitmap_and_into (bb_live_info->in, bb_lr_info->in);
          bitmap_and_into (bb_live_info->out, bb_lr_info->out);
        }
          /* No register may reach a location where it is not used.  Thus
             we trim the rr result to the places where it is used.  */
          bitmap_and_into (bb_live_info->in, bb_lr_info->in);
          bitmap_and_into (bb_live_info->out, bb_lr_info->out);
        }
-      
+
       df_live->solutions_dirty = false;
     }
 }
       df_live->solutions_dirty = false;
     }
 }
@@ -1622,7 +1621,7 @@ df_live_free (void)
   if (df_live->block_info)
     {
       unsigned int i;
   if (df_live->block_info)
     {
       unsigned int i;
-      
+
       for (i = 0; i < df_live->block_info_size; i++)
        {
          struct df_live_bb_info *bb_info = df_live_get_bb_info (i);
       for (i = 0; i < df_live->block_info_size; i++)
        {
          struct df_live_bb_info *bb_info = df_live_get_bb_info (i);
@@ -1634,7 +1633,7 @@ df_live_free (void)
              BITMAP_FREE (bb_info->out);
            }
        }
              BITMAP_FREE (bb_info->out);
            }
        }
-      
+
       free_alloc_pool (df_live->block_pool);
       df_live->block_info_size = 0;
       free (df_live->block_info);
       free_alloc_pool (df_live->block_pool);
       df_live->block_info_size = 0;
       free (df_live->block_info);
@@ -1657,7 +1656,7 @@ df_live_top_dump (basic_block bb, FILE *file)
 
   if (!bb_info || !bb_info->in)
     return;
 
   if (!bb_info || !bb_info->in)
     return;
-      
+
   fprintf (file, ";; live  in  \t");
   df_print_regset (file, bb_info->in);
   if (df_live->problem_data)
   fprintf (file, ";; live  in  \t");
   df_print_regset (file, bb_info->in);
   if (df_live->problem_data)
@@ -1683,7 +1682,7 @@ df_live_bottom_dump (basic_block bb, FILE *file)
 
   if (!bb_info || !bb_info->out)
     return;
 
   if (!bb_info || !bb_info->out)
     return;
-      
+
   fprintf (file, ";; live  out \t");
   df_print_regset (file, bb_info->out);
   if (df_live->problem_data)
   fprintf (file, ";; live  out \t");
   df_print_regset (file, bb_info->out);
   if (df_live->problem_data)
@@ -1709,7 +1708,7 @@ df_live_verify_solution_start (void)
       return;
     }
 
       return;
     }
 
-  /* Set it true so that the solution is recomputed.  */ 
+  /* Set it true so that the solution is recomputed.  */
   df_live->solutions_dirty = true;
 
   problem_data = XNEW (struct df_live_problem_data);
   df_live->solutions_dirty = true;
 
   problem_data = XNEW (struct df_live_problem_data);
@@ -1778,8 +1777,8 @@ static struct df_problem problem_LIVE =
   df_live_local_compute,        /* Local compute function.  */
   df_live_init,                 /* Init the solution specific data.  */
   df_worklist_dataflow,         /* Worklist solver.  */
   df_live_local_compute,        /* Local compute function.  */
   df_live_init,                 /* Init the solution specific data.  */
   df_worklist_dataflow,         /* Worklist solver.  */
-  NULL,                         /* Confluence operator 0.  */ 
-  df_live_confluence_n,         /* Confluence operator n.  */ 
+  NULL,                         /* Confluence operator 0.  */
+  df_live_confluence_n,         /* Confluence operator n.  */
   df_live_transfer_function,    /* Transfer function.  */
   df_live_finalize,             /* Finalize function.  */
   df_live_free,                 /* Free all of the problem information.  */
   df_live_transfer_function,    /* Transfer function.  */
   df_live_finalize,             /* Finalize function.  */
   df_live_free,                 /* Free all of the problem information.  */
@@ -1817,7 +1816,7 @@ df_live_set_all_dirty (void)
 {
   basic_block bb;
   FOR_ALL_BB (bb)
 {
   basic_block bb;
   FOR_ALL_BB (bb)
-    bitmap_set_bit (df_live->out_of_date_transfer_functions, 
+    bitmap_set_bit (df_live->out_of_date_transfer_functions,
                    bb->index);
 }
 
                    bb->index);
 }
 
@@ -1852,7 +1851,7 @@ df_live_verify_transfer_functions (void)
          /* Make a copy of the transfer functions and then compute
             new ones to see if the transfer functions have
             changed.  */
          /* Make a copy of the transfer functions and then compute
             new ones to see if the transfer functions have
             changed.  */
-         if (!bitmap_bit_p (df_live->out_of_date_transfer_functions, 
+         if (!bitmap_bit_p (df_live->out_of_date_transfer_functions,
                             bb->index))
            {
              bitmap_copy (saved_gen, bb_info->gen);
                             bb->index))
            {
              bitmap_copy (saved_gen, bb_info->gen);
@@ -1870,7 +1869,7 @@ df_live_verify_transfer_functions (void)
          /* If we do not have basic block info, the block must be in
             the list of dirty blocks or else some one has added a
             block behind our backs. */
          /* If we do not have basic block info, the block must be in
             the list of dirty blocks or else some one has added a
             block behind our backs. */
-         gcc_assert (bitmap_bit_p (df_live->out_of_date_transfer_functions, 
+         gcc_assert (bitmap_bit_p (df_live->out_of_date_transfer_functions,
                                    bb->index));
        }
       /* Make sure no one created a block without following
                                    bb->index));
        }
       /* Make sure no one created a block without following
@@ -1879,8 +1878,8 @@ df_live_verify_transfer_functions (void)
     }
 
   /* Make sure there are no dirty bits in blocks that have been deleted.  */
     }
 
   /* Make sure there are no dirty bits in blocks that have been deleted.  */
-  gcc_assert (!bitmap_intersect_compl_p (df_live->out_of_date_transfer_functions, 
-                                        all_blocks)); 
+  gcc_assert (!bitmap_intersect_compl_p (df_live->out_of_date_transfer_functions,
+                                        all_blocks));
   BITMAP_FREE (saved_gen);
   BITMAP_FREE (saved_kill);
   BITMAP_FREE (all_blocks);
   BITMAP_FREE (saved_gen);
   BITMAP_FREE (saved_kill);
   BITMAP_FREE (all_blocks);
@@ -1902,11 +1901,11 @@ df_live_verify_transfer_functions (void)
 /* Create a du or ud chain from SRC to DST and link it into SRC.   */
 
 struct df_link *
 /* Create a du or ud chain from SRC to DST and link it into SRC.   */
 
 struct df_link *
-df_chain_create (struct df_ref *src, struct df_ref *dst)
+df_chain_create (df_ref src, df_ref dst)
 {
   struct df_link *head = DF_REF_CHAIN (src);
 {
   struct df_link *head = DF_REF_CHAIN (src);
-  struct df_link *link = pool_alloc (df_chain->block_pool);;
-  
+  struct df_link *link = (struct df_link *) pool_alloc (df_chain->block_pool);
+
   DF_REF_CHAIN (src) = link;
   link->next = head;
   link->ref = dst;
   DF_REF_CHAIN (src) = link;
   link->next = head;
   link->ref = dst;
@@ -1915,9 +1914,9 @@ df_chain_create (struct df_ref *src, struct df_ref *dst)
 
 
 /* Delete any du or ud chains that start at REF and point to
 
 
 /* Delete any du or ud chains that start at REF and point to
-   TARGET.  */ 
+   TARGET.  */
 static void
 static void
-df_chain_unlink_1 (struct df_ref *ref, struct df_ref *target)
+df_chain_unlink_1 (df_ref ref, df_ref target)
 {
   struct df_link *chain = DF_REF_CHAIN (ref);
   struct df_link *prev = NULL;
 {
   struct df_link *chain = DF_REF_CHAIN (ref);
   struct df_link *prev = NULL;
@@ -1942,7 +1941,7 @@ df_chain_unlink_1 (struct df_ref *ref, struct df_ref *target)
 /* Delete a du or ud chain that leave or point to REF.  */
 
 void
 /* Delete a du or ud chain that leave or point to REF.  */
 
 void
-df_chain_unlink (struct df_ref *ref)
+df_chain_unlink (df_ref ref)
 {
   struct df_link *chain = DF_REF_CHAIN (ref);
   while (chain)
 {
   struct df_link *chain = DF_REF_CHAIN (ref);
   while (chain)
@@ -1958,10 +1957,10 @@ df_chain_unlink (struct df_ref *ref)
 
 
 /* Copy the du or ud chain starting at FROM_REF and attach it to
 
 
 /* Copy the du or ud chain starting at FROM_REF and attach it to
-   TO_REF.  */ 
+   TO_REF.  */
 
 
-void 
-df_chain_copy (struct df_ref *to_ref, 
+void
+df_chain_copy (df_ref to_ref,
               struct df_link *from_ref)
 {
   while (from_ref)
               struct df_link *from_ref)
 {
   while (from_ref)
@@ -1980,15 +1979,15 @@ df_chain_remove_problem (void)
   bitmap_iterator bi;
   unsigned int bb_index;
 
   bitmap_iterator bi;
   unsigned int bb_index;
 
-  /* Wholesale destruction of the old chains.  */ 
+  /* Wholesale destruction of the old chains.  */
   if (df_chain->block_pool)
     free_alloc_pool (df_chain->block_pool);
 
   EXECUTE_IF_SET_IN_BITMAP (df_chain->out_of_date_transfer_functions, 0, bb_index, bi)
     {
       rtx insn;
   if (df_chain->block_pool)
     free_alloc_pool (df_chain->block_pool);
 
   EXECUTE_IF_SET_IN_BITMAP (df_chain->out_of_date_transfer_functions, 0, bb_index, bi)
     {
       rtx insn;
-      struct df_ref **def_rec;
-      struct df_ref **use_rec;
+      df_ref *def_rec;
+      df_ref *use_rec;
       basic_block bb = BASIC_BLOCK (bb_index);
 
       if (df_chain_problem_p (DF_DU_CHAIN))
       basic_block bb = BASIC_BLOCK (bb_index);
 
       if (df_chain_problem_p (DF_DU_CHAIN))
@@ -1997,11 +1996,11 @@ df_chain_remove_problem (void)
       if (df_chain_problem_p (DF_UD_CHAIN))
        for (use_rec = df_get_artificial_uses (bb->index); *use_rec; use_rec++)
          DF_REF_CHAIN (*use_rec) = NULL;
       if (df_chain_problem_p (DF_UD_CHAIN))
        for (use_rec = df_get_artificial_uses (bb->index); *use_rec; use_rec++)
          DF_REF_CHAIN (*use_rec) = NULL;
-      
+
       FOR_BB_INSNS (bb, insn)
        {
          unsigned int uid = INSN_UID (insn);
       FOR_BB_INSNS (bb, insn)
        {
          unsigned int uid = INSN_UID (insn);
-         
+
          if (INSN_P (insn))
            {
              if (df_chain_problem_p (DF_DU_CHAIN))
          if (INSN_P (insn))
            {
              if (df_chain_problem_p (DF_DU_CHAIN))
@@ -2036,11 +2035,11 @@ df_chain_fully_remove_problem (void)
 
 /* Create def-use or use-def chains.  */
 
 
 /* Create def-use or use-def chains.  */
 
-static void  
+static void
 df_chain_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   df_chain_remove_problem ();
 df_chain_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
 {
   df_chain_remove_problem ();
-  df_chain->block_pool = create_alloc_pool ("df_chain_block pool", 
+  df_chain->block_pool = create_alloc_pool ("df_chain_block pool",
                                         sizeof (struct df_link), 50);
   df_chain->optional_p = true;
 }
                                         sizeof (struct df_link), 50);
   df_chain->optional_p = true;
 }
@@ -2059,15 +2058,15 @@ df_chain_reset (bitmap blocks_to_clear ATTRIBUTE_UNUSED)
 
 static void
 df_chain_create_bb_process_use (bitmap local_rd,
 
 static void
 df_chain_create_bb_process_use (bitmap local_rd,
-                               struct df_ref **use_rec,
-                               enum df_ref_flags top_flag)
+                               df_ref *use_rec,
+                               int top_flag)
 {
   bitmap_iterator bi;
   unsigned int def_index;
 {
   bitmap_iterator bi;
   unsigned int def_index;
-  
+
   while (*use_rec)
     {
   while (*use_rec)
     {
-      struct df_ref *use = *use_rec;
+      df_ref use = *use_rec;
       unsigned int uregno = DF_REF_REGNO (use);
       if ((!(df->changeable_flags & DF_NO_HARD_REGS))
          || (uregno >= FIRST_PSEUDO_REGISTER))
       unsigned int uregno = DF_REF_REGNO (use);
       if ((!(df->changeable_flags & DF_NO_HARD_REGS))
          || (uregno >= FIRST_PSEUDO_REGISTER))
@@ -2080,13 +2079,13 @@ df_chain_create_bb_process_use (bitmap local_rd,
                {
                  unsigned int first_index = DF_DEFS_BEGIN (uregno);
                  unsigned int last_index = first_index + count - 1;
                {
                  unsigned int first_index = DF_DEFS_BEGIN (uregno);
                  unsigned int last_index = first_index + count - 1;
-                 
+
                  EXECUTE_IF_SET_IN_BITMAP (local_rd, first_index, def_index, bi)
                    {
                  EXECUTE_IF_SET_IN_BITMAP (local_rd, first_index, def_index, bi)
                    {
-                     struct df_ref *def;
-                     if (def_index > last_index) 
+                     df_ref def;
+                     if (def_index > last_index)
                        break;
                        break;
-                     
+
                      def = DF_DEFS_GET (def_index);
                      if (df_chain_problem_p (DF_DU_CHAIN))
                        df_chain_create (def, use);
                      def = DF_DEFS_GET (def_index);
                      if (df_chain_problem_p (DF_DU_CHAIN))
                        df_chain_create (def, use);
@@ -2111,7 +2110,6 @@ df_chain_create_bb (unsigned int bb_index)
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
   rtx insn;
   bitmap cpy = BITMAP_ALLOC (NULL);
   struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
   rtx insn;
   bitmap cpy = BITMAP_ALLOC (NULL);
-  struct df_ref **def_rec;
 
   bitmap_copy (cpy, bb_info->in);
   bitmap_set_bit (df_chain->out_of_date_transfer_functions, bb_index);
 
   bitmap_copy (cpy, bb_info->in);
   bitmap_set_bit (df_chain->out_of_date_transfer_functions, bb_index);
@@ -2122,71 +2120,37 @@ df_chain_create_bb (unsigned int bb_index)
 #ifdef EH_USES
   /* Create the chains for the artificial uses from the EH_USES at the
      beginning of the block.  */
 #ifdef EH_USES
   /* Create the chains for the artificial uses from the EH_USES at the
      beginning of the block.  */
-  
+
   /* Artificials are only hard regs.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_chain_create_bb_process_use (cpy,
   /* Artificials are only hard regs.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_chain_create_bb_process_use (cpy,
-                                   df_get_artificial_uses (bb->index), 
+                                   df_get_artificial_uses (bb->index),
                                    DF_REF_AT_TOP);
 #endif
 
                                    DF_REF_AT_TOP);
 #endif
 
-  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
-    {
-      struct df_ref *def = *def_rec;
-      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
-       {
-         unsigned int dregno = DF_REF_REGNO (def);
-         if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
-           bitmap_clear_range (cpy, 
-                               DF_DEFS_BEGIN (dregno), 
-                               DF_DEFS_COUNT (dregno));
-         bitmap_set_bit (cpy, DF_REF_ID (def));
-       }
-    }
-  
+  df_rd_simulate_artificial_defs_at_top (bb, cpy);
+
   /* Process the regular instructions next.  */
   FOR_BB_INSNS (bb, insn)
   /* Process the regular instructions next.  */
   FOR_BB_INSNS (bb, insn)
-    {
-      struct df_ref **def_rec;
-      unsigned int uid = INSN_UID (insn);
-
-      if (!INSN_P (insn))
-       continue;
-
-      /* Now scan the uses and link them up with the defs that remain
-        in the cpy vector.  */
-      
-      df_chain_create_bb_process_use (cpy, DF_INSN_UID_USES (uid), 0);
-
-      if (df->changeable_flags & DF_EQ_NOTES)
-       df_chain_create_bb_process_use (cpy, DF_INSN_UID_EQ_USES (uid), 0);
+    if (INSN_P (insn))
+      {
+        unsigned int uid = INSN_UID (insn);
 
 
+        /* First scan the uses and link them up with the defs that remain
+          in the cpy vector.  */
+        df_chain_create_bb_process_use (cpy, DF_INSN_UID_USES (uid), 0);
+        if (df->changeable_flags & DF_EQ_NOTES)
+         df_chain_create_bb_process_use (cpy, DF_INSN_UID_EQ_USES (uid), 0);
 
 
-      /* Since we are going forwards, process the defs second.  This
-         pass only changes the bits in cpy.  */
-      for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
-       {
-         struct df_ref *def = *def_rec;
-         unsigned int dregno = DF_REF_REGNO (def);
-         if ((!(df->changeable_flags & DF_NO_HARD_REGS))
-             || (dregno >= FIRST_PSEUDO_REGISTER))
-           {
-             if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
-               bitmap_clear_range (cpy, 
-                                   DF_DEFS_BEGIN (dregno), 
-                                   DF_DEFS_COUNT (dregno));
-             if (!(DF_REF_FLAGS (def) 
-                   & (DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER)))
-               bitmap_set_bit (cpy, DF_REF_ID (def));
-           }
-       }
-    }
+        /* Since we are going forwards, process the defs second.  */
+        df_rd_simulate_one_insn (bb, insn, cpy);
+      }
 
   /* Create the chains for the artificial uses of the hard registers
      at the end of the block.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_chain_create_bb_process_use (cpy,
 
   /* Create the chains for the artificial uses of the hard registers
      at the end of the block.  */
   if (!(df->changeable_flags & DF_NO_HARD_REGS))
     df_chain_create_bb_process_use (cpy,
-                                   df_get_artificial_uses (bb->index), 
+                                   df_get_artificial_uses (bb->index),
                                    0);
 
   BITMAP_FREE (cpy);
                                    0);
 
   BITMAP_FREE (cpy);
@@ -2200,7 +2164,7 @@ df_chain_finalize (bitmap all_blocks)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 {
   unsigned int bb_index;
   bitmap_iterator bi;
-  
+
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
       df_chain_create_bb (bb_index);
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
       df_chain_create_bb (bb_index);
@@ -2227,37 +2191,37 @@ df_chain_top_dump (basic_block bb, FILE *file)
   if (df_chain_problem_p (DF_DU_CHAIN))
     {
       rtx insn;
   if (df_chain_problem_p (DF_DU_CHAIN))
     {
       rtx insn;
-      struct df_ref **def_rec = df_get_artificial_defs (bb->index);
+      df_ref *def_rec = df_get_artificial_defs (bb->index);
       if (*def_rec)
        {
       if (*def_rec)
        {
-         
+
          fprintf (file, ";;  DU chains for artificial defs\n");
          while (*def_rec)
            {
          fprintf (file, ";;  DU chains for artificial defs\n");
          while (*def_rec)
            {
-             struct df_ref *def = *def_rec;
+             df_ref def = *def_rec;
              fprintf (file, ";;   reg %d ", DF_REF_REGNO (def));
              df_chain_dump (DF_REF_CHAIN (def), file);
              fprintf (file, "\n");
              def_rec++;
            }
              fprintf (file, ";;   reg %d ", DF_REF_REGNO (def));
              df_chain_dump (DF_REF_CHAIN (def), file);
              fprintf (file, "\n");
              def_rec++;
            }
-       }      
+       }
 
       FOR_BB_INSNS (bb, insn)
        {
 
       FOR_BB_INSNS (bb, insn)
        {
-         unsigned int uid = INSN_UID (insn);
          if (INSN_P (insn))
            {
          if (INSN_P (insn))
            {
-             def_rec = DF_INSN_UID_DEFS (uid);
+             struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
+             def_rec = DF_INSN_INFO_DEFS (insn_info);
              if (*def_rec)
                {
              if (*def_rec)
                {
-                 fprintf (file, ";;   DU chains for insn luid %d uid %d\n", 
-                          DF_INSN_LUID (insn), uid);
-                 
+                 fprintf (file, ";;   DU chains for insn luid %d uid %d\n",
+                          DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
+
                  while (*def_rec)
                    {
                  while (*def_rec)
                    {
-                     struct df_ref *def = *def_rec;
+                     df_ref def = *def_rec;
                      fprintf (file, ";;      reg %d ", DF_REF_REGNO (def));
                      fprintf (file, ";;      reg %d ", DF_REF_REGNO (def));
-                     if (def->flags & DF_REF_READ_WRITE)
+                     if (DF_REF_FLAGS (def) & DF_REF_READ_WRITE)
                        fprintf (file, "read/write ");
                      df_chain_dump (DF_REF_CHAIN (def), file);
                      fprintf (file, "\n");
                        fprintf (file, "read/write ");
                      df_chain_dump (DF_REF_CHAIN (def), file);
                      fprintf (file, "\n");
@@ -2276,38 +2240,38 @@ df_chain_bottom_dump (basic_block bb, FILE *file)
   if (df_chain_problem_p (DF_UD_CHAIN))
     {
       rtx insn;
   if (df_chain_problem_p (DF_UD_CHAIN))
     {
       rtx insn;
-      struct df_ref **use_rec = df_get_artificial_uses (bb->index);
+      df_ref *use_rec = df_get_artificial_uses (bb->index);
 
       if (*use_rec)
        {
          fprintf (file, ";;  UD chains for artificial uses\n");
          while (*use_rec)
            {
 
       if (*use_rec)
        {
          fprintf (file, ";;  UD chains for artificial uses\n");
          while (*use_rec)
            {
-             struct df_ref *use = *use_rec;
+             df_ref use = *use_rec;
              fprintf (file, ";;   reg %d ", DF_REF_REGNO (use));
              df_chain_dump (DF_REF_CHAIN (use), file);
              fprintf (file, "\n");
              use_rec++;
            }
              fprintf (file, ";;   reg %d ", DF_REF_REGNO (use));
              df_chain_dump (DF_REF_CHAIN (use), file);
              fprintf (file, "\n");
              use_rec++;
            }
-       }      
+       }
 
       FOR_BB_INSNS (bb, insn)
        {
 
       FOR_BB_INSNS (bb, insn)
        {
-         unsigned int uid = INSN_UID (insn);
          if (INSN_P (insn))
            {
          if (INSN_P (insn))
            {
-             struct df_ref **eq_use_rec = DF_INSN_UID_EQ_USES (uid);
-             use_rec = DF_INSN_UID_USES (uid);
+             struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
+             df_ref *eq_use_rec = DF_INSN_INFO_EQ_USES (insn_info);
+             use_rec = DF_INSN_INFO_USES (insn_info);
              if (*use_rec || *eq_use_rec)
                {
              if (*use_rec || *eq_use_rec)
                {
-                 fprintf (file, ";;   UD chains for insn luid %d uid %d\n", 
-                          DF_INSN_LUID (insn), uid);
-                 
+                 fprintf (file, ";;   UD chains for insn luid %d uid %d\n",
+                          DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
+
                  while (*use_rec)
                    {
                  while (*use_rec)
                    {
-                     struct df_ref *use = *use_rec;
+                     df_ref use = *use_rec;
                      fprintf (file, ";;      reg %d ", DF_REF_REGNO (use));
                      fprintf (file, ";;      reg %d ", DF_REF_REGNO (use));
-                     if (use->flags & DF_REF_READ_WRITE)
+                     if (DF_REF_FLAGS (use) & DF_REF_READ_WRITE)
                        fprintf (file, "read/write ");
                      df_chain_dump (DF_REF_CHAIN (use), file);
                      fprintf (file, "\n");
                        fprintf (file, "read/write ");
                      df_chain_dump (DF_REF_CHAIN (use), file);
                      fprintf (file, "\n");
@@ -2315,7 +2279,7 @@ df_chain_bottom_dump (basic_block bb, FILE *file)
                    }
                  while (*eq_use_rec)
                    {
                    }
                  while (*eq_use_rec)
                    {
-                     struct df_ref *use = *eq_use_rec;
+                     df_ref use = *eq_use_rec;
                      fprintf (file, ";;   eq_note reg %d ", DF_REF_REGNO (use));
                      df_chain_dump (DF_REF_CHAIN (use), file);
                      fprintf (file, "\n");
                      fprintf (file, ";;   eq_note reg %d ", DF_REF_REGNO (use));
                      df_chain_dump (DF_REF_CHAIN (use), file);
                      fprintf (file, "\n");
@@ -2338,8 +2302,8 @@ static struct df_problem problem_CHAIN =
   NULL,                       /* Local compute function.  */
   NULL,                       /* Init the solution specific data.  */
   NULL,                       /* Iterative solver.  */
   NULL,                       /* Local compute function.  */
   NULL,                       /* Init the solution specific data.  */
   NULL,                       /* Iterative solver.  */
-  NULL,                       /* Confluence operator 0.  */ 
-  NULL,                       /* Confluence operator n.  */ 
+  NULL,                       /* Confluence operator 0.  */
+  NULL,                       /* Confluence operator n.  */
   NULL,                       /* Transfer function.  */
   df_chain_finalize,          /* Finalize function.  */
   df_chain_free,              /* Free all of the problem information.  */
   NULL,                       /* Transfer function.  */
   df_chain_finalize,          /* Finalize function.  */
   df_chain_free,              /* Free all of the problem information.  */
@@ -2360,10 +2324,10 @@ static struct df_problem problem_CHAIN =
    solution.  */
 
 void
    solution.  */
 
 void
-df_chain_add_problem (enum df_chain_flags chain_flags)
+df_chain_add_problem (unsigned int chain_flags)
 {
   df_add_problem (&problem_CHAIN);
 {
   df_add_problem (&problem_CHAIN);
-  df_chain->local_flags = (unsigned int)chain_flags;
+  df_chain->local_flags = chain_flags;
   df_chain->out_of_date_transfer_functions = BITMAP_ALLOC (NULL);
 }
 
   df_chain->out_of_date_transfer_functions = BITMAP_ALLOC (NULL);
 }
 
@@ -2371,785 +2335,2270 @@ df_chain_add_problem (enum df_chain_flags chain_flags)
 
 \f
 /*----------------------------------------------------------------------------
 
 \f
 /*----------------------------------------------------------------------------
-   This pass computes REG_DEAD and REG_UNUSED notes.
+   BYTE LEVEL LIVE REGISTERS
+
+   Find the locations in the function where any use of a pseudo can
+   reach in the backwards direction.  In and out bitvectors are built
+   for each basic block.  There are two mapping functions,
+   df_byte_lr_get_regno_start and df_byte_lr_get_regno_len that are
+   used to map regnos into bit vector positions.
+
+   This problem differs from the regular df_lr function in the way
+   that subregs, *_extracts and strict_low_parts are handled. In lr
+   these are consider partial kills, here, the exact set of bytes is
+   modeled.  Note that any reg that has none of these operations is
+   only modeled with a single bit since all operations access the
+   entire register.
+
+   This problem is more brittle that the regular lr.  It currently can
+   be used in dce incrementally, but cannot be used in an environment
+   where insns are created or modified.  The problem is that the
+   mapping of regnos to bitmap positions is relatively compact, in
+   that if a pseudo does not do any of the byte wise operations, only
+   one slot is allocated, rather than a slot for each byte.  If insn
+   are created, where a subreg is used for a reg that had no subregs,
+   the mapping would be wrong.  Likewise, there are no checks to see
+   that new pseudos have been added.  These issues could be addressed
+   by adding a problem specific flag to not use the compact mapping,
+   if there was a need to do so.
+
    ----------------------------------------------------------------------------*/
 
    ----------------------------------------------------------------------------*/
 
-static void 
-df_note_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
+/* Private data used to verify the solution for this problem.  */
+struct df_byte_lr_problem_data
 {
 {
-  df_note->optional_p = true;
+  /* Expanded versions of bitvectors used in lr.  */
+  bitmap invalidated_by_call;
+  bitmap hardware_regs_used;
+
+  /* Indexed by regno, this is true if there are subregs, extracts or
+     strict_low_parts for this regno.  */
+  bitmap needs_expansion;
+
+  /* The start position and len for each regno in the various bit
+     vectors.  */
+  unsigned int* regno_start;
+  unsigned int* regno_len;
+  /* An obstack for the bitmaps we need for this problem.  */
+  bitmap_obstack byte_lr_bitmaps;
+};
+
+
+/* Get the starting location for REGNO in the df_byte_lr bitmaps.  */
+
+int
+df_byte_lr_get_regno_start (unsigned int regno)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;;
+  return problem_data->regno_start[regno];
 }
 
 }
 
-#ifdef REG_DEAD_DEBUGGING
-static void 
-df_print_note (const char *prefix, rtx insn, rtx note)
+
+/* Get the len for REGNO in the df_byte_lr bitmaps.  */
+
+int
+df_byte_lr_get_regno_len (unsigned int regno)
 {
 {
-  if (dump_file)
-    {
-      fprintf (dump_file, "%s %d ", prefix, INSN_UID (insn));
-      print_rtl (dump_file, note);
-      fprintf (dump_file, "\n");
-    }
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;;
+  return problem_data->regno_len[regno];
 }
 }
-#endif
 
 
 
 
-/* After reg-stack, the x86 floating point stack regs are difficult to
-   analyze because of all of the pushes, pops and rotations.  Thus, we
-   just leave the notes alone. */
+/* Set basic block info.  */
 
 
-#ifdef STACK_REGS
-static inline bool 
-df_ignore_stack_reg (int regno)
+static void
+df_byte_lr_set_bb_info (unsigned int index,
+                       struct df_byte_lr_bb_info *bb_info)
 {
 {
-  return regstack_completed
-    && IN_RANGE (regno, FIRST_STACK_REG, LAST_STACK_REG);
+  gcc_assert (df_byte_lr);
+  gcc_assert (index < df_byte_lr->block_info_size);
+  df_byte_lr->block_info[index] = bb_info;
 }
 }
-#else
-static inline bool 
-df_ignore_stack_reg (int regno ATTRIBUTE_UNUSED)
+
+
+/* Free basic block info.  */
+
+static void
+df_byte_lr_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
+                        void *vbb_info)
 {
 {
-  return false;
+  struct df_byte_lr_bb_info *bb_info = (struct df_byte_lr_bb_info *) vbb_info;
+  if (bb_info)
+    {
+      BITMAP_FREE (bb_info->use);
+      BITMAP_FREE (bb_info->def);
+      BITMAP_FREE (bb_info->in);
+      BITMAP_FREE (bb_info->out);
+      pool_free (df_byte_lr->block_pool, bb_info);
+    }
 }
 }
-#endif
 
 
 
 
-/* Remove all of the REG_DEAD or REG_UNUSED notes from INSN and add
-   them to OLD_DEAD_NOTES and OLD_UNUSED_NOTES.  */
+/* Check all of the refs in REF_REC to see if any of them are
+   extracts, subregs or strict_low_parts.  */
 
 static void
 
 static void
-df_kill_notes (rtx insn, rtx *old_dead_notes, rtx *old_unused_notes)
+df_byte_lr_check_regs (df_ref *ref_rec)
 {
 {
-  rtx *pprev = &REG_NOTES (insn);
-  rtx link = *pprev;
-  rtx dead = NULL;
-  rtx unused = NULL;
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
 
 
-  while (link)
+  for (; *ref_rec; ref_rec++)
     {
     {
-      switch (REG_NOTE_KIND (link))
-       {
+      df_ref ref = *ref_rec;
+      if (DF_REF_FLAGS_IS_SET (ref, DF_REF_SIGN_EXTRACT
+                              | DF_REF_ZERO_EXTRACT
+                              | DF_REF_STRICT_LOW_PART)
+         || GET_CODE (DF_REF_REG (ref)) == SUBREG)
+       bitmap_set_bit (problem_data->needs_expansion, DF_REF_REGNO (ref));
+    }
+}
+
+
+/* Expand bitmap SRC which is indexed by regno to DEST which is indexed by
+   regno_start and regno_len.  */
+
+static void
+df_byte_lr_expand_bitmap (bitmap dest, bitmap src)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  bitmap_iterator bi;
+  unsigned int i;
+
+  bitmap_clear (dest);
+  EXECUTE_IF_SET_IN_BITMAP (src, 0, i, bi)
+    {
+      bitmap_set_range (dest, problem_data->regno_start[i],
+                       problem_data->regno_len[i]);
+    }
+}
+
+
+/* Allocate or reset bitmaps for DF_BYTE_LR blocks. The solution bits are
+   not touched unless the block is new.  */
+
+static void
+df_byte_lr_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
+{
+  unsigned int bb_index;
+  bitmap_iterator bi;
+  basic_block bb;
+  unsigned int regno;
+  unsigned int index = 0;
+  unsigned int max_reg = max_reg_num();
+  struct df_byte_lr_problem_data *problem_data
+    = XNEW (struct df_byte_lr_problem_data);
+
+  df_byte_lr->problem_data = problem_data;
+
+  if (!df_byte_lr->block_pool)
+    df_byte_lr->block_pool = create_alloc_pool ("df_byte_lr_block pool",
+                                          sizeof (struct df_byte_lr_bb_info), 50);
+
+  df_grow_bb_info (df_byte_lr);
+
+  /* Create the mapping from regnos to slots. This does not change
+     unless the problem is destroyed and recreated.  In particular, if
+     we end up deleting the only insn that used a subreg, we do not
+     want to redo the mapping because this would invalidate everything
+     else.  */
+
+  bitmap_obstack_initialize (&problem_data->byte_lr_bitmaps);
+  problem_data->regno_start = XNEWVEC (unsigned int, max_reg);
+  problem_data->regno_len = XNEWVEC (unsigned int, max_reg);
+  problem_data->hardware_regs_used = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+  problem_data->invalidated_by_call = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+  problem_data->needs_expansion = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+
+  /* Discover which regno's use subregs, extracts or
+     strict_low_parts.  */
+  FOR_EACH_BB (bb)
+    {
+      rtx insn;
+      FOR_BB_INSNS (bb, insn)
+       {
+         if (INSN_P (insn))
+           {
+             struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
+             df_byte_lr_check_regs (DF_INSN_INFO_DEFS (insn_info));
+             df_byte_lr_check_regs (DF_INSN_INFO_USES (insn_info));
+           }
+       }
+      bitmap_set_bit (df_byte_lr->out_of_date_transfer_functions, bb->index);
+    }
+
+  bitmap_set_bit (df_byte_lr->out_of_date_transfer_functions, ENTRY_BLOCK);
+  bitmap_set_bit (df_byte_lr->out_of_date_transfer_functions, EXIT_BLOCK);
+
+  /* Allocate the slots for each regno.  */
+  for (regno = 0; regno < max_reg; regno++)
+    {
+      int len;
+      problem_data->regno_start[regno] = index;
+      if (bitmap_bit_p (problem_data->needs_expansion, regno))
+       len = GET_MODE_SIZE (GET_MODE (regno_reg_rtx[regno]));
+      else
+       len = 1;
+
+      problem_data->regno_len[regno] = len;
+      index += len;
+    }
+
+  df_byte_lr_expand_bitmap (problem_data->hardware_regs_used,
+                           df->hardware_regs_used);
+  df_byte_lr_expand_bitmap (problem_data->invalidated_by_call,
+                           regs_invalidated_by_call_regset);
+
+  EXECUTE_IF_SET_IN_BITMAP (df_byte_lr->out_of_date_transfer_functions, 0, bb_index, bi)
+    {
+      struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb_index);
+      if (bb_info)
+       {
+         bitmap_clear (bb_info->def);
+         bitmap_clear (bb_info->use);
+       }
+      else
+       {
+         bb_info = (struct df_byte_lr_bb_info *) pool_alloc (df_byte_lr->block_pool);
+         df_byte_lr_set_bb_info (bb_index, bb_info);
+         bb_info->use = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+         bb_info->def = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+         bb_info->in = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+         bb_info->out = BITMAP_ALLOC (&problem_data->byte_lr_bitmaps);
+       }
+    }
+
+  df_byte_lr->optional_p = true;
+}
+
+
+/* Reset the global solution for recalculation.  */
+
+static void
+df_byte_lr_reset (bitmap all_blocks)
+{
+  unsigned int bb_index;
+  bitmap_iterator bi;
+
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
+    {
+      struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb_index);
+      gcc_assert (bb_info);
+      bitmap_clear (bb_info->in);
+      bitmap_clear (bb_info->out);
+    }
+}
+
+
+/* Compute local live register info for basic block BB.  */
+
+static void
+df_byte_lr_bb_local_compute (unsigned int bb_index)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  basic_block bb = BASIC_BLOCK (bb_index);
+  struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb_index);
+  rtx insn;
+  df_ref *def_rec;
+  df_ref *use_rec;
+
+  /* Process the registers set in an exception handler.  */
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         unsigned int start = problem_data->regno_start[dregno];
+         unsigned int len = problem_data->regno_len[dregno];
+         bitmap_set_range (bb_info->def, start, len);
+         bitmap_clear_range (bb_info->use, start, len);
+       }
+    }
+
+  /* Process the hardware registers that are always live.  */
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      /* Add use to set of uses in this BB.  */
+      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
+       {
+         unsigned int uregno = DF_REF_REGNO (use);
+         unsigned int start = problem_data->regno_start[uregno];
+         unsigned int len = problem_data->regno_len[uregno];
+         bitmap_set_range (bb_info->use, start, len);
+       }
+    }
+
+  FOR_BB_INSNS_REVERSE (bb, insn)
+    {
+      unsigned int uid = INSN_UID (insn);
+
+      if (!INSN_P (insn))
+       continue;
+
+      for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+       {
+         df_ref def = *def_rec;
+         /* If the def is to only part of the reg, it does
+            not kill the other defs that reach here.  */
+         if (!(DF_REF_FLAGS (def) & (DF_REF_CONDITIONAL)))
+           {
+             unsigned int dregno = DF_REF_REGNO (def);
+             unsigned int start = problem_data->regno_start[dregno];
+             unsigned int len = problem_data->regno_len[dregno];
+             unsigned int sb;
+             unsigned int lb;
+             if (!df_compute_accessed_bytes (def, DF_MM_MUST, &sb, &lb))
+               {
+                 start += sb;
+                 len = lb - sb;
+               }
+             if (len)
+               {
+                 bitmap_set_range (bb_info->def, start, len);
+                 bitmap_clear_range (bb_info->use, start, len);
+               }
+           }
+       }
+
+      for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
+       {
+         df_ref use = *use_rec;
+         unsigned int uregno = DF_REF_REGNO (use);
+         unsigned int start = problem_data->regno_start[uregno];
+         unsigned int len = problem_data->regno_len[uregno];
+         unsigned int sb;
+         unsigned int lb;
+         if (!df_compute_accessed_bytes (use, DF_MM_MAY, &sb, &lb))
+           {
+             start += sb;
+             len = lb - sb;
+           }
+         /* Add use to set of uses in this BB.  */
+         if (len)
+           bitmap_set_range (bb_info->use, start, len);
+       }
+    }
+
+  /* Process the registers set in an exception handler or the hard
+     frame pointer if this block is the target of a non local
+     goto.  */
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         unsigned int start = problem_data->regno_start[dregno];
+         unsigned int len = problem_data->regno_len[dregno];
+         bitmap_set_range (bb_info->def, start, len);
+         bitmap_clear_range (bb_info->use, start, len);
+       }
+    }
+
+#ifdef EH_USES
+  /* Process the uses that are live into an exception handler.  */
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      /* Add use to set of uses in this BB.  */
+      if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
+       {
+         unsigned int uregno = DF_REF_REGNO (use);
+         unsigned int start = problem_data->regno_start[uregno];
+         unsigned int len = problem_data->regno_len[uregno];
+         bitmap_set_range (bb_info->use, start, len);
+       }
+    }
+#endif
+}
+
+
+/* Compute local live register info for each basic block within BLOCKS.  */
+
+static void
+df_byte_lr_local_compute (bitmap all_blocks ATTRIBUTE_UNUSED)
+{
+  unsigned int bb_index;
+  bitmap_iterator bi;
+
+  EXECUTE_IF_SET_IN_BITMAP (df_byte_lr->out_of_date_transfer_functions, 0, bb_index, bi)
+    {
+      if (bb_index == EXIT_BLOCK)
+       {
+         /* The exit block is special for this problem and its bits are
+            computed from thin air.  */
+         struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (EXIT_BLOCK);
+         df_byte_lr_expand_bitmap (bb_info->use, df->exit_block_uses);
+       }
+      else
+       df_byte_lr_bb_local_compute (bb_index);
+    }
+
+  bitmap_clear (df_byte_lr->out_of_date_transfer_functions);
+}
+
+
+/* Initialize the solution vectors.  */
+
+static void
+df_byte_lr_init (bitmap all_blocks)
+{
+  unsigned int bb_index;
+  bitmap_iterator bi;
+
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
+    {
+      struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb_index);
+      bitmap_copy (bb_info->in, bb_info->use);
+      bitmap_clear (bb_info->out);
+    }
+}
+
+
+/* Confluence function that processes infinite loops.  This might be a
+   noreturn function that throws.  And even if it isn't, getting the
+   unwind info right helps debugging.  */
+static void
+df_byte_lr_confluence_0 (basic_block bb)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  bitmap op1 = df_byte_lr_get_bb_info (bb->index)->out;
+  if (bb != EXIT_BLOCK_PTR)
+    bitmap_copy (op1, problem_data->hardware_regs_used);
+}
+
+
+/* Confluence function that ignores fake edges.  */
+
+static void
+df_byte_lr_confluence_n (edge e)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  bitmap op1 = df_byte_lr_get_bb_info (e->src->index)->out;
+  bitmap op2 = df_byte_lr_get_bb_info (e->dest->index)->in;
+
+  /* Call-clobbered registers die across exception and call edges.  */
+  /* ??? Abnormal call edges ignored for the moment, as this gets
+     confused by sibling call edges, which crashes reg-stack.  */
+  if (e->flags & EDGE_EH)
+    bitmap_ior_and_compl_into (op1, op2, problem_data->invalidated_by_call);
+  else
+    bitmap_ior_into (op1, op2);
+
+  bitmap_ior_into (op1, problem_data->hardware_regs_used);
+}
+
+
+/* Transfer function.  */
+
+static bool
+df_byte_lr_transfer_function (int bb_index)
+{
+  struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb_index);
+  bitmap in = bb_info->in;
+  bitmap out = bb_info->out;
+  bitmap use = bb_info->use;
+  bitmap def = bb_info->def;
+
+  return bitmap_ior_and_compl (in, use, out, def);
+}
+
+
+/* Free all storage associated with the problem.  */
+
+static void
+df_byte_lr_free (void)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+
+
+  if (df_byte_lr->block_info)
+    {
+      free_alloc_pool (df_byte_lr->block_pool);
+      df_byte_lr->block_info_size = 0;
+      free (df_byte_lr->block_info);
+    }
+
+  BITMAP_FREE (df_byte_lr->out_of_date_transfer_functions);
+  bitmap_obstack_release (&problem_data->byte_lr_bitmaps);
+  free (problem_data->regno_start);
+  free (problem_data->regno_len);
+  free (problem_data);
+  free (df_byte_lr);
+}
+
+
+/* Debugging info at top of bb.  */
+
+static void
+df_byte_lr_top_dump (basic_block bb, FILE *file)
+{
+  struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb->index);
+  if (!bb_info || !bb_info->in)
+    return;
+
+  fprintf (file, ";; blr  in  \t");
+  df_print_byte_regset (file, bb_info->in);
+  fprintf (file, ";; blr  use \t");
+  df_print_byte_regset (file, bb_info->use);
+  fprintf (file, ";; blr  def \t");
+  df_print_byte_regset (file, bb_info->def);
+}
+
+
+/* Debugging info at bottom of bb.  */
+
+static void
+df_byte_lr_bottom_dump (basic_block bb, FILE *file)
+{
+  struct df_byte_lr_bb_info *bb_info = df_byte_lr_get_bb_info (bb->index);
+  if (!bb_info || !bb_info->out)
+    return;
+
+  fprintf (file, ";; blr  out \t");
+  df_print_byte_regset (file, bb_info->out);
+}
+
+
+/* All of the information associated with every instance of the problem.  */
+
+static struct df_problem problem_BYTE_LR =
+{
+  DF_BYTE_LR,                      /* Problem id.  */
+  DF_BACKWARD,                     /* Direction.  */
+  df_byte_lr_alloc,                /* Allocate the problem specific data.  */
+  df_byte_lr_reset,                /* Reset global information.  */
+  df_byte_lr_free_bb_info,         /* Free basic block info.  */
+  df_byte_lr_local_compute,        /* Local compute function.  */
+  df_byte_lr_init,                 /* Init the solution specific data.  */
+  df_worklist_dataflow,            /* Worklist solver.  */
+  df_byte_lr_confluence_0,         /* Confluence operator 0.  */
+  df_byte_lr_confluence_n,         /* Confluence operator n.  */
+  df_byte_lr_transfer_function,    /* Transfer function.  */
+  NULL,                            /* Finalize function.  */
+  df_byte_lr_free,                 /* Free all of the problem information.  */
+  df_byte_lr_free,                 /* Remove this problem from the stack of dataflow problems.  */
+  NULL,                            /* Debugging.  */
+  df_byte_lr_top_dump,             /* Debugging start block.  */
+  df_byte_lr_bottom_dump,          /* Debugging end block.  */
+  NULL,                            /* Incremental solution verify start.  */
+  NULL,                            /* Incremental solution verify end.  */
+  NULL,                            /* Dependent problem.  */
+  TV_DF_BYTE_LR,                   /* Timing variable.  */
+  false                            /* Reset blocks on dropping out of blocks_to_analyze.  */
+};
+
+
+/* Create a new DATAFLOW instance and add it to an existing instance
+   of DF.  The returned structure is what is used to get at the
+   solution.  */
+
+void
+df_byte_lr_add_problem (void)
+{
+  df_add_problem (&problem_BYTE_LR);
+  /* These will be initialized when df_scan_blocks processes each
+     block.  */
+  df_byte_lr->out_of_date_transfer_functions = BITMAP_ALLOC (NULL);
+}
+
+
+/* Simulate the effects of the defs of INSN on LIVE.  */
+
+void
+df_byte_lr_simulate_defs (rtx insn, bitmap live)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  df_ref *def_rec;
+  unsigned int uid = INSN_UID (insn);
+
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+
+      /* If the def is to only part of the reg, it does
+        not kill the other defs that reach here.  */
+      if (!(DF_REF_FLAGS (def) & DF_REF_CONDITIONAL))
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         unsigned int start = problem_data->regno_start[dregno];
+         unsigned int len = problem_data->regno_len[dregno];
+         unsigned int sb;
+         unsigned int lb;
+         if (!df_compute_accessed_bytes (def, DF_MM_MUST, &sb, &lb))
+           {
+             start += sb;
+             len = lb - sb;
+           }
+
+         if (len)
+           bitmap_clear_range (live, start, len);
+       }
+    }
+}
+
+
+/* Simulate the effects of the uses of INSN on LIVE.  */
+
+void
+df_byte_lr_simulate_uses (rtx insn, bitmap live)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  df_ref *use_rec;
+  unsigned int uid = INSN_UID (insn);
+
+  for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      unsigned int uregno = DF_REF_REGNO (use);
+      unsigned int start = problem_data->regno_start[uregno];
+      unsigned int len = problem_data->regno_len[uregno];
+      unsigned int sb;
+      unsigned int lb;
+
+      if (!df_compute_accessed_bytes (use, DF_MM_MAY, &sb, &lb))
+       {
+         start += sb;
+         len = lb - sb;
+       }
+
+      /* Add use to set of uses in this BB.  */
+      if (len)
+       bitmap_set_range (live, start, len);
+    }
+}
+
+
+/* Apply the artificial uses and defs at the top of BB in a forwards
+   direction.  */
+
+void
+df_byte_lr_simulate_artificial_refs_at_top (basic_block bb, bitmap live)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  df_ref *def_rec;
+#ifdef EH_USES
+  df_ref *use_rec;
+#endif
+  int bb_index = bb->index;
+
+#ifdef EH_USES
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
+       {
+         unsigned int uregno = DF_REF_REGNO (use);
+         unsigned int start = problem_data->regno_start[uregno];
+         unsigned int len = problem_data->regno_len[uregno];
+         bitmap_set_range (live, start, len);
+       }
+    }
+#endif
+
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         unsigned int start = problem_data->regno_start[dregno];
+         unsigned int len = problem_data->regno_len[dregno];
+         bitmap_clear_range (live, start, len);
+       }
+    }
+}
+
+
+/* Apply the artificial uses and defs at the end of BB in a backwards
+   direction.  */
+
+void
+df_byte_lr_simulate_artificial_refs_at_end (basic_block bb, bitmap live)
+{
+  struct df_byte_lr_problem_data *problem_data
+    = (struct df_byte_lr_problem_data *)df_byte_lr->problem_data;
+  df_ref *def_rec;
+  df_ref *use_rec;
+  int bb_index = bb->index;
+
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         unsigned int start = problem_data->regno_start[dregno];
+         unsigned int len = problem_data->regno_len[dregno];
+         bitmap_clear_range (live, start, len);
+       }
+    }
+
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
+       {
+         unsigned int uregno = DF_REF_REGNO (use);
+         unsigned int start = problem_data->regno_start[uregno];
+         unsigned int len = problem_data->regno_len[uregno];
+         bitmap_set_range (live, start, len);
+       }
+    }
+}
+
+
+\f
+/*----------------------------------------------------------------------------
+   This problem computes REG_DEAD and REG_UNUSED notes.
+   ----------------------------------------------------------------------------*/
+
+static void
+df_note_alloc (bitmap all_blocks ATTRIBUTE_UNUSED)
+{
+  df_note->optional_p = true;
+}
+
+#ifdef REG_DEAD_DEBUGGING
+static void
+df_print_note (const char *prefix, rtx insn, rtx note)
+{
+  if (dump_file)
+    {
+      fprintf (dump_file, "%s %d ", prefix, INSN_UID (insn));
+      print_rtl (dump_file, note);
+      fprintf (dump_file, "\n");
+    }
+}
+#endif
+
+
+/* After reg-stack, the x86 floating point stack regs are difficult to
+   analyze because of all of the pushes, pops and rotations.  Thus, we
+   just leave the notes alone. */
+
+#ifdef STACK_REGS
+static inline bool
+df_ignore_stack_reg (int regno)
+{
+  return regstack_completed
+    && IN_RANGE (regno, FIRST_STACK_REG, LAST_STACK_REG);
+}
+#else
+static inline bool
+df_ignore_stack_reg (int regno ATTRIBUTE_UNUSED)
+{
+  return false;
+}
+#endif
+
+
+/* Remove all of the REG_DEAD or REG_UNUSED notes from INSN and add
+   them to OLD_DEAD_NOTES and OLD_UNUSED_NOTES.  */
+
+static void
+df_kill_notes (rtx insn, rtx *old_dead_notes, rtx *old_unused_notes)
+{
+  rtx *pprev = &REG_NOTES (insn);
+  rtx link = *pprev;
+  rtx dead = NULL;
+  rtx unused = NULL;
+
+  while (link)
+    {
+      switch (REG_NOTE_KIND (link))
+       {
        case REG_DEAD:
        case REG_DEAD:
-         /* After reg-stack, we need to ignore any unused notes 
+         /* After reg-stack, we need to ignore any unused notes
+            for the stack registers.  */
+         if (df_ignore_stack_reg (REGNO (XEXP (link, 0))))
+           {
+             pprev = &XEXP (link, 1);
+             link = *pprev;
+           }
+         else
+           {
+             rtx next = XEXP (link, 1);
+#ifdef REG_DEAD_DEBUGGING
+             df_print_note ("deleting: ", insn, link);
+#endif
+             XEXP (link, 1) = dead;
+             dead = link;
+             *pprev = link = next;
+           }
+         break;
+
+       case REG_UNUSED:
+         /* After reg-stack, we need to ignore any unused notes
             for the stack registers.  */
          if (df_ignore_stack_reg (REGNO (XEXP (link, 0))))
            {
             for the stack registers.  */
          if (df_ignore_stack_reg (REGNO (XEXP (link, 0))))
            {
-             pprev = &XEXP (link, 1);
-             link = *pprev;
+             pprev = &XEXP (link, 1);
+             link = *pprev;
+           }
+         else
+           {
+             rtx next = XEXP (link, 1);
+#ifdef REG_DEAD_DEBUGGING
+             df_print_note ("deleting: ", insn, link);
+#endif
+             XEXP (link, 1) = unused;
+             unused = link;
+             *pprev = link = next;
+           }
+         break;
+
+       default:
+         pprev = &XEXP (link, 1);
+         link = *pprev;
+         break;
+       }
+    }
+
+  *old_dead_notes = dead;
+  *old_unused_notes = unused;
+}
+
+
+/* Set a NOTE_TYPE note for REG in INSN.  Try to pull it from the OLD
+   list, otherwise create a new one.  */
+
+static inline rtx
+df_set_note (enum reg_note note_type, rtx insn, rtx old, rtx reg)
+{
+  rtx curr = old;
+  rtx prev = NULL;
+
+  gcc_assert (!DEBUG_INSN_P (insn));
+
+  while (curr)
+    if (XEXP (curr, 0) == reg)
+      {
+       if (prev)
+         XEXP (prev, 1) = XEXP (curr, 1);
+       else
+         old = XEXP (curr, 1);
+       XEXP (curr, 1) = REG_NOTES (insn);
+       REG_NOTES (insn) = curr;
+       return old;
+      }
+    else
+      {
+       prev = curr;
+       curr = XEXP (curr, 1);
+      }
+
+  /* Did not find the note.  */
+  add_reg_note (insn, note_type, reg);
+  return old;
+}
+
+/* A subroutine of df_set_unused_notes_for_mw, with a selection of its
+   arguments.  Return true if the register value described by MWS's
+   mw_reg is known to be completely unused, and if mw_reg can therefore
+   be used in a REG_UNUSED note.  */
+
+static bool
+df_whole_mw_reg_unused_p (struct df_mw_hardreg *mws,
+                         bitmap live, bitmap artificial_uses)
+{
+  unsigned int r;
+
+  /* If MWS describes a partial reference, create REG_UNUSED notes for
+     individual hard registers.  */
+  if (mws->flags & DF_REF_PARTIAL)
+    return false;
+
+  /* Likewise if some part of the register is used.  */
+  for (r = mws->start_regno; r <= mws->end_regno; r++)
+    if (bitmap_bit_p (live, r)
+       || bitmap_bit_p (artificial_uses, r))
+      return false;
+
+  gcc_assert (REG_P (mws->mw_reg));
+  return true;
+}
+
+/* Set the REG_UNUSED notes for the multiword hardreg defs in INSN
+   based on the bits in LIVE.  Do not generate notes for registers in
+   artificial uses.  DO_NOT_GEN is updated so that REG_DEAD notes are
+   not generated if the reg is both read and written by the
+   instruction.
+*/
+
+static rtx
+df_set_unused_notes_for_mw (rtx insn, rtx old, struct df_mw_hardreg *mws,
+                           bitmap live, bitmap do_not_gen,
+                           bitmap artificial_uses)
+{
+  unsigned int r;
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    fprintf (dump_file, "mw_set_unused looking at mws[%d..%d]\n",
+            mws->start_regno, mws->end_regno);
+#endif
+
+  if (df_whole_mw_reg_unused_p (mws, live, artificial_uses))
+    {
+      unsigned int regno = mws->start_regno;
+      old = df_set_note (REG_UNUSED, insn, old, mws->mw_reg);
+
+#ifdef REG_DEAD_DEBUGGING
+      df_print_note ("adding 1: ", insn, REG_NOTES (insn));
+#endif
+      bitmap_set_bit (do_not_gen, regno);
+      /* Only do this if the value is totally dead.  */
+    }
+  else
+    for (r = mws->start_regno; r <= mws->end_regno; r++)
+      {
+       if (!bitmap_bit_p (live, r)
+           && !bitmap_bit_p (artificial_uses, r))
+         {
+           old = df_set_note (REG_UNUSED, insn, old, regno_reg_rtx[r]);
+#ifdef REG_DEAD_DEBUGGING
+           df_print_note ("adding 2: ", insn, REG_NOTES (insn));
+#endif
+         }
+       bitmap_set_bit (do_not_gen, r);
+      }
+  return old;
+}
+
+
+/* A subroutine of df_set_dead_notes_for_mw, with a selection of its
+   arguments.  Return true if the register value described by MWS's
+   mw_reg is known to be completely dead, and if mw_reg can therefore
+   be used in a REG_DEAD note.  */
+
+static bool
+df_whole_mw_reg_dead_p (struct df_mw_hardreg *mws,
+                       bitmap live, bitmap artificial_uses,
+                       bitmap do_not_gen)
+{
+  unsigned int r;
+
+  /* If MWS describes a partial reference, create REG_DEAD notes for
+     individual hard registers.  */
+  if (mws->flags & DF_REF_PARTIAL)
+    return false;
+
+  /* Likewise if some part of the register is not dead.  */
+  for (r = mws->start_regno; r <= mws->end_regno; r++)
+    if (bitmap_bit_p (live, r)
+       || bitmap_bit_p (artificial_uses, r)
+       || bitmap_bit_p (do_not_gen, r))
+      return false;
+
+  gcc_assert (REG_P (mws->mw_reg));
+  return true;
+}
+
+/* Set the REG_DEAD notes for the multiword hardreg use in INSN based
+   on the bits in LIVE.  DO_NOT_GEN is used to keep REG_DEAD notes
+   from being set if the instruction both reads and writes the
+   register.  */
+
+static rtx
+df_set_dead_notes_for_mw (rtx insn, rtx old, struct df_mw_hardreg *mws,
+                         bitmap live, bitmap do_not_gen,
+                         bitmap artificial_uses, bool *added_notes_p)
+{
+  unsigned int r;
+  bool is_debug = *added_notes_p;
+
+  *added_notes_p = false;
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    {
+      fprintf (dump_file, "mw_set_dead looking at mws[%d..%d]\n  do_not_gen =",
+              mws->start_regno, mws->end_regno);
+      df_print_regset (dump_file, do_not_gen);
+      fprintf (dump_file, "  live =");
+      df_print_regset (dump_file, live);
+      fprintf (dump_file, "  artificial uses =");
+      df_print_regset (dump_file, artificial_uses);
+    }
+#endif
+
+  if (df_whole_mw_reg_dead_p (mws, live, artificial_uses, do_not_gen))
+    {
+      /* Add a dead note for the entire multi word register.  */
+      if (is_debug)
+       {
+         *added_notes_p = true;
+         return old;
+       }
+      old = df_set_note (REG_DEAD, insn, old, mws->mw_reg);
+#ifdef REG_DEAD_DEBUGGING
+      df_print_note ("adding 1: ", insn, REG_NOTES (insn));
+#endif
+    }
+  else
+    {
+      for (r = mws->start_regno; r <= mws->end_regno; r++)
+       if (!bitmap_bit_p (live, r)
+           && !bitmap_bit_p (artificial_uses, r)
+           && !bitmap_bit_p (do_not_gen, r))
+         {
+           if (is_debug)
+             {
+               *added_notes_p = true;
+               return old;
+             }
+           old = df_set_note (REG_DEAD, insn, old, regno_reg_rtx[r]);
+#ifdef REG_DEAD_DEBUGGING
+           df_print_note ("adding 2: ", insn, REG_NOTES (insn));
+#endif
+         }
+    }
+  return old;
+}
+
+
+/* Create a REG_UNUSED note if necessary for DEF in INSN updating
+   LIVE.  Do not generate notes for registers in ARTIFICIAL_USES.  */
+
+static rtx
+df_create_unused_note (rtx insn, rtx old, df_ref def,
+                      bitmap live, bitmap artificial_uses)
+{
+  unsigned int dregno = DF_REF_REGNO (def);
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    {
+      fprintf (dump_file, "  regular looking at def ");
+      df_ref_debug (def, dump_file);
+    }
+#endif
+
+  if (!(bitmap_bit_p (live, dregno)
+       || (DF_REF_FLAGS (def) & DF_REF_MW_HARDREG)
+       || bitmap_bit_p (artificial_uses, dregno)
+       || df_ignore_stack_reg (dregno)))
+    {
+      rtx reg = (DF_REF_LOC (def))
+                ? *DF_REF_REAL_LOC (def): DF_REF_REG (def);
+      old = df_set_note (REG_UNUSED, insn, old, reg);
+#ifdef REG_DEAD_DEBUGGING
+      df_print_note ("adding 3: ", insn, REG_NOTES (insn));
+#endif
+    }
+
+  return old;
+}
+
+/* Node of a linked list of uses of dead REGs in debug insns.  */
+struct dead_debug_use
+{
+  df_ref use;
+  struct dead_debug_use *next;
+};
+
+/* Linked list of the above, with a bitmap of the REGs in the
+   list.  */
+struct dead_debug
+{
+  struct dead_debug_use *head;
+  bitmap used;
+  bitmap to_rescan;
+};
+
+/* Initialize DEBUG to an empty list, and clear USED, if given.  */
+static inline void
+dead_debug_init (struct dead_debug *debug, bitmap used)
+{
+  debug->head = NULL;
+  debug->used = used;
+  debug->to_rescan = NULL;
+  if (used)
+    bitmap_clear (used);
+}
+
+/* Reset all debug insns with pending uses.  Release the bitmap in it,
+   unless it is USED.  USED must be the same bitmap passed to
+   dead_debug_init.  */
+static inline void
+dead_debug_finish (struct dead_debug *debug, bitmap used)
+{
+  struct dead_debug_use *head;
+  rtx insn = NULL;
+
+  if (debug->used != used)
+    BITMAP_FREE (debug->used);
+
+  while ((head = debug->head))
+    {
+      insn = DF_REF_INSN (head->use);
+      if (!head->next || DF_REF_INSN (head->next->use) != insn)
+       {
+         INSN_VAR_LOCATION_LOC (insn) = gen_rtx_UNKNOWN_VAR_LOC ();
+         df_insn_rescan_debug_internal (insn);
+         if (debug->to_rescan)
+           bitmap_clear_bit (debug->to_rescan, INSN_UID (insn));
+       }
+      debug->head = head->next;
+      XDELETE (head);
+    }
+
+  if (debug->to_rescan)
+    {
+      bitmap_iterator bi;
+      unsigned int uid;
+
+      EXECUTE_IF_SET_IN_BITMAP (debug->to_rescan, 0, uid, bi)
+       {
+         struct df_insn_info *insn_info = DF_INSN_UID_SAFE_GET (uid);
+         if (insn_info)
+           df_insn_rescan (insn_info->insn);
+       }
+      BITMAP_FREE (debug->to_rescan);
+    }
+}
+
+/* Add USE to DEBUG.  It must be a dead reference to UREGNO in a debug
+   insn.  Create a bitmap for DEBUG as needed.  */
+static inline void
+dead_debug_add (struct dead_debug *debug, df_ref use, unsigned int uregno)
+{
+  struct dead_debug_use *newddu = XNEW (struct dead_debug_use);
+
+  newddu->use = use;
+  newddu->next = debug->head;
+  debug->head = newddu;
+
+  if (!debug->used)
+    debug->used = BITMAP_ALLOC (NULL);
+
+  bitmap_set_bit (debug->used, uregno);
+}
+
+/* If UREGNO is referenced by any entry in DEBUG, emit a debug insn
+   before INSN that binds the REG to a debug temp, and replace all
+   uses of UREGNO in DEBUG with uses of the debug temp.  INSN must be
+   the insn where UREGNO dies.  */
+static inline void
+dead_debug_insert_before (struct dead_debug *debug, unsigned int uregno,
+                         rtx insn)
+{
+  struct dead_debug_use **tailp = &debug->head;
+  struct dead_debug_use *cur;
+  struct dead_debug_use *uses = NULL;
+  struct dead_debug_use **usesp = &uses;
+  rtx reg = NULL;
+  rtx dval;
+  rtx bind;
+
+  if (!debug->used || !bitmap_clear_bit (debug->used, uregno))
+    return;
+
+  /* Move all uses of uregno from debug->head to uses, setting mode to
+     the widest referenced mode.  */
+  while ((cur = *tailp))
+    {
+      if (DF_REF_REGNO (cur->use) == uregno)
+       {
+         *usesp = cur;
+         usesp = &cur->next;
+         *tailp = cur->next;
+         cur->next = NULL;
+         if (!reg
+             || (GET_MODE_BITSIZE (GET_MODE (reg))
+                 < GET_MODE_BITSIZE (GET_MODE (*DF_REF_REAL_LOC (cur->use)))))
+           reg = *DF_REF_REAL_LOC (cur->use);
+       }
+      else
+       tailp = &(*tailp)->next;
+    }
+
+  gcc_assert (reg);
+
+  /* Create DEBUG_EXPR (and DEBUG_EXPR_DECL).  */
+  dval = make_debug_expr_from_rtl (reg);
+
+  /* Emit a debug bind insn before the insn in which reg dies.  */
+  bind = gen_rtx_VAR_LOCATION (GET_MODE (reg),
+                              DEBUG_EXPR_TREE_DECL (dval), reg,
+                              VAR_INIT_STATUS_INITIALIZED);
+
+  bind = emit_debug_insn_before (bind, insn);
+  df_insn_rescan (bind);
+
+  /* Adjust all uses.  */
+  while ((cur = uses))
+    {
+      if (GET_MODE (*DF_REF_REAL_LOC (cur->use)) == GET_MODE (reg))
+       *DF_REF_REAL_LOC (cur->use) = dval;
+      else
+       *DF_REF_REAL_LOC (cur->use)
+         = gen_lowpart_SUBREG (GET_MODE (*DF_REF_REAL_LOC (cur->use)), dval);
+      /* ??? Should we simplify subreg of subreg?  */
+      if (debug->to_rescan == NULL)
+       debug->to_rescan = BITMAP_ALLOC (NULL);
+      bitmap_set_bit (debug->to_rescan, INSN_UID (DF_REF_INSN (cur->use)));
+      uses = cur->next;
+      XDELETE (cur);
+    }
+}
+
+/* Recompute the REG_DEAD and REG_UNUSED notes and compute register
+   info: lifetime, bb, and number of defs and uses for basic block
+   BB.  The three bitvectors are scratch regs used here.  */
+
+static void
+df_note_bb_compute (unsigned int bb_index,
+                   bitmap live, bitmap do_not_gen, bitmap artificial_uses)
+{
+  basic_block bb = BASIC_BLOCK (bb_index);
+  rtx insn;
+  df_ref *def_rec;
+  df_ref *use_rec;
+  struct dead_debug debug;
+
+  dead_debug_init (&debug, NULL);
+
+  bitmap_copy (live, df_get_live_out (bb));
+  bitmap_clear (artificial_uses);
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    {
+      fprintf (dump_file, "live at bottom ");
+      df_print_regset (dump_file, live);
+    }
+#endif
+
+  /* Process the artificial defs and uses at the bottom of the block
+     to begin processing.  */
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+#ifdef REG_DEAD_DEBUGGING
+      if (dump_file)
+       fprintf (dump_file, "artificial def %d\n", DF_REF_REGNO (def));
+#endif
+
+      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
+       bitmap_clear_bit (live, DF_REF_REGNO (def));
+    }
+
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+    {
+      df_ref use = *use_rec;
+      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
+       {
+         unsigned int regno = DF_REF_REGNO (use);
+         bitmap_set_bit (live, regno);
+
+         /* Notes are not generated for any of the artificial registers
+            at the bottom of the block.  */
+         bitmap_set_bit (artificial_uses, regno);
+       }
+    }
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    {
+      fprintf (dump_file, "live before artificials out ");
+      df_print_regset (dump_file, live);
+    }
+#endif
+
+  FOR_BB_INSNS_REVERSE (bb, insn)
+    {
+      unsigned int uid = INSN_UID (insn);
+      struct df_mw_hardreg **mws_rec;
+      rtx old_dead_notes;
+      rtx old_unused_notes;
+      int debug_insn;
+
+      if (!INSN_P (insn))
+       continue;
+
+      debug_insn = DEBUG_INSN_P (insn);
+
+      bitmap_clear (do_not_gen);
+      df_kill_notes (insn, &old_dead_notes, &old_unused_notes);
+
+      /* Process the defs.  */
+      if (CALL_P (insn))
+       {
+#ifdef REG_DEAD_DEBUGGING
+         if (dump_file)
+           {
+             fprintf (dump_file, "processing call %d\n  live =", INSN_UID (insn));
+             df_print_regset (dump_file, live);
            }
            }
-         else
-           {
-             rtx next = XEXP (link, 1);
-#ifdef REG_DEAD_DEBUGGING
-             df_print_note ("deleting: ", insn, link);
 #endif
 #endif
-             XEXP (link, 1) = dead;
-             dead = link;
-             *pprev = link = next;
+         /* We only care about real sets for calls.  Clobbers cannot
+            be depended on to really die.  */
+         mws_rec = DF_INSN_UID_MWS (uid);
+         while (*mws_rec)
+           {
+             struct df_mw_hardreg *mws = *mws_rec;
+             if ((DF_MWS_REG_DEF_P (mws))
+                 && !df_ignore_stack_reg (mws->start_regno))
+               old_unused_notes
+                 = df_set_unused_notes_for_mw (insn, old_unused_notes,
+                                               mws, live, do_not_gen,
+                                               artificial_uses);
+             mws_rec++;
            }
            }
-         break;
 
 
-       case REG_UNUSED:
-         /* After reg-stack, we need to ignore any unused notes 
-            for the stack registers.  */
-         if (df_ignore_stack_reg (REGNO (XEXP (link, 0))))
+         /* All of the defs except the return value are some sort of
+            clobber.  This code is for the return.  */
+         for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
            {
            {
-             pprev = &XEXP (link, 1);
-             link = *pprev;
+             df_ref def = *def_rec;
+             unsigned int dregno = DF_REF_REGNO (def);
+             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER))
+               {
+                 old_unused_notes
+                   = df_create_unused_note (insn, old_unused_notes,
+                                            def, live, artificial_uses);
+                 bitmap_set_bit (do_not_gen, dregno);
+               }
+
+             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_PARTIAL | DF_REF_CONDITIONAL))
+               bitmap_clear_bit (live, dregno);
            }
            }
-         else
+       }
+      else
+       {
+         /* Regular insn.  */
+         mws_rec = DF_INSN_UID_MWS (uid);
+         while (*mws_rec)
            {
            {
-             rtx next = XEXP (link, 1);
+             struct df_mw_hardreg *mws = *mws_rec;
+             if (DF_MWS_REG_DEF_P (mws))
+               old_unused_notes
+                 = df_set_unused_notes_for_mw (insn, old_unused_notes,
+                                               mws, live, do_not_gen,
+                                               artificial_uses);
+             mws_rec++;
+           }
+
+         for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+           {
+             df_ref def = *def_rec;
+             unsigned int dregno = DF_REF_REGNO (def);
+             old_unused_notes
+               = df_create_unused_note (insn, old_unused_notes,
+                                        def, live, artificial_uses);
+
+             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER))
+               bitmap_set_bit (do_not_gen, dregno);
+
+             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_PARTIAL | DF_REF_CONDITIONAL))
+               bitmap_clear_bit (live, dregno);
+           }
+       }
+
+      /* Process the uses.  */
+      mws_rec = DF_INSN_UID_MWS (uid);
+      while (*mws_rec)
+       {
+         struct df_mw_hardreg *mws = *mws_rec;
+         if ((DF_MWS_REG_DEF_P (mws))
+             && !df_ignore_stack_reg (mws->start_regno))
+           {
+             bool really_add_notes = debug_insn != 0;
+
+             old_dead_notes
+               = df_set_dead_notes_for_mw (insn, old_dead_notes,
+                                           mws, live, do_not_gen,
+                                           artificial_uses,
+                                           &really_add_notes);
+
+             if (really_add_notes)
+               debug_insn = -1;
+           }
+         mws_rec++;
+       }
+
+      for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
+       {
+         df_ref use = *use_rec;
+         unsigned int uregno = DF_REF_REGNO (use);
+
 #ifdef REG_DEAD_DEBUGGING
 #ifdef REG_DEAD_DEBUGGING
-             df_print_note ("deleting: ", insn, link);
+         if (dump_file && !debug_insn)
+           {
+             fprintf (dump_file, "  regular looking at use ");
+             df_ref_debug (use, dump_file);
+           }
+#endif
+         if (!bitmap_bit_p (live, uregno))
+           {
+             if (debug_insn)
+               {
+                 if (debug_insn > 0)
+                   {
+                     dead_debug_add (&debug, use, uregno);
+                     continue;
+                   }
+                 break;
+               }
+             else
+               dead_debug_insert_before (&debug, uregno, insn);
+
+             if ( (!(DF_REF_FLAGS (use) & DF_REF_MW_HARDREG))
+                  && (!bitmap_bit_p (do_not_gen, uregno))
+                  && (!bitmap_bit_p (artificial_uses, uregno))
+                  && (!(DF_REF_FLAGS (use) & DF_REF_READ_WRITE))
+                  && (!df_ignore_stack_reg (uregno)))
+               {
+                 rtx reg = (DF_REF_LOC (use))
+                            ? *DF_REF_REAL_LOC (use) : DF_REF_REG (use);
+                 old_dead_notes = df_set_note (REG_DEAD, insn, old_dead_notes, reg);
+
+#ifdef REG_DEAD_DEBUGGING
+                 df_print_note ("adding 4: ", insn, REG_NOTES (insn));
+#endif
+               }
+             /* This register is now live.  */
+             bitmap_set_bit (live, uregno);
+           }
+       }
+
+      while (old_unused_notes)
+       {
+         rtx next = XEXP (old_unused_notes, 1);
+         free_EXPR_LIST_node (old_unused_notes);
+         old_unused_notes = next;
+       }
+      while (old_dead_notes)
+       {
+         rtx next = XEXP (old_dead_notes, 1);
+         free_EXPR_LIST_node (old_dead_notes);
+         old_dead_notes = next;
+       }
+
+      if (debug_insn == -1)
+       {
+         /* ??? We could probably do better here, replacing dead
+            registers with their definitions.  */
+         INSN_VAR_LOCATION_LOC (insn) = gen_rtx_UNKNOWN_VAR_LOC ();
+         df_insn_rescan_debug_internal (insn);
+       }
+    }
+
+  dead_debug_finish (&debug, NULL);
+}
+
+
+/* Compute register info: lifetime, bb, and number of defs and uses.  */
+static void
+df_note_compute (bitmap all_blocks)
+{
+  unsigned int bb_index;
+  bitmap_iterator bi;
+  bitmap live = BITMAP_ALLOC (&df_bitmap_obstack);
+  bitmap do_not_gen = BITMAP_ALLOC (&df_bitmap_obstack);
+  bitmap artificial_uses = BITMAP_ALLOC (&df_bitmap_obstack);
+
+#ifdef REG_DEAD_DEBUGGING
+  if (dump_file)
+    print_rtl_with_bb (dump_file, get_insns());
 #endif
 #endif
-             XEXP (link, 1) = unused;
-             unused = link;
-             *pprev = link = next;
-           }
-         break;
-         
-       default:
-         pprev = &XEXP (link, 1);
-         link = *pprev;
-         break;
-       }
-    }
 
 
-  *old_dead_notes = dead;
-  *old_unused_notes = unused;
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
+  {
+    df_note_bb_compute (bb_index, live, do_not_gen, artificial_uses);
+  }
+
+  BITMAP_FREE (live);
+  BITMAP_FREE (do_not_gen);
+  BITMAP_FREE (artificial_uses);
 }
 
 
 }
 
 
-/* Set a NOTE_TYPE note for REG in INSN.  Try to pull it from the OLD
-   list, otherwise create a new one.  */
+/* Free all storage associated with the problem.  */
 
 
-static inline rtx
-df_set_note (enum reg_note note_type, rtx insn, rtx old, rtx reg)
+static void
+df_note_free (void)
 {
 {
-  rtx this = old;
-  rtx prev = NULL;
-
-  while (this)
-    if (XEXP (this, 0) == reg)
-      {
-       if (prev)
-         XEXP (prev, 1) = XEXP (this, 1);
-       else
-         old = XEXP (this, 1);
-       XEXP (this, 1) = REG_NOTES (insn);
-       REG_NOTES (insn) = this;
-       return old;
-      }
-    else
-      {
-       prev = this;
-       this = XEXP (this, 1);
-      }
-  
-  /* Did not find the note.  */
-  REG_NOTES (insn) = alloc_EXPR_LIST (note_type, reg, REG_NOTES (insn));
-  return old;
+  free (df_note);
 }
 
 }
 
-/* A subroutine of df_set_unused_notes_for_mw, with a selection of its
-   arguments.  Return true if the register value described by MWS's
-   mw_reg is known to be completely unused, and if mw_reg can therefore
-   be used in a REG_UNUSED note.  */
 
 
-static bool
-df_whole_mw_reg_unused_p (struct df_mw_hardreg *mws,
-                         bitmap live, bitmap artificial_uses)
+/* All of the information associated every instance of the problem.  */
+
+static struct df_problem problem_NOTE =
 {
 {
-  unsigned int r;
+  DF_NOTE,                    /* Problem id.  */
+  DF_NONE,                    /* Direction.  */
+  df_note_alloc,              /* Allocate the problem specific data.  */
+  NULL,                       /* Reset global information.  */
+  NULL,                       /* Free basic block info.  */
+  df_note_compute,            /* Local compute function.  */
+  NULL,                       /* Init the solution specific data.  */
+  NULL,                       /* Iterative solver.  */
+  NULL,                       /* Confluence operator 0.  */
+  NULL,                       /* Confluence operator n.  */
+  NULL,                       /* Transfer function.  */
+  NULL,                       /* Finalize function.  */
+  df_note_free,               /* Free all of the problem information.  */
+  df_note_free,               /* Remove this problem from the stack of dataflow problems.  */
+  NULL,                       /* Debugging.  */
+  NULL,                       /* Debugging start block.  */
+  NULL,                       /* Debugging end block.  */
+  NULL,                       /* Incremental solution verify start.  */
+  NULL,                       /* Incremental solution verify end.  */
+  &problem_LR,                /* Dependent problem.  */
+  TV_DF_NOTE,                 /* Timing variable.  */
+  false                       /* Reset blocks on dropping out of blocks_to_analyze.  */
+};
 
 
-  /* If MWS describes a partial reference, create REG_UNUSED notes for
-     individual hard registers.  */
-  if (mws->flags & DF_REF_PARTIAL)
-    return false;
 
 
-  /* Likewise if some part of the register is used.  */
-  for (r = mws->start_regno; r <= mws->end_regno; r++)
-    if (bitmap_bit_p (live, r)
-       || bitmap_bit_p (artificial_uses, r))
-      return false;
+/* Create a new DATAFLOW instance and add it to an existing instance
+   of DF.  The returned structure is what is used to get at the
+   solution.  */
 
 
-  gcc_assert (REG_P (mws->mw_reg));
-  return true;
+void
+df_note_add_problem (void)
+{
+  df_add_problem (&problem_NOTE);
 }
 
 }
 
-/* Set the REG_UNUSED notes for the multiword hardreg defs in INSN
-   based on the bits in LIVE.  Do not generate notes for registers in
-   artificial uses.  DO_NOT_GEN is updated so that REG_DEAD notes are
-   not generated if the reg is both read and written by the
-   instruction.
-*/
 
 
-static rtx
-df_set_unused_notes_for_mw (rtx insn, rtx old, struct df_mw_hardreg *mws,
-                           bitmap live, bitmap do_not_gen, 
-                           bitmap artificial_uses)
+
+\f
+/*----------------------------------------------------------------------------
+   Functions for simulating the effects of single insns.
+
+   You can either simulate in the forwards direction, starting from
+   the top of a block or the backwards direction from the end of the
+   block.  If you go backwards, defs are examined first to clear bits,
+   then uses are examined to set bits.  If you go forwards, defs are
+   examined first to set bits, then REG_DEAD and REG_UNUSED notes
+   are examined to clear bits.  In either case, the result of examining
+   a def can be undone (respectively by a use or a REG_UNUSED note).
+
+   If you start at the top of the block, use one of DF_LIVE_IN or
+   DF_LR_IN.  If you start at the bottom of the block use one of
+   DF_LIVE_OUT or DF_LR_OUT.  BE SURE TO PASS A COPY OF THESE SETS,
+   THEY WILL BE DESTROYED.
+----------------------------------------------------------------------------*/
+
+
+/* Find the set of DEFs for INSN.  */
+
+void
+df_simulate_find_defs (rtx insn, bitmap defs)
 {
 {
-  unsigned int r;
-  
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
-    fprintf (dump_file, "mw_set_unused looking at mws[%d..%d]\n", 
-            mws->start_regno, mws->end_regno);
-#endif
+  df_ref *def_rec;
+  unsigned int uid = INSN_UID (insn);
 
 
-  if (df_whole_mw_reg_unused_p (mws, live, artificial_uses))
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
     {
     {
-      unsigned int regno = mws->start_regno;
-      old = df_set_note (REG_UNUSED, insn, old, mws->mw_reg);
+      df_ref def = *def_rec;
+      bitmap_set_bit (defs, DF_REF_REGNO (def));
+    }
+}
 
 
-#ifdef REG_DEAD_DEBUGGING
-      df_print_note ("adding 1: ", insn, REG_NOTES (insn));
-#endif
-      bitmap_set_bit (do_not_gen, regno);
-      /* Only do this if the value is totally dead.  */
+/* Find the set of real DEFs, which are not clobbers, for INSN.  */
+
+void
+df_simulate_find_noclobber_defs (rtx insn, bitmap defs)
+{
+  df_ref *def_rec;
+  unsigned int uid = INSN_UID (insn);
+
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if (!(DF_REF_FLAGS (def) & (DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER)))
+       bitmap_set_bit (defs, DF_REF_REGNO (def));
     }
     }
-  else
-    for (r = mws->start_regno; r <= mws->end_regno; r++)
-      {
-       if (!bitmap_bit_p (live, r)
-           && !bitmap_bit_p (artificial_uses, r))
-         {
-           old = df_set_note (REG_UNUSED, insn, old, regno_reg_rtx[r]);
-#ifdef REG_DEAD_DEBUGGING
-           df_print_note ("adding 2: ", insn, REG_NOTES (insn));
-#endif
-         }
-       bitmap_set_bit (do_not_gen, r);
-      }
-  return old;
 }
 
 
 }
 
 
-/* A subroutine of df_set_dead_notes_for_mw, with a selection of its
-   arguments.  Return true if the register value described by MWS's
-   mw_reg is known to be completely dead, and if mw_reg can therefore
-   be used in a REG_DEAD note.  */
+/* Simulate the effects of the defs of INSN on LIVE.  */
 
 
-static bool
-df_whole_mw_reg_dead_p (struct df_mw_hardreg *mws,
-                       bitmap live, bitmap artificial_uses,
-                       bitmap do_not_gen)
+void
+df_simulate_defs (rtx insn, bitmap live)
 {
 {
-  unsigned int r;
-
-  /* If MWS describes a partial reference, create REG_DEAD notes for
-     individual hard registers.  */
-  if (mws->flags & DF_REF_PARTIAL)
-    return false;
+  df_ref *def_rec;
+  unsigned int uid = INSN_UID (insn);
 
 
-  /* Likewise if some part of the register is not dead.  */
-  for (r = mws->start_regno; r <= mws->end_regno; r++)
-    if (bitmap_bit_p (live, r)
-       || bitmap_bit_p (artificial_uses, r)
-       || bitmap_bit_p (do_not_gen, r))
-      return false;
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      unsigned int dregno = DF_REF_REGNO (def);
 
 
-  gcc_assert (REG_P (mws->mw_reg));
-  return true;
+      /* If the def is to only part of the reg, it does
+        not kill the other defs that reach here.  */
+      if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
+       bitmap_clear_bit (live, dregno);
+    }
 }
 
 }
 
-/* Set the REG_DEAD notes for the multiword hardreg use in INSN based
-   on the bits in LIVE.  DO_NOT_GEN is used to keep REG_DEAD notes
-   from being set if the instruction both reads and writes the
-   register.  */
 
 
-static rtx
-df_set_dead_notes_for_mw (rtx insn, rtx old, struct df_mw_hardreg *mws,
-                         bitmap live, bitmap do_not_gen,
-                         bitmap artificial_uses)
+/* Simulate the effects of the uses of INSN on LIVE.  */
+
+void
+df_simulate_uses (rtx insn, bitmap live)
 {
 {
-  unsigned int r;
-  
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
-    {
-      fprintf (dump_file, "mw_set_dead looking at mws[%d..%d]\n  do_not_gen =", 
-              mws->start_regno, mws->end_regno);
-      df_print_regset (dump_file, do_not_gen);
-      fprintf (dump_file, "  live =");
-      df_print_regset (dump_file, live);
-      fprintf (dump_file, "  artificial uses =");
-      df_print_regset (dump_file, artificial_uses);
-    }
-#endif
+  df_ref *use_rec;
+  unsigned int uid = INSN_UID (insn);
 
 
-  if (df_whole_mw_reg_dead_p (mws, live, artificial_uses, do_not_gen))
+  if (DEBUG_INSN_P (insn))
+    return;
+
+  for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
     {
     {
-      /* Add a dead note for the entire multi word register.  */
-      old = df_set_note (REG_DEAD, insn, old, mws->mw_reg);
-#ifdef REG_DEAD_DEBUGGING
-      df_print_note ("adding 1: ", insn, REG_NOTES (insn));
-#endif
+      df_ref use = *use_rec;
+      /* Add use to set of uses in this BB.  */
+      bitmap_set_bit (live, DF_REF_REGNO (use));
     }
     }
+}
+
+
+/* Add back the always live regs in BB to LIVE.  */
+
+static inline void
+df_simulate_fixup_sets (basic_block bb, bitmap live)
+{
+  /* These regs are considered always live so if they end up dying
+     because of some def, we need to bring the back again.  */
+  if (bb_has_eh_pred (bb))
+    bitmap_ior_into (live, df->eh_block_artificial_uses);
   else
   else
-    {
-      for (r = mws->start_regno; r <= mws->end_regno; r++)
-       if (!bitmap_bit_p (live, r)
-           && !bitmap_bit_p (artificial_uses, r)
-           && !bitmap_bit_p (do_not_gen, r))
-         {
-           old = df_set_note (REG_DEAD, insn, old, regno_reg_rtx[r]);
-#ifdef REG_DEAD_DEBUGGING
-           df_print_note ("adding 2: ", insn, REG_NOTES (insn));
-#endif
-         }
-    }
-  return old;
+    bitmap_ior_into (live, df->regular_block_artificial_uses);
 }
 
 
 }
 
 
-/* Create a REG_UNUSED note if necessary for DEF in INSN updating
-   LIVE.  Do not generate notes for registers in ARTIFICIAL_USES.  */
+/*----------------------------------------------------------------------------
+   The following three functions are used only for BACKWARDS scanning:
+   i.e. they process the defs before the uses.
+
+   df_simulate_initialize_backwards should be called first with a
+   bitvector copyied from the DF_LIVE_OUT or DF_LR_OUT.  Then
+   df_simulate_one_insn_backwards should be called for each insn in
+   the block, starting with the last one.  Finally,
+   df_simulate_finalize_backwards can be called to get a new value
+   of the sets at the top of the block (this is rarely used).
+   ----------------------------------------------------------------------------*/
 
 
-static rtx
-df_create_unused_note (rtx insn, rtx old, struct df_ref *def, 
-                      bitmap live, bitmap artificial_uses)
+/* Apply the artificial uses and defs at the end of BB in a backwards
+   direction.  */
+
+void
+df_simulate_initialize_backwards (basic_block bb, bitmap live)
 {
 {
-  unsigned int dregno = DF_REF_REGNO (def);
-  
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
+  df_ref *def_rec;
+  df_ref *use_rec;
+  int bb_index = bb->index;
+
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
     {
-      fprintf (dump_file, "  regular looking at def ");
-      df_ref_debug (def, dump_file);
+      df_ref def = *def_rec;
+      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
+       bitmap_clear_bit (live, DF_REF_REGNO (def));
     }
     }
-#endif
 
 
-  if (!(bitmap_bit_p (live, dregno)
-       || (DF_REF_FLAGS (def) & DF_REF_MW_HARDREG)
-       || bitmap_bit_p (artificial_uses, dregno)
-       || df_ignore_stack_reg (dregno)))
+  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
     {
-      rtx reg = (DF_REF_LOC (def)) 
-                ? *DF_REF_REAL_LOC (def): DF_REF_REG (def);
-      old = df_set_note (REG_UNUSED, insn, old, reg);
-#ifdef REG_DEAD_DEBUGGING
-      df_print_note ("adding 3: ", insn, REG_NOTES (insn));
-#endif
+      df_ref use = *use_rec;
+      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
+       bitmap_set_bit (live, DF_REF_REGNO (use));
     }
     }
-  
-  return old;
 }
 
 
 }
 
 
-/* Recompute the REG_DEAD and REG_UNUSED notes and compute register
-   info: lifetime, bb, and number of defs and uses for basic block
-   BB.  The three bitvectors are scratch regs used here.  */
+/* Simulate the backwards effects of INSN on the bitmap LIVE.  */
 
 
-static void
-df_note_bb_compute (unsigned int bb_index, 
-                   bitmap live, bitmap do_not_gen, bitmap artificial_uses)
+void
+df_simulate_one_insn_backwards (basic_block bb, rtx insn, bitmap live)
 {
 {
-  basic_block bb = BASIC_BLOCK (bb_index);
-  rtx insn;
-  struct df_ref **def_rec;
-  struct df_ref **use_rec;
+  if (!NONDEBUG_INSN_P (insn))
+    return;
 
 
-  bitmap_copy (live, df_get_live_out (bb));
-  bitmap_clear (artificial_uses);
+  df_simulate_defs (insn, live);
+  df_simulate_uses (insn, live);
+  df_simulate_fixup_sets (bb, live);
+}
 
 
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
-    {
-      fprintf (dump_file, "live at bottom ");
-      df_print_regset (dump_file, live);
-    }
+
+/* Apply the artificial uses and defs at the top of BB in a backwards
+   direction.  */
+
+void
+df_simulate_finalize_backwards (basic_block bb, bitmap live)
+{
+  df_ref *def_rec;
+#ifdef EH_USES
+  df_ref *use_rec;
 #endif
 #endif
+  int bb_index = bb->index;
 
 
-  /* Process the artificial defs and uses at the bottom of the block
-     to begin processing.  */
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
   for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
     {
-      struct df_ref *def = *def_rec;
-#ifdef REG_DEAD_DEBUGGING
-      if (dump_file)
-       fprintf (dump_file, "artificial def %d\n", DF_REF_REGNO (def));
-#endif
-
-      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
        bitmap_clear_bit (live, DF_REF_REGNO (def));
     }
 
        bitmap_clear_bit (live, DF_REF_REGNO (def));
     }
 
+#ifdef EH_USES
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
   for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
     {
-      struct df_ref *use = *use_rec;
-      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
-       {
-         unsigned int regno = DF_REF_REGNO (use);
-         bitmap_set_bit (live, regno);
-         
-         /* Notes are not generated for any of the artificial registers
-            at the bottom of the block.  */
-         bitmap_set_bit (artificial_uses, regno);
-       }
-    }
-  
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
-    {
-      fprintf (dump_file, "live before artificials out ");
-      df_print_regset (dump_file, live);
+      df_ref use = *use_rec;
+      if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
+       bitmap_set_bit (live, DF_REF_REGNO (use));
     }
 #endif
     }
 #endif
+}
+/*----------------------------------------------------------------------------
+   The following three functions are used only for FORWARDS scanning:
+   i.e. they process the defs and the REG_DEAD and REG_UNUSED notes.
+   Thus it is important to add the DF_NOTES problem to the stack of
+   problems computed before using these functions.
+
+   df_simulate_initialize_forwards should be called first with a
+   bitvector copyied from the DF_LIVE_IN or DF_LR_IN.  Then
+   df_simulate_one_insn_forwards should be called for each insn in
+   the block, starting with the first one.
+   ----------------------------------------------------------------------------*/
 
 
-  FOR_BB_INSNS_REVERSE (bb, insn)
-    {
-      unsigned int uid = INSN_UID (insn);
-      struct df_mw_hardreg **mws_rec;
-      rtx old_dead_notes;
-      rtx old_unused_notes;
-      if (!INSN_P (insn))
-       continue;
+/* Initialize the LIVE bitmap, which should be copied from DF_LIVE_IN or
+   DF_LR_IN for basic block BB, for forward scanning by marking artificial
+   defs live.  */
 
 
-      bitmap_clear (do_not_gen);
-      df_kill_notes (insn, &old_dead_notes, &old_unused_notes);
+void
+df_simulate_initialize_forwards (basic_block bb, bitmap live)
+{
+  df_ref *def_rec;
+  int bb_index = bb->index;
 
 
-      /* Process the defs.  */
-      if (CALL_P (insn))
-       {
-#ifdef REG_DEAD_DEBUGGING
-         if (dump_file)
-           {
-             fprintf (dump_file, "processing call %d\n  live =", INSN_UID (insn));
-             df_print_regset (dump_file, live);
-           }
-#endif
-         /* We only care about real sets for calls.  Clobbers cannot
-            be depended on to really die.  */
-         mws_rec = DF_INSN_UID_MWS (uid);
-         while (*mws_rec)
-           {
-             struct df_mw_hardreg *mws = *mws_rec; 
-             if ((mws->type == DF_REF_REG_DEF) 
-                 && !df_ignore_stack_reg (mws->start_regno))
-               old_unused_notes 
-                 = df_set_unused_notes_for_mw (insn, old_unused_notes, 
-                                               mws, live, do_not_gen, 
-                                               artificial_uses);
-             mws_rec++;
-           }
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
+       bitmap_set_bit (live, DF_REF_REGNO (def));
+    }
+}
 
 
-         /* All of the defs except the return value are some sort of
-            clobber.  This code is for the return.  */
-         for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
-           {
-             struct df_ref *def = *def_rec;
-             unsigned int dregno = DF_REF_REGNO (def);
-             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER))
-               {
-                 old_unused_notes
-                   = df_create_unused_note (insn, old_unused_notes, 
-                                            def, live, artificial_uses);
-                 bitmap_set_bit (do_not_gen, dregno);
-               }
+/* Simulate the forwards effects of INSN on the bitmap LIVE.  */
 
 
-             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_PARTIAL | DF_REF_CONDITIONAL))
-               bitmap_clear_bit (live, dregno);
-           }
-       }
-      else
-       {
-         /* Regular insn.  */
-         mws_rec = DF_INSN_UID_MWS (uid);
-         while (*mws_rec)
-           {
-             struct df_mw_hardreg *mws = *mws_rec; 
-             if (mws->type == DF_REF_REG_DEF)
-               old_unused_notes
-                 = df_set_unused_notes_for_mw (insn, old_unused_notes, 
-                                               mws, live, do_not_gen, 
-                                               artificial_uses);
-             mws_rec++;
-           }
+void
+df_simulate_one_insn_forwards (basic_block bb, rtx insn, bitmap live)
+{
+  rtx link;
+  if (! INSN_P (insn))
+    return;
 
 
-         for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
-           {
-             struct df_ref *def = *def_rec;
-             unsigned int dregno = DF_REF_REGNO (def);
-             old_unused_notes
-               = df_create_unused_note (insn, old_unused_notes, 
-                                        def, live, artificial_uses);
+  /* Make sure that DF_NOTE really is an active df problem.  */
+  gcc_assert (df_note);
 
 
-             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_MUST_CLOBBER | DF_REF_MAY_CLOBBER))
-               bitmap_set_bit (do_not_gen, dregno);
+  /* Note that this is the opposite as how the problem is defined, because
+     in the LR problem defs _kill_ liveness.  However, they do so backwards,
+     while here the scan is performed forwards!  So, first assume that the
+     def is live, and if this is not true REG_UNUSED notes will rectify the
+     situation.  */
+  df_simulate_find_noclobber_defs (insn, live);
 
 
-             if (!DF_REF_FLAGS_IS_SET (def, DF_REF_PARTIAL | DF_REF_CONDITIONAL))
-               bitmap_clear_bit (live, dregno);
-           }
-       }
-      
-      /* Process the uses.  */
-      mws_rec = DF_INSN_UID_MWS (uid);
-      while (*mws_rec)
+  /* Clear all of the registers that go dead.  */
+  for (link = REG_NOTES (insn); link; link = XEXP (link, 1))
+    {
+      switch (REG_NOTE_KIND (link))
        {
        {
-         struct df_mw_hardreg *mws = *mws_rec; 
-         if ((mws->type != DF_REF_REG_DEF)  
-             && !df_ignore_stack_reg (mws->start_regno))
-           old_dead_notes
-             = df_set_dead_notes_for_mw (insn, old_dead_notes, 
-                                         mws, live, do_not_gen,
-                                         artificial_uses);
-         mws_rec++;
+       case REG_DEAD:
+       case REG_UNUSED:
+         {
+           rtx reg = XEXP (link, 0);
+           int regno = REGNO (reg);
+           if (regno < FIRST_PSEUDO_REGISTER)
+             {
+               int n = hard_regno_nregs[regno][GET_MODE (reg)];
+               while (--n >= 0)
+                 bitmap_clear_bit (live, regno + n);
+             }
+           else
+             bitmap_clear_bit (live, regno);
+         }
+         break;
+       default:
+         break;
        }
        }
+    }
+  df_simulate_fixup_sets (bb, live);
+}
 
 
-      for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
-       {
-         struct df_ref *use = *use_rec;
-         unsigned int uregno = DF_REF_REGNO (use);
 
 
-#ifdef REG_DEAD_DEBUGGING
-         if (dump_file)
-           {
-             fprintf (dump_file, "  regular looking at use ");
-             df_ref_debug (use, dump_file);
-           }
-#endif
-         if (!bitmap_bit_p (live, uregno))
-           {
-             if ( (!(DF_REF_FLAGS (use) & DF_REF_MW_HARDREG))
-                  && (!bitmap_bit_p (do_not_gen, uregno))
-                  && (!bitmap_bit_p (artificial_uses, uregno))
-                  && (!(DF_REF_FLAGS (use) & DF_REF_READ_WRITE))
-                  && (!df_ignore_stack_reg (uregno)))
-               {
-                 rtx reg = (DF_REF_LOC (use)) 
-                            ? *DF_REF_REAL_LOC (use) : DF_REF_REG (use);
-                 old_dead_notes = df_set_note (REG_DEAD, insn, old_dead_notes, reg);
+\f
+/*----------------------------------------------------------------------------
+   MULTIPLE DEFINITIONS
 
 
-#ifdef REG_DEAD_DEBUGGING
-                 df_print_note ("adding 4: ", insn, REG_NOTES (insn));
-#endif
-               }
-             /* This register is now live.  */
-             bitmap_set_bit (live, uregno);
-           }
-       }
+   Find the locations in the function reached by multiple definition sites
+   for a live pseudo.  In and out bitvectors are built for each basic
+   block.  They are restricted for efficiency to live registers.
 
 
-      while (old_unused_notes)
-       {
-         rtx next = XEXP (old_unused_notes, 1);
-         free_EXPR_LIST_node (old_unused_notes);
-         old_unused_notes = next;
-       }
-      while (old_dead_notes)
-       {
-         rtx next = XEXP (old_dead_notes, 1);
-         free_EXPR_LIST_node (old_dead_notes);
-         old_dead_notes = next;
-       }
+   The gen and kill sets for the problem are obvious.  Together they
+   include all defined registers in a basic block; the gen set includes
+   registers where a partial or conditional or may-clobber definition is
+   last in the BB, while the kill set includes registers with a complete
+   definition coming last.  However, the computation of the dataflow
+   itself is interesting.
+
+   The idea behind it comes from SSA form's iterated dominance frontier
+   criterion for inserting PHI functions.  Just like in that case, we can use
+   the dominance frontier to find places where multiple definitions meet;
+   a register X defined in a basic block BB1 has multiple definitions in
+   basic blocks in BB1's dominance frontier.
+
+   So, the in-set of a basic block BB2 is not just the union of the
+   out-sets of BB2's predecessors, but includes some more bits that come
+   from the basic blocks of whose dominance frontier BB2 is part (BB1 in
+   the previous paragraph).  I called this set the init-set of BB2.
+
+      (Note: I actually use the kill-set only to build the init-set.
+      gen bits are anyway propagated from BB1 to BB2 by dataflow).
+
+    For example, if you have
+
+       BB1 : r10 = 0
+             r11 = 0
+             if <...> goto BB2 else goto BB3;
+
+       BB2 : r10 = 1
+             r12 = 1
+             goto BB3;
+
+       BB3 :
+
+    you have BB3 in BB2's dominance frontier but not in BB1's, so that the
+    init-set of BB3 includes r10 and r12, but not r11.  Note that we do
+    not need to iterate the dominance frontier, because we do not insert
+    anything like PHI functions there!  Instead, dataflow will take care of
+    propagating the information to BB3's successors.
+   ---------------------------------------------------------------------------*/
+
+/* Scratch var used by transfer functions.  This is used to do md analysis
+   only for live registers.  */
+static bitmap df_md_scratch;
+
+/* Set basic block info.  */
+
+static void
+df_md_set_bb_info (unsigned int index,
+                   struct df_md_bb_info *bb_info)
+{
+  gcc_assert (df_md);
+  gcc_assert (index < df_md->block_info_size);
+  df_md->block_info[index] = bb_info;
+}
+
+
+static void
+df_md_free_bb_info (basic_block bb ATTRIBUTE_UNUSED,
+                    void *vbb_info)
+{
+  struct df_md_bb_info *bb_info = (struct df_md_bb_info *) vbb_info;
+  if (bb_info)
+    {
+      BITMAP_FREE (bb_info->kill);
+      BITMAP_FREE (bb_info->gen);
+      BITMAP_FREE (bb_info->init);
+      BITMAP_FREE (bb_info->in);
+      BITMAP_FREE (bb_info->out);
+      pool_free (df_md->block_pool, bb_info);
     }
 }
 
 
     }
 }
 
 
-/* Compute register info: lifetime, bb, and number of defs and uses.  */
+/* Allocate or reset bitmaps for DF_MD. The solution bits are
+   not touched unless the block is new.  */
+
 static void
 static void
-df_note_compute (bitmap all_blocks)
+df_md_alloc (bitmap all_blocks)
 {
   unsigned int bb_index;
   bitmap_iterator bi;
 {
   unsigned int bb_index;
   bitmap_iterator bi;
-  bitmap live = BITMAP_ALLOC (&df_bitmap_obstack);
-  bitmap do_not_gen = BITMAP_ALLOC (&df_bitmap_obstack);
-  bitmap artificial_uses = BITMAP_ALLOC (&df_bitmap_obstack);
 
 
-#ifdef REG_DEAD_DEBUGGING
-  if (dump_file)
-    print_rtl_with_bb (dump_file, get_insns());
-#endif
+  if (!df_md->block_pool)
+    df_md->block_pool = create_alloc_pool ("df_md_block pool",
+                                           sizeof (struct df_md_bb_info), 50);
+
+  df_grow_bb_info (df_md);
+  df_md_scratch = BITMAP_ALLOC (NULL);
 
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
 
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
-  {
-    df_note_bb_compute (bb_index, live, do_not_gen, artificial_uses);
-  }
+    {
+      struct df_md_bb_info *bb_info = df_md_get_bb_info (bb_index);
+      if (bb_info)
+        {
+          bitmap_clear (bb_info->init);
+          bitmap_clear (bb_info->gen);
+          bitmap_clear (bb_info->kill);
+          bitmap_clear (bb_info->in);
+          bitmap_clear (bb_info->out);
+        }
+      else
+        {
+          bb_info = (struct df_md_bb_info *) pool_alloc (df_md->block_pool);
+          df_md_set_bb_info (bb_index, bb_info);
+          bb_info->init = BITMAP_ALLOC (NULL);
+          bb_info->gen = BITMAP_ALLOC (NULL);
+          bb_info->kill = BITMAP_ALLOC (NULL);
+          bb_info->in = BITMAP_ALLOC (NULL);
+          bb_info->out = BITMAP_ALLOC (NULL);
+        }
+    }
 
 
-  BITMAP_FREE (live);
-  BITMAP_FREE (do_not_gen);
-  BITMAP_FREE (artificial_uses);
+  df_md->optional_p = true;
 }
 
 }
 
+/* Add the effect of the top artificial defs of BB to the multiple definitions
+   bitmap LOCAL_MD.  */
 
 
-/* Free all storage associated with the problem.  */
-
-static void
-df_note_free (void)
+void
+df_md_simulate_artificial_defs_at_top (basic_block bb, bitmap local_md)
 {
 {
-  free (df_note);
+  int bb_index = bb->index;
+  df_ref *def_rec;
+  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
+       {
+         unsigned int dregno = DF_REF_REGNO (def);
+         if (DF_REF_FLAGS (def)
+             & (DF_REF_PARTIAL | DF_REF_CONDITIONAL | DF_REF_MAY_CLOBBER))
+           bitmap_set_bit (local_md, dregno);
+         else
+           bitmap_clear_bit (local_md, dregno);
+       }
+    }
 }
 
 
 }
 
 
-/* All of the information associated every instance of the problem.  */
+/* Add the effect of the defs of INSN to the reaching definitions bitmap
+   LOCAL_MD.  */
 
 
-static struct df_problem problem_NOTE =
+void
+df_md_simulate_one_insn (basic_block bb ATTRIBUTE_UNUSED, rtx insn,
+                        bitmap local_md)
 {
 {
-  DF_NOTE,                    /* Problem id.  */
-  DF_NONE,                    /* Direction.  */
-  df_note_alloc,              /* Allocate the problem specific data.  */
-  NULL,                       /* Reset global information.  */
-  NULL,                       /* Free basic block info.  */
-  df_note_compute,            /* Local compute function.  */
-  NULL,                       /* Init the solution specific data.  */
-  NULL,                       /* Iterative solver.  */
-  NULL,                       /* Confluence operator 0.  */ 
-  NULL,                       /* Confluence operator n.  */ 
-  NULL,                       /* Transfer function.  */
-  NULL,                       /* Finalize function.  */
-  df_note_free,               /* Free all of the problem information.  */
-  df_note_free,               /* Remove this problem from the stack of dataflow problems.  */
-  NULL,                       /* Debugging.  */
-  NULL,                       /* Debugging start block.  */
-  NULL,                       /* Debugging end block.  */
-  NULL,                       /* Incremental solution verify start.  */
-  NULL,                       /* Incremental solution verify end.  */
-  &problem_LR,                /* Dependent problem.  */
-  TV_DF_NOTE,                 /* Timing variable.  */
-  false                       /* Reset blocks on dropping out of blocks_to_analyze.  */
-};
-
+  unsigned uid = INSN_UID (insn);
+  df_ref *def_rec;
 
 
-/* Create a new DATAFLOW instance and add it to an existing instance
-   of DF.  The returned structure is what is used to get at the
-   solution.  */
+  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+    {
+      df_ref def = *def_rec;
+      unsigned int dregno = DF_REF_REGNO (def);
+      if ((!(df->changeable_flags & DF_NO_HARD_REGS))
+          || (dregno >= FIRST_PSEUDO_REGISTER))
+        {
+          if (DF_REF_FLAGS (def)
+             & (DF_REF_PARTIAL | DF_REF_CONDITIONAL | DF_REF_MAY_CLOBBER))
+           bitmap_set_bit (local_md, DF_REF_ID (def));
+         else
+           bitmap_clear_bit (local_md, DF_REF_ID (def));
+        }
+    }
+}
 
 
-void
-df_note_add_problem (void)
+static void
+df_md_bb_local_compute_process_def (struct df_md_bb_info *bb_info,
+                                    df_ref *def_rec,
+                                    int top_flag)
 {
 {
-  df_add_problem (&problem_NOTE);
+  df_ref def;
+  bitmap_clear (seen_in_insn);
+
+  while ((def = *def_rec++) != NULL)
+    {
+      unsigned int dregno = DF_REF_REGNO (def);
+      if (((!(df->changeable_flags & DF_NO_HARD_REGS))
+           || (dregno >= FIRST_PSEUDO_REGISTER))
+         && top_flag == (DF_REF_FLAGS (def) & DF_REF_AT_TOP))
+       {
+          if (!bitmap_bit_p (seen_in_insn, dregno))
+           {
+             if (DF_REF_FLAGS (def)
+                 & (DF_REF_PARTIAL | DF_REF_CONDITIONAL | DF_REF_MAY_CLOBBER))
+               {
+                 bitmap_set_bit (bb_info->gen, dregno);
+                 bitmap_clear_bit (bb_info->kill, dregno);
+               }
+             else
+               {
+                 /* When we find a clobber and a regular def,
+                    make sure the regular def wins.  */
+                 bitmap_set_bit (seen_in_insn, dregno);
+                 bitmap_set_bit (bb_info->kill, dregno);
+                 bitmap_clear_bit (bb_info->gen, dregno);
+               }
+           }
+       }
+    }
 }
 
 
 }
 
 
+/* Compute local multiple def info for basic block BB.  */
 
 
-\f
-/*----------------------------------------------------------------------------
-   Functions for simulating the effects of single insns.  
+static void
+df_md_bb_local_compute (unsigned int bb_index)
+{
+  basic_block bb = BASIC_BLOCK (bb_index);
+  struct df_md_bb_info *bb_info = df_md_get_bb_info (bb_index);
+  rtx insn;
 
 
-   You can either simulate in the forwards direction, starting from
-   the top of a block or the backwards direction from the end of the
-   block.  The main difference is that if you go forwards, the uses
-   are examined first then the defs, and if you go backwards, the defs
-   are examined first then the uses.
+  /* Artificials are only hard regs.  */
+  if (!(df->changeable_flags & DF_NO_HARD_REGS))
+    df_md_bb_local_compute_process_def (bb_info,
+                                        df_get_artificial_defs (bb_index),
+                                        DF_REF_AT_TOP);
 
 
-   If you start at the top of the block, use one of DF_LIVE_IN or
-   DF_LR_IN.  If you start at the bottom of the block use one of
-   DF_LIVE_OUT or DF_LR_OUT.  BE SURE TO PASS A COPY OF THESE SETS,
-   THEY WILL BE DESTROYED.
+  FOR_BB_INSNS (bb, insn)
+    {
+      unsigned int uid = INSN_UID (insn);
+      if (!INSN_P (insn))
+        continue;
 
 
-----------------------------------------------------------------------------*/
+      df_md_bb_local_compute_process_def (bb_info, DF_INSN_UID_DEFS (uid), 0);
+    }
 
 
+  if (!(df->changeable_flags & DF_NO_HARD_REGS))
+    df_md_bb_local_compute_process_def (bb_info,
+                                        df_get_artificial_defs (bb_index),
+                                        0);
+}
 
 
-/* Find the set of DEFs for INSN.  */
+/* Compute local reaching def info for each basic block within BLOCKS.  */
 
 
-void
-df_simulate_find_defs (rtx insn, bitmap defs)
+static void
+df_md_local_compute (bitmap all_blocks)
 {
 {
-  struct df_ref **def_rec;
-  unsigned int uid = INSN_UID (insn);
+  unsigned int bb_index, df_bb_index;
+  bitmap_iterator bi1, bi2;
+  basic_block bb;
+  bitmap *frontiers;
 
 
-  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+  seen_in_insn = BITMAP_ALLOC (NULL);
+
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi1)
     {
     {
-      struct df_ref *def = *def_rec;
-      /* If the def is to only part of the reg, it does
-        not kill the other defs that reach here.  */
-      if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
-       bitmap_set_bit (defs, DF_REF_REGNO (def));
+      df_md_bb_local_compute (bb_index);
+    }
+
+  BITMAP_FREE (seen_in_insn);
+
+  frontiers = XNEWVEC (bitmap, last_basic_block);
+  FOR_ALL_BB (bb)
+    frontiers[bb->index] = BITMAP_ALLOC (NULL);
+
+  compute_dominance_frontiers (frontiers);
+
+  /* Add each basic block's kills to the nodes in the frontier of the BB.  */
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi1)
+    {
+      bitmap kill = df_md_get_bb_info (bb_index)->kill;
+      EXECUTE_IF_SET_IN_BITMAP (frontiers[bb_index], 0, df_bb_index, bi2)
+       {
+         basic_block bb = BASIC_BLOCK (df_bb_index);
+         if (bitmap_bit_p (all_blocks, df_bb_index))
+           bitmap_ior_and_into (df_md_get_bb_info (df_bb_index)->init, kill,
+                                df_get_live_in (bb));
+       }
     }
     }
+
+  FOR_ALL_BB (bb)
+    BITMAP_FREE (frontiers[bb->index]);
+  free (frontiers);
 }
 
 
 }
 
 
-/* Simulate the effects of the defs of INSN on LIVE.  */
+/* Reset the global solution for recalculation.  */
 
 
-void
-df_simulate_defs (rtx insn, bitmap live)
+static void
+df_md_reset (bitmap all_blocks)
 {
 {
-  struct df_ref **def_rec;
-  unsigned int uid = INSN_UID (insn);
+  unsigned int bb_index;
+  bitmap_iterator bi;
 
 
-  for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
     {
-      struct df_ref *def = *def_rec;
-      unsigned int dregno = DF_REF_REGNO (def);
-
-      /* If the def is to only part of the reg, it does
-        not kill the other defs that reach here.  */
-      if (!(DF_REF_FLAGS (def) & (DF_REF_PARTIAL | DF_REF_CONDITIONAL)))
-       bitmap_clear_bit (live, dregno);
+      struct df_md_bb_info *bb_info = df_md_get_bb_info (bb_index);
+      gcc_assert (bb_info);
+      bitmap_clear (bb_info->in);
+      bitmap_clear (bb_info->out);
     }
     }
-}  
+}
+
+static bool
+df_md_transfer_function (int bb_index)
+{
+  basic_block bb = BASIC_BLOCK (bb_index);
+  struct df_md_bb_info *bb_info = df_md_get_bb_info (bb_index);
+  bitmap in = bb_info->in;
+  bitmap out = bb_info->out;
+  bitmap gen = bb_info->gen;
+  bitmap kill = bb_info->kill;
 
 
+  /* We need to use a scratch set here so that the value returned from this
+     function invocation properly reflects whether the sets changed in a
+     significant way; i.e. not just because the live set was anded in.  */
+  bitmap_and (df_md_scratch, gen, df_get_live_out (bb));
 
 
-/* Simulate the effects of the uses of INSN on LIVE.  */
+  /* Multiple definitions of a register are not relevant if it is not
+     live.  Thus we trim the result to the places where it is live.  */
+  bitmap_and_into (in, df_get_live_in (bb));
 
 
-void 
-df_simulate_uses (rtx insn, bitmap live)
+  return bitmap_ior_and_compl (out, df_md_scratch, in, kill);
+}
+
+/* Initialize the solution bit vectors for problem.  */
+
+static void
+df_md_init (bitmap all_blocks)
 {
 {
-  struct df_ref **use_rec;
-  unsigned int uid = INSN_UID (insn);
+  unsigned int bb_index;
+  bitmap_iterator bi;
 
 
-  for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
+  EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
     {
-      struct df_ref *use = *use_rec;
-      /* Add use to set of uses in this BB.  */
-      bitmap_set_bit (live, DF_REF_REGNO (use));
+      struct df_md_bb_info *bb_info = df_md_get_bb_info (bb_index);
+
+      bitmap_copy (bb_info->in, bb_info->init);
+      df_md_transfer_function (bb_index);
     }
 }
 
     }
 }
 
+static void
+df_md_confluence_0 (basic_block bb)
+{
+  struct df_md_bb_info *bb_info = df_md_get_bb_info (bb->index);
+  bitmap_copy (bb_info->in, bb_info->init);
+}
 
 
-/* Add back the always live regs in BB to LIVE.  */
+/* In of target gets or of out of source.  */
 
 
-static inline void
-df_simulate_fixup_sets (basic_block bb, bitmap live)
+static void
+df_md_confluence_n (edge e)
 {
 {
-  /* These regs are considered always live so if they end up dying
-     because of some def, we need to bring the back again.  */
-  if (bb_has_eh_pred (bb))
-    bitmap_ior_into (live, df->eh_block_artificial_uses);
+  bitmap op1 = df_md_get_bb_info (e->dest->index)->in;
+  bitmap op2 = df_md_get_bb_info (e->src->index)->out;
+
+  if (e->flags & EDGE_FAKE)
+    return;
+
+  if (e->flags & EDGE_EH)
+    bitmap_ior_and_compl_into (op1, op2, regs_invalidated_by_call_regset);
   else
   else
-    bitmap_ior_into (live, df->regular_block_artificial_uses);
+    bitmap_ior_into (op1, op2);
 }
 
 }
 
+/* Free all storage associated with the problem.  */
 
 
-/* Apply the artificial uses and defs at the top of BB in a forwards
-   direction.  */
-
-void 
-df_simulate_artificial_refs_at_top (basic_block bb, bitmap live)
+static void
+df_md_free (void)
 {
 {
-  struct df_ref **def_rec;
-  struct df_ref **use_rec;
-  int bb_index = bb->index;
-  
-  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
+  unsigned int i;
+  for (i = 0; i < df_md->block_info_size; i++)
     {
     {
-      struct df_ref *use = *use_rec;
-      if (DF_REF_FLAGS (use) & DF_REF_AT_TOP)
-       bitmap_set_bit (live, DF_REF_REGNO (use));
+      struct df_md_bb_info *bb_info = df_md_get_bb_info (i);
+      if (bb_info)
+       {
+         BITMAP_FREE (bb_info->kill);
+         BITMAP_FREE (bb_info->gen);
+         BITMAP_FREE (bb_info->init);
+         BITMAP_FREE (bb_info->in);
+         BITMAP_FREE (bb_info->out);
+       }
     }
 
     }
 
-  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
-    {
-      struct df_ref *def = *def_rec;
-      if (DF_REF_FLAGS (def) & DF_REF_AT_TOP)
-       bitmap_clear_bit (live, DF_REF_REGNO (def));
-    }
+  BITMAP_FREE (df_md_scratch);
+  free_alloc_pool (df_md->block_pool);
+
+  df_md->block_info_size = 0;
+  free (df_md->block_info);
+  free (df_md);
 }
 
 
 }
 
 
-/* Simulate the forwards effects of INSN on the bitmap LIVE.  */
+/* Debugging info at top of bb.  */
 
 
-void 
-df_simulate_one_insn_forwards (basic_block bb, rtx insn, bitmap live)
+static void
+df_md_top_dump (basic_block bb, FILE *file)
 {
 {
-  if (! INSN_P (insn))
-    return;    
-  
-  df_simulate_uses (insn, live);
-  df_simulate_defs (insn, live);
-  df_simulate_fixup_sets (bb, live);
-}
+  struct df_md_bb_info *bb_info = df_md_get_bb_info (bb->index);
+  if (!bb_info || !bb_info->in)
+    return;
 
 
+  fprintf (file, ";; md  in  \t");
+  df_print_regset (file, bb_info->in);
+  fprintf (file, ";; md  init  \t");
+  df_print_regset (file, bb_info->init);
+  fprintf (file, ";; md  gen \t");
+  df_print_regset (file, bb_info->gen);
+  fprintf (file, ";; md  kill \t");
+  df_print_regset (file, bb_info->kill);
+}
 
 
-/* Apply the artificial uses and defs at the end of BB in a backwards
-   direction.  */
+/* Debugging info at bottom of bb.  */
 
 
-void 
-df_simulate_artificial_refs_at_end (basic_block bb, bitmap live)
+static void
+df_md_bottom_dump (basic_block bb, FILE *file)
 {
 {
-  struct df_ref **def_rec;
-  struct df_ref **use_rec;
-  int bb_index = bb->index;
-  
-  for (def_rec = df_get_artificial_defs (bb_index); *def_rec; def_rec++)
-    {
-      struct df_ref *def = *def_rec;
-      if ((DF_REF_FLAGS (def) & DF_REF_AT_TOP) == 0)
-       bitmap_clear_bit (live, DF_REF_REGNO (def));
-    }
+  struct df_md_bb_info *bb_info = df_md_get_bb_info (bb->index);
+  if (!bb_info || !bb_info->out)
+    return;
 
 
-  for (use_rec = df_get_artificial_uses (bb_index); *use_rec; use_rec++)
-    {
-      struct df_ref *use = *use_rec;
-      if ((DF_REF_FLAGS (use) & DF_REF_AT_TOP) == 0)
-       bitmap_set_bit (live, DF_REF_REGNO (use));
-    }
+  fprintf (file, ";; md  out \t");
+  df_print_regset (file, bb_info->out);
 }
 
 }
 
+static struct df_problem problem_MD =
+{
+  DF_MD,                      /* Problem id.  */
+  DF_FORWARD,                 /* Direction.  */
+  df_md_alloc,                /* Allocate the problem specific data.  */
+  df_md_reset,                /* Reset global information.  */
+  df_md_free_bb_info,         /* Free basic block info.  */
+  df_md_local_compute,        /* Local compute function.  */
+  df_md_init,                 /* Init the solution specific data.  */
+  df_worklist_dataflow,       /* Worklist solver.  */
+  df_md_confluence_0,         /* Confluence operator 0.  */
+  df_md_confluence_n,         /* Confluence operator n.  */
+  df_md_transfer_function,    /* Transfer function.  */
+  NULL,                       /* Finalize function.  */
+  df_md_free,                 /* Free all of the problem information.  */
+  df_md_free,                 /* Remove this problem from the stack of dataflow problems.  */
+  NULL,                       /* Debugging.  */
+  df_md_top_dump,             /* Debugging start block.  */
+  df_md_bottom_dump,          /* Debugging end block.  */
+  NULL,                              /* Incremental solution verify start.  */
+  NULL,                              /* Incremental solution verify end.  */
+  NULL,                       /* Dependent problem.  */
+  TV_DF_MD,                   /* Timing variable.  */
+  false                       /* Reset blocks on dropping out of blocks_to_analyze.  */
+};
 
 
-/* Simulate the backwards effects of INSN on the bitmap LIVE.  */
+/* Create a new MD instance and add it to the existing instance
+   of DF.  */
 
 
-void 
-df_simulate_one_insn_backwards (basic_block bb, rtx insn, bitmap live)
+void
+df_md_add_problem (void)
 {
 {
-  if (! INSN_P (insn))
-    return;    
-  
-  df_simulate_defs (insn, live);
-  df_simulate_uses (insn, live);
-  df_simulate_fixup_sets (bb, live);
+  df_add_problem (&problem_MD);
 }
 
 
 }
 
 
+