Merge "Add synchronization to LambdaClass.synthesizeLambdaClass"
diff --git a/src/main/java/com/android/tools/r8/dex/FileWriter.java b/src/main/java/com/android/tools/r8/dex/FileWriter.java
index 6ac3826..f09d92a 100644
--- a/src/main/java/com/android/tools/r8/dex/FileWriter.java
+++ b/src/main/java/com/android/tools/r8/dex/FileWriter.java
@@ -615,7 +615,7 @@
   }
 
   private void addStaticFieldValues(DexProgramClass clazz) {
-    clazz.computeStaticValues(application.dexItemFactory);
+    clazz.computeStaticValues();
     // We have collected the individual components of this array due to the data stored in
     // DexEncodedField#staticValues. However, we have to collect the DexEncodedArray itself
     // here.
diff --git a/src/main/java/com/android/tools/r8/graph/DexProgramClass.java b/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
index 837b805..51fb04a 100644
--- a/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
+++ b/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
@@ -16,7 +16,6 @@
 import java.util.Comparator;
 import java.util.HashSet;
 import java.util.List;
-import java.util.Set;
 import java.util.function.Supplier;
 
 public class DexProgramClass extends DexClass implements Supplier<DexProgramClass> {
@@ -99,7 +98,8 @@
         skipNameValidationForTesting);
     assert classAnnotations != null;
     this.originKind = originKind;
-    this.synthesizedFrom = accumulateSynthesizedFrom(new HashSet<>(), synthesizedDirectlyFrom);
+    this.synthesizedFrom = new HashSet<>();
+    synthesizedDirectlyFrom.forEach(this::addSynthesizedFrom);
   }
 
   public boolean originatesFromDexResource() {
@@ -266,20 +266,15 @@
     }
   }
 
-  private static Collection<DexProgramClass> accumulateSynthesizedFrom(
-      Set<DexProgramClass> accumulated,
-      Collection<DexProgramClass> toAccumulate) {
-    for (DexProgramClass dexProgramClass : toAccumulate) {
-      if (dexProgramClass.synthesizedFrom.isEmpty()) {
-        accumulated.add(dexProgramClass);
-      } else {
-        accumulateSynthesizedFrom(accumulated, dexProgramClass.synthesizedFrom);
-      }
+  public void addSynthesizedFrom(DexProgramClass clazz) {
+    if (clazz.synthesizedFrom.isEmpty()) {
+      synthesizedFrom.add(clazz);
+    } else {
+      clazz.synthesizedFrom.forEach(this::addSynthesizedFrom);
     }
-    return accumulated;
   }
 
-  public void computeStaticValues(DexItemFactory factory) {
+  public void computeStaticValues() {
     // It does not actually hurt to compute this multiple times. So racing on staticValues is OK.
     if (staticValues == SENTINEL_NOT_YET_COMPUTED) {
       synchronized (staticFields) {
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java b/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
index 5e18649..03a3818 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
@@ -65,7 +65,7 @@
   final DexField instanceField;
   final Target target;
   final AtomicBoolean addToMainDexList = new AtomicBoolean(false);
-  private final Collection<DexProgramClass> synthesizedFrom = new ArrayList<DexProgramClass>(1);
+  private final Collection<DexProgramClass> synthesizedFrom = new ArrayList<>(1);
   private final Supplier<DexProgramClass> lazyDexClass =
       Suppliers.memoize(this::synthesizeLambdaClass); // NOTE: thread-safe.
 
@@ -128,26 +128,32 @@
   }
 
   private DexProgramClass synthesizeLambdaClass() {
-    return new DexProgramClass(
-        type,
-        null,
-        new SynthesizedOrigin("lambda desugaring", getClass()),
-        // Make the synthesized class public, as it might end up being accessed from a different
-        // classloader (package private access is not allowed across classloaders, b/72538146).
-        ClassAccessFlags.fromDexAccessFlags(
-            Constants.ACC_FINAL | Constants.ACC_SYNTHETIC | Constants.ACC_PUBLIC),
-        rewriter.factory.objectType,
-        buildInterfaces(),
-        rewriter.factory.createString("lambda"),
-        null,
-        Collections.emptyList(),
-        DexAnnotationSet.empty(),
-        synthesizeStaticFields(),
-        synthesizeInstanceFields(),
-        synthesizeDirectMethods(),
-        synthesizeVirtualMethods(),
-        rewriter.factory.getSkipNameValidationForTesting(),
-        synthesizedFrom);
+    DexProgramClass clazz =
+        new DexProgramClass(
+            type,
+            null,
+            new SynthesizedOrigin("lambda desugaring", getClass()),
+            // Make the synthesized class public, as it might end up being accessed from a different
+            // classloader (package private access is not allowed across classloaders, b/72538146).
+            ClassAccessFlags.fromDexAccessFlags(
+                Constants.ACC_FINAL | Constants.ACC_SYNTHETIC | Constants.ACC_PUBLIC),
+            rewriter.factory.objectType,
+            buildInterfaces(),
+            rewriter.factory.createString("lambda"),
+            null,
+            Collections.emptyList(),
+            DexAnnotationSet.empty(),
+            synthesizeStaticFields(),
+            synthesizeInstanceFields(),
+            synthesizeDirectMethods(),
+            synthesizeVirtualMethods(),
+            rewriter.factory.getSkipNameValidationForTesting());
+    // The method addSynthesizedFrom() may be called concurrently. To avoid a Concurrent-
+    // ModificationException we must use synchronization.
+    synchronized (synthesizedFrom) {
+      synthesizedFrom.forEach(clazz::addSynthesizedFrom);
+    }
+    return clazz;
   }
 
   final DexField getCaptureField(int index) {
@@ -159,9 +165,15 @@
     return descriptor.isStateless();
   }
 
-  synchronized void addSynthesizedFrom(DexProgramClass synthesizedFrom) {
-    assert synthesizedFrom != null;
-    this.synthesizedFrom.add(synthesizedFrom);
+  void addSynthesizedFrom(DexProgramClass clazz) {
+    assert clazz != null;
+    synchronized (synthesizedFrom) {
+      if (synthesizedFrom.add(clazz)) {
+        // The lambda class may already have been synthesized, and we therefore need to update the
+        // synthesized lambda class as well.
+        getLambdaClass().addSynthesizedFrom(clazz);
+      }
+    }
   }
 
   // Synthesize virtual methods.
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java b/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
index ffca4d1..9d475c3 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
@@ -242,13 +242,13 @@
     return lambdaClass;
   }
 
-  private <K, V> V getKnown(Map<K, V> map, K key) {
+  private static <K, V> V getKnown(Map<K, V> map, K key) {
     synchronized (map) {
       return map.get(key);
     }
   }
 
-  private <K, V> V putIfAbsent(Map<K, V> map, K key, V value) {
+  private static <K, V> V putIfAbsent(Map<K, V> map, K key, V value) {
     synchronized (map) {
       V known = map.get(key);
       if (known != null) {