Only overwrite destination points-to-set if we have a single object on the LHS.

Adds multiple tests; of these, Overwrite_MultipleDestinations fails before this
change.

PiperOrigin-RevId: 453662886
diff --git a/lifetime_analysis/test/basic.cc b/lifetime_analysis/test/basic.cc
index d442510..d9f2f84 100644
--- a/lifetime_analysis/test/basic.cc
+++ b/lifetime_analysis/test/basic.cc
@@ -456,6 +456,54 @@
               LifetimesAre({{"f", "a -> a"}, {"target", "a -> a"}}));
 }
 
+TEST_F(LifetimeAnalysisTest, Overwrite_SingleDestination) {
+  EXPECT_THAT(GetLifetimes(R"(
+    int* target(int* a, int* b) {
+      int** pp = &b;
+      // There is only one thing that `pp` can be pointing at, so the analysis
+      // should conclude that `b` is being overwritten with `a`.
+      *pp = a;
+      return b;
+    }
+  )"),
+              LifetimesAre({{"target", "a, b -> a"}}));
+}
+
+TEST_F(LifetimeAnalysisTest, Overwrite_SingleDestinationVariant) {
+  EXPECT_THAT(GetLifetimes(R"(
+    // Similar to above, but potentially leave `pp` uninitialized.
+    int* target(int* a, int* b) {
+      int** pp;
+      if (*a > 0) {
+        pp = &b;
+      }
+      // If `pp` is uninitialized, the following is UB, so the analysis can
+      // assume that `pp` was initialized to point to `b`.
+      // This particular test function is pretty terrible style, but it seems
+      // plausible that similar situations can come up in more reasonable code.
+      *pp = a;
+      return b;
+    }
+  )"),
+              LifetimesAre({{"target", "a, b -> a"}}));
+}
+
+TEST_F(LifetimeAnalysisTest, Overwrite_MultipleDestinations) {
+  EXPECT_THAT(GetLifetimes(R"(
+    // This is a regression test. The analysis used to conclude falsely that `b`
+    // was unconditionally being overwritten with `a` in the assignment and was
+    // therefore producing the wrong lifetimes "a, b -> a".
+    int* target(int* a, int* b) {
+      int** pp = *a > 0? &a : &b;
+      // The analysis should understand that the following assignment _might_
+      // overwrite `b` with `a` but does not necessarily do so.
+      *pp = a;
+      return b;
+    }
+  )"),
+              LifetimesAre({{"target", "a, a -> a"}}));
+}
+
 }  // namespace
 }  // namespace lifetimes
 }  // namespace tidy