pseudo reaches.  In and out bitvectors are built for each basic
    block.  The id field in the ref is used to index into these sets.
    See df.h for details.
+
+   If the DF_RD_PRUNE_DEAD_DEFS changable flag is set, only DEFs reaching
+   existing uses are included in the global reaching DEFs set, or in other
+   words only DEFs that are still live.  This is a kind of pruned version
+   of the traditional reaching definitions problem that is much less
+   complex to compute and produces enough information to compute UD-chains.
+   In this context, live must be interpreted in the DF_LR sense: Uses that
+   are upward exposed but maybe not initialized on all paths through the
+   CFG.  For a USE that is not reached by a DEF on all paths, we still want
+   to make those DEFs that do reach the USE visible, and pruning based on
+   DF_LIVE would make that impossible.
    ----------------------------------------------------------------------------*/
 
 /* This problem plays a large number of games for the sake of
   df_grow_bb_info (df_rd);
 
   /* Because of the clustering of all use sites for the same pseudo,
-     we have to process all of the blocks before doing the
-     analysis.  */
+     we have to process all of the blocks before doing the analysis.  */
 
   EXECUTE_IF_SET_IN_BITMAP (all_blocks, 0, bb_index, bi)
     {
   /* Set up the knockout bit vectors to be applied across EH_EDGES.  */
   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
-       bitmap_set_range (dense_invalidated,
-                         DF_DEFS_BEGIN (regno),
-                         DF_DEFS_COUNT (regno));
+      if (! HARD_REGISTER_NUM_P (regno)
+         || !(df->changeable_flags & DF_NO_HARD_REGS))
+       {
+         if (DF_DEFS_COUNT (regno) > DF_SPARSE_THRESHOLD)
+           bitmap_set_bit (sparse_invalidated, regno);
+         else
+           bitmap_set_range (dense_invalidated,
+                             DF_DEFS_BEGIN (regno),
+                             DF_DEFS_COUNT (regno));
+       }
     }
 
   bitmap_clear (&seen_in_block);
   bitmap gen = &bb_info->gen;
   bitmap kill = &bb_info->kill;
   bitmap sparse_kill = &bb_info->sparse_kill;
+  bool changed = false;
 
   if (bitmap_empty_p (sparse_kill))
-    return  bitmap_ior_and_compl (out, gen, in, kill);
+    changed = bitmap_ior_and_compl (out, gen, in, kill);
   else
     {
       struct df_rd_problem_data *problem_data;
-      bool changed = false;
       bitmap_head tmp;
 
       /* Note that TMP is _not_ a temporary bitmap if we end up replacing
          bb_info->out = tmp;
        }
       else
-         bitmap_clear (&tmp);
-      return changed;
+       bitmap_clear (&tmp);
     }
-}
 
+  if (df->changeable_flags & DF_RD_PRUNE_DEAD_DEFS)
+    {
+      /* Create a mask of DEFs for all registers live at the end of this
+        basic block, and mask out DEFs of registers that are not live.
+        Computing the mask looks costly, but the benefit of the pruning
+        outweighs the cost.  */
+      struct df_rd_bb_info *bb_info = df_rd_get_bb_info (bb_index);
+      bitmap regs_live_out = &df_lr_get_bb_info (bb_index)->out;
+      bitmap live_defs = BITMAP_ALLOC (&df_bitmap_obstack);
+      unsigned int regno;
+      bitmap_iterator bi;
+
+      EXECUTE_IF_SET_IN_BITMAP (regs_live_out, 0, regno, bi)
+       bitmap_set_range (live_defs,
+                         DF_DEFS_BEGIN (regno),
+                         DF_DEFS_COUNT (regno));
+      changed |= bitmap_and_into (&bb_info->out, live_defs);
+      BITMAP_FREE (live_defs);
+    }
+
+  return changed;
+}
 
 /* Free all storage associated with the problem.  */
 
   if (!df_rd->block_info)
     return;
 
-  fprintf (file, ";; Reaching defs:\n\n");
+  fprintf (file, ";; Reaching defs:\n");
 
-  fprintf (file, "  sparse invalidated \t");
+  fprintf (file, ";;  sparse invalidated \t");
   dump_bitmap (file, &problem_data->sparse_invalidated_by_call);
-  fprintf (file, "  dense invalidated \t");
+  fprintf (file, ";;  dense invalidated \t");
   dump_bitmap (file, &problem_data->dense_invalidated_by_call);
 
