Split GenerateConstraintsForAssignment in a Recursive and a NonRecursive part.

PiperOrigin-RevId: 479004659
diff --git a/lifetime_analysis/lifetime_analysis.cc b/lifetime_analysis/lifetime_analysis.cc
index 2e7d4e5..f215b7d 100644
--- a/lifetime_analysis/lifetime_analysis.cc
+++ b/lifetime_analysis/lifetime_analysis.cc
@@ -100,8 +100,31 @@
   const DiagnosticReporter& diag_reporter_;
 };
 
+void GenerateConstraintsForAssignmentNonRecursive(
+    const ObjectSet& old_pointees, const ObjectSet& new_pointees,
+    bool is_in_invariant_context, LifetimeConstraints& constraints) {
+  // The new pointees must always outlive the old pointees.
+  for (const Object* old : old_pointees) {
+    for (const Object* newp : new_pointees) {
+      constraints.AddOutlivesConstraint(old->GetLifetime(),
+                                        newp->GetLifetime());
+    }
+  }
+
+  // If we are in an invariant context, we need to insert constraints in the
+  // opposite direction too (i.e. we need equality).
+  if (is_in_invariant_context) {
+    for (const Object* old : old_pointees) {
+      for (const Object* newp : new_pointees) {
+        constraints.AddOutlivesConstraint(newp->GetLifetime(),
+                                          old->GetLifetime());
+      }
+    }
+  }
+}
+
 // TODO(veluca): this is quadratic.
-void GenerateConstraintsForAssignmentImpl(
+void GenerateConstraintsForAssignmentRecursive(
     const ObjectSet& pointers, const ObjectSet& new_pointees,
     clang::QualType pointer_type, const ObjectRepository& object_repository,
     const PointsToMap& points_to_map, bool is_in_invariant_context,
@@ -131,24 +154,8 @@
 
   ObjectSet old_pointees = points_to_map.GetPointerPointsToSet(pointers);
 
-  // The new pointees must always outlive the old pointees.
-  for (const Object* old : old_pointees) {
-    for (const Object* newp : new_pointees) {
-      constraints.AddOutlivesConstraint(old->GetLifetime(),
-                                        newp->GetLifetime());
-    }
-  }
-
-  // If we are in an invariant context, we need to insert constraints in the
-  // opposite direction too (i.e. we need equality).
-  if (is_in_invariant_context) {
-    for (const Object* old : old_pointees) {
-      for (const Object* newp : new_pointees) {
-        constraints.AddOutlivesConstraint(newp->GetLifetime(),
-                                          old->GetLifetime());
-      }
-    }
-  }
+  GenerateConstraintsForAssignmentNonRecursive(
+      old_pointees, new_pointees, is_in_invariant_context, constraints);
 
   // See https://doc.rust-lang.org/nomicon/subtyping.html for an explanation of
   // variance; here in particular, we use the fact that the pointee of a pointer
@@ -194,7 +201,7 @@
         }
       }
     } else {
-      GenerateConstraintsForAssignmentImpl(
+      GenerateConstraintsForAssignmentRecursive(
           call.old_pointees,
           points_to_map.GetPointerPointsToSet(call.new_pointees), call.type,
           object_repository, points_to_map, is_in_invariant_context,
@@ -211,7 +218,7 @@
                                       LifetimeConstraints& constraints) {
   llvm::DenseSet<std::pair<const Object*, const Object*>> seen_pairs;
   // Outer-most pointers are never invariant.
-  GenerateConstraintsForAssignmentImpl(
+  GenerateConstraintsForAssignmentRecursive(
       pointers, new_pointees, pointer_type, object_repository, points_to_map,
       /*is_in_invariant_context=*/false, constraints, seen_pairs);
 }