+  fprintf (file, ";;  reg->defs[] map:\t");
   for (regno = 0; regno < m; regno++)
     if (DF_DEFS_COUNT (regno))
       fprintf (file, "%d[%d,%d] ", regno,
               DF_DEFS_BEGIN (regno),
-              DF_DEFS_COUNT (regno));
+              DF_DEFS_BEGIN (regno) + DF_DEFS_COUNT (regno) - 1);
   fprintf (file, "\n");
-
 }
 
 
+static void
+df_rd_dump_defs_set (bitmap defs_set, const char *prefix, FILE *file)
+{
+  bitmap_head tmp;
+  unsigned int regno;
+  unsigned int m = DF_REG_SIZE(df);
+  bool first_reg = true;
+
+  fprintf (file, "%s\t(%d) ", prefix, (int) bitmap_count_bits (defs_set));
+
+  bitmap_initialize (&tmp, &df_bitmap_obstack);
+  for (regno = 0; regno < m; regno++)
+    {
+      if (HARD_REGISTER_NUM_P (regno)
+         && (df->changeable_flags & DF_NO_HARD_REGS))
+       continue;
+      bitmap_set_range (&tmp, DF_DEFS_BEGIN (regno), DF_DEFS_COUNT (regno));
+      bitmap_and_into (&tmp, defs_set);
+      if (! bitmap_empty_p (&tmp))
+       {
+         bitmap_iterator bi;
+         unsigned int ix;
+         bool first_def = true;
+
+         if (! first_reg)
+           fprintf (file, ",");
+         first_reg = false;
+
+         fprintf (file, "%u[", regno);
+         EXECUTE_IF_SET_IN_BITMAP (&tmp, 0, ix, bi)
+           {
+             fprintf (file, "%s%u", first_def ? "" : ",", ix);
+             first_def = false;
+           }
+         fprintf (file, "]");
+       }
+      bitmap_clear (&tmp);
+    }
+
+  fprintf (file, "\n");
+  bitmap_clear (&tmp);
+}
+
 /* Debugging info at top of bb.  */
 
 static void
   if (!bb_info)
     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));
-  dump_bitmap (file, &bb_info->gen);
-  fprintf (file, ";; rd  kill\t(%d)\n", (int) bitmap_count_bits (&bb_info->kill));
-  dump_bitmap (file, &bb_info->kill);
+  df_rd_dump_defs_set (&bb_info->in, ";; rd  in  ", file);
+  df_rd_dump_defs_set (&bb_info->gen, ";; rd  gen ", file);
+  df_rd_dump_defs_set (&bb_info->kill, ";; rd  kill", file);
 }
 
 
-/* Debugging info at top of bb.  */
+/* Debugging info at bottom of bb.  */
 
 static void
 df_rd_bottom_dump (basic_block bb, FILE *file)
   if (!bb_info)
     return;
 
-  fprintf (file, ";; rd  out \t(%d)\n", (int) bitmap_count_bits (&bb_info->out));
-  dump_bitmap (file, &bb_info->out);
+  df_rd_dump_defs_set (&bb_info->out, ";; rd  out ", file);
 }
 
 /* All of the information associated with every instance of the problem.  */
   df_rd_start_dump,           /* Debugging.  */
   df_rd_top_dump,             /* Debugging start block.  */
   df_rd_bottom_dump,          /* Debugging end block.  */
+  NULL,                       /* Debugging start insn.  */
+  NULL,                       /* Debugging end insn.  */
   NULL,                       /* Incremental solution verify start.  */
   NULL,                       /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
   NULL,                       /* Debugging.  */
   df_lr_top_dump,             /* Debugging start block.  */
   df_lr_bottom_dump,          /* Debugging end block.  */
+  NULL,                       /* Debugging start insn.  */
+  NULL,                       /* Debugging end insn.  */
   df_lr_verify_solution_start,/* Incremental solution verify start.  */
   df_lr_verify_solution_end,  /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */
   NULL,                         /* Debugging.  */
   df_live_top_dump,             /* Debugging start block.  */
   df_live_bottom_dump,          /* Debugging end block.  */
+  NULL,                         /* Debugging start insn.  */
+  NULL,                         /* Debugging end insn.  */
   df_live_verify_solution_start,/* Incremental solution verify start.  */
   df_live_verify_solution_end,  /* Incremental solution verify end.  */
   &problem_LR,                  /* Dependent problem.  */
 /* Debugging info.  */
 
 static void
-df_chain_top_dump (basic_block bb, FILE *file)
+df_chain_bb_dump (basic_block bb, FILE *file, bool top)
 {
+  /* Artificials are only hard regs.  */
+  if (df->changeable_flags & DF_NO_HARD_REGS)
+    return;
+  if (df_chain_problem_p (DF_UD_CHAIN))
+    {
+      fprintf (file,
+              ";;  UD chains for artificial uses at %s\n",
+              top ? "top" : "bottom");
+      df_ref *use_rec = df_get_artificial_uses (bb->index);
+      if (*use_rec)
+       {
+         while (*use_rec)
+           {
+             df_ref use = *use_rec;
+             if ((top && (DF_REF_FLAGS (use) & DF_REF_AT_TOP))
+                 || (!top && !(DF_REF_FLAGS (use) & DF_REF_AT_TOP)))
+               {
+                 fprintf (file, ";;   reg %d ", DF_REF_REGNO (use));
+                 df_chain_dump (DF_REF_CHAIN (use), file);
+                 fprintf (file, "\n");
+               }
+             use_rec++;
+           }
+       }
+    }
   if (df_chain_problem_p (DF_DU_CHAIN))
     {
-      rtx insn;
+      fprintf (file,
+              ";;  DU chains for artificial defs at %s\n",
+              top ? "top" : "bottom");
       df_ref *def_rec = df_get_artificial_defs (bb->index);
       if (*def_rec)
        {
-
-         fprintf (file, ";;  DU chains for artificial defs\n");
          while (*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++;
-           }
-       }
 
-      FOR_BB_INSNS (bb, insn)
-       {
-         if (INSN_P (insn))
-           {
-             struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
-             def_rec = DF_INSN_INFO_DEFS (insn_info);
-             if (*def_rec)
+             if ((top && (DF_REF_FLAGS (def) & DF_REF_AT_TOP))
+                 || (!top && !(DF_REF_FLAGS (def) & DF_REF_AT_TOP)))
                {
-                 fprintf (file, ";;   DU chains for insn luid %d uid %d\n",
-                          DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
-
-                 while (*def_rec)
-                   {
-                     df_ref def = *def_rec;
-                     fprintf (file, ";;      reg %d ", DF_REF_REGNO (def));
-                     if (DF_REF_FLAGS (def) & DF_REF_READ_WRITE)
-                       fprintf (file, "read/write ");
-                     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++;
            }
        }
     }
 }
 
+static void
+df_chain_top_dump (basic_block bb, FILE *file)
+{
+  df_chain_bb_dump (bb, file, /*top=*/true);
+}
 
 static void
 df_chain_bottom_dump (basic_block bb, FILE *file)
 {
-  if (df_chain_problem_p (DF_UD_CHAIN))
-    {
-      rtx insn;
-      df_ref *use_rec = df_get_artificial_uses (bb->index);
+  df_chain_bb_dump (bb, file, /*top=*/false);
+}
 
-      if (*use_rec)
+static void
+df_chain_insn_top_dump (const_rtx insn, FILE *file)
+{
+  if (df_chain_problem_p (DF_UD_CHAIN) && INSN_P (insn))
+    {
+      struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
+      df_ref *use_rec = DF_INSN_INFO_USES (insn_info);
+      df_ref *eq_use_rec = DF_INSN_INFO_EQ_USES (insn_info);
+      fprintf (file, ";;   UD chains for insn luid %d uid %d\n",
+              DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
+      if (*use_rec || *eq_use_rec)
        {
-         fprintf (file, ";;  UD chains for artificial uses\n");
          while (*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");
+             if (! HARD_REGISTER_NUM_P (DF_REF_REGNO (use))
+                 || !(df->changeable_flags & DF_NO_HARD_REGS))
+               {
+                 fprintf (file, ";;      reg %d ", DF_REF_REGNO (use));
+                 if (DF_REF_FLAGS (use) & DF_REF_READ_WRITE)
+                   fprintf (file, "read/write ");
+                 df_chain_dump (DF_REF_CHAIN (use), file);
+                 fprintf (file, "\n");
+               }
              use_rec++;
            }
+         while (*eq_use_rec)
+           {
+             df_ref use = *eq_use_rec;
+             if (! HARD_REGISTER_NUM_P (DF_REF_REGNO (use))
+                 || !(df->changeable_flags & DF_NO_HARD_REGS))
+               {
+                 fprintf (file, ";;   eq_note reg %d ", DF_REF_REGNO (use));
+                 df_chain_dump (DF_REF_CHAIN (use), file);
+                 fprintf (file, "\n");
+               }
+             eq_use_rec++;
+           }
        }
+    }
+}
 
-      FOR_BB_INSNS (bb, insn)
+static void
+df_chain_insn_bottom_dump (const_rtx insn, FILE *file)
+{
+  if (df_chain_problem_p (DF_DU_CHAIN) && INSN_P (insn))
+    {
+      struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
+      df_ref *def_rec = DF_INSN_INFO_DEFS (insn_info);
+      fprintf (file, ";;   DU chains for insn luid %d uid %d\n",
+              DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
+      if (*def_rec)
        {
-         if (INSN_P (insn))
+         while (*def_rec)
            {
-             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)
+             df_ref def = *def_rec;
+             if (! HARD_REGISTER_NUM_P (DF_REF_REGNO (def))
+                 || !(df->changeable_flags & DF_NO_HARD_REGS))
                {
-                 fprintf (file, ";;   UD chains for insn luid %d uid %d\n",
-                          DF_INSN_INFO_LUID (insn_info), INSN_UID (insn));
-
-                 while (*use_rec)
-                   {
-                     df_ref use = *use_rec;
-                     fprintf (file, ";;      reg %d ", DF_REF_REGNO (use));
-                     if (DF_REF_FLAGS (use) & DF_REF_READ_WRITE)
-                       fprintf (file, "read/write ");
-                     df_chain_dump (DF_REF_CHAIN (use), file);
-                     fprintf (file, "\n");
-                     use_rec++;
-                   }
-                 while (*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");
-                     eq_use_rec++;
-                   }
+                 fprintf (file, ";;      reg %d ", DF_REF_REGNO (def));
+                 if (DF_REF_FLAGS (def) & DF_REF_READ_WRITE)
+                   fprintf (file, "read/write ");
+                 df_chain_dump (DF_REF_CHAIN (def), file);
+                 fprintf (file, "\n");
                }
+             def_rec++;
            }
        }
+      fprintf (file, "\n");
     }
 }
 
-
 static struct df_problem problem_CHAIN =
 {
   DF_CHAIN,                   /* Problem id.  */
   NULL,                       /* Debugging.  */
   df_chain_top_dump,          /* Debugging start block.  */
   df_chain_bottom_dump,       /* Debugging end block.  */
+  df_chain_insn_top_dump,     /* Debugging start insn.  */
+  df_chain_insn_bottom_dump,  /* Debugging end insn.  */
   NULL,                       /* Incremental solution verify start.  */
   NULL,                       /* Incremental solution verify end.  */
   &problem_RD,                /* Dependent problem.  */
   NULL,                            /* Debugging.  */
   df_word_lr_top_dump,             /* Debugging start block.  */
   df_word_lr_bottom_dump,          /* Debugging end block.  */
+  NULL,                            /* Debugging start insn.  */
+  NULL,                            /* Debugging end insn.  */
   NULL,                            /* Incremental solution verify start.  */
   NULL,                            /* Incremental solution verify end.  */
-  NULL,                       /* Dependent problem.  */
+  NULL,                            /* Dependent problem.  */
   sizeof (struct df_word_lr_bb_info),/* Size of entry of block_info array.  */
   TV_DF_WORD_LR,                   /* Timing variable.  */
   false                            /* Reset blocks on dropping out of blocks_to_analyze.  */
   NULL,                       /* Debugging.  */
   NULL,                       /* Debugging start block.  */
   NULL,                       /* Debugging end block.  */
+  NULL,                       /* Debugging start insn.  */
+  NULL,                       /* Debugging end insn.  */
   NULL,                       /* Incremental solution verify start.  */
   NULL,                       /* Incremental solution verify end.  */
   &problem_LR,                /* Dependent problem.  */
   NULL,                       /* Debugging.  */
   df_md_top_dump,             /* Debugging start block.  */
   df_md_bottom_dump,          /* Debugging end block.  */
+  NULL,                       /* Debugging start insn.  */
+  NULL,                       /* Debugging end insn.  */
   NULL,                              /* Incremental solution verify start.  */
   NULL,                              /* Incremental solution verify end.  */
   NULL,                       /* Dependent problem.  */