Update task to sanity check links in release notes and samples + Cloudflare redirects (#31491)
diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
index 5e693dd..9e40988 100644
--- a/gradle/wrapper/gradle-wrapper.properties
+++ b/gradle/wrapper/gradle-wrapper.properties
@@ -1,6 +1,6 @@
 distributionBase=GRADLE_USER_HOME
 distributionPath=wrapper/dists
-distributionUrl=https\://services.gradle.org/distributions-snapshots/gradle-8.12-20241126002544+0000-bin.zip
+distributionUrl=https\://services.gradle.org/distributions/gradle-8.12-rc-1-bin.zip
 networkTimeout=10000
 validateDistributionUrl=true
 zipStoreBase=GRADLE_USER_HOME
diff --git a/gradlew b/gradlew
index d95bf61..057afac 100755
--- a/gradlew
+++ b/gradlew
@@ -86,8 +86,7 @@
 # shellcheck disable=SC2034
 APP_BASE_NAME=${0##*/}
 # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
-APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
-' "$PWD" ) || exit
+APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit
 
 # Use the maximum available, or set MAX_FD != -1 to use that value.
 MAX_FD=maximum
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/AbstractCollectionProperty.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/AbstractCollectionProperty.java
index 649524c..444ad2b 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/AbstractCollectionProperty.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/AbstractCollectionProperty.java
@@ -19,16 +19,20 @@
 import com.google.common.base.Preconditions;
 import com.google.common.collect.ImmutableCollection;
 import com.google.common.collect.ImmutableList;
+import com.google.common.collect.Lists;
 import org.gradle.api.Action;
+import org.gradle.api.Task;
 import org.gradle.api.Transformer;
 import org.gradle.api.internal.provider.Collectors.ElementFromProvider;
 import org.gradle.api.internal.provider.Collectors.ElementsFromArray;
 import org.gradle.api.internal.provider.Collectors.ElementsFromCollection;
 import org.gradle.api.internal.provider.Collectors.ElementsFromCollectionProvider;
 import org.gradle.api.internal.provider.Collectors.SingleElement;
+import org.gradle.api.internal.tasks.TaskDependencyResolveContext;
 import org.gradle.api.provider.HasMultipleValues;
 import org.gradle.api.provider.Provider;
 import org.gradle.internal.Cast;
+import org.gradle.internal.Pair;
 import org.gradle.internal.evaluation.EvaluationScopeContext;
 
 import javax.annotation.Nonnull;
@@ -38,14 +42,18 @@
 import java.util.List;
 import java.util.Locale;
 import java.util.function.Supplier;
+import java.util.stream.Stream;
+
+import static java.util.stream.Collectors.toCollection;
+import static java.util.stream.Collectors.toList;
 
 /**
  * The base class for collection properties.
  * <p>
- *     Value suppliers for collection properties are implementations of {@link CollectionSupplier}.
+ * Value suppliers for collection properties are implementations of {@link CollectionSupplier}.
  * </p>
  * <p>
- *     Elements stored in collection property values are implemented via various implementations of {@link Collector}.
+ * Elements stored in collection property values are implemented via various implementations of {@link Collector}.
  * </p>
  * <h2>Collection suppliers</h2>
  * The value of a collection property is represented at any time as an instance of an implementation of {@link CollectionSupplier}, namely:
@@ -70,8 +78,8 @@
  *         <li>{@link ElementsFromCollection} to represent a batch of elements added (or set wholesale) as an <code>Iterable</code>
  *         <li>{@link ElementsFromCollectionProvider} to represent a batch of elements added (or set wholesale) as a provider of <code>Iterable</code>
  *     </ul>
- * <p>Also, if a collection is built up via multiple additions, which is quite common, after each addition operation, its value will be represented via a new {@link PlusCollector} instance
- * that references the previous value as the {@link PlusCollector#left left side}, and the added element(s) as {@link PlusCollector#right right side} of the operation.
+ * <p>Also, if a collection is built up via multiple additions, which is quite common, after each addition operation, its value will be represented via a new {@link CollectionSupplier} instance.
+ * Each addition operation adds an individual collector to the shared underlying append-only list of collectors.
  * </p>
  *
  * @param <T> the type of element this collection property can hold
@@ -210,14 +218,14 @@ public int size() {
      */
     private void addExplicitCollector(Collector<T> collector, boolean ignoreAbsent) {
         assertCanMutate();
-        CollectionSupplier<T, C> explicitValue = getExplicitValue(defaultValue).absentIgnoringIfNeeded(ignoreAbsent);
-        setSupplier(explicitValue.plus(collector.absentIgnoringIfNeeded(ignoreAbsent)));
+        CollectionSupplier<T, C> explicitValue = getExplicitValue(defaultValue);
+        setSupplier(explicitValue.plus(collector, ignoreAbsent));
     }
 
-    @Nullable
     @Override
+    @Nonnull
     public Class<C> getType() {
-        return Cast.uncheckedCast(collectionType);
+        return Cast.uncheckedNonnullCast(collectionType);
     }
 
     @Override
@@ -234,8 +242,8 @@ public void fromState(ExecutionTimeValue<? extends C> value) {
         } else if (value.hasFixedValue()) {
             setSupplier(new FixedSupplier(value.getFixedValue(), Cast.uncheckedCast(value.getSideEffect())));
         } else {
-            CollectingProvider<T, C> asCollectingProvider = Cast.uncheckedNonnullCast(value.getChangingValue());
-            setSupplier(new CollectingSupplier(new ElementsFromCollectionProvider<>(asCollectingProvider)));
+            CollectingSupplier<T, C> asSupplier = Cast.uncheckedNonnullCast(value.getChangingValue());
+            setSupplier(asSupplier);
         }
     }
 
@@ -256,7 +264,7 @@ public void set(@Nullable final Iterable<? extends T> elements) {
         if (elements == null) {
             unsetValueAndDefault();
         } else {
-            setSupplier(new CollectingSupplier(new ElementsFromCollection<>(elements)));
+            setSupplier(newSupplierOf(new ElementsFromCollection<>(elements)));
         }
     }
 
@@ -275,7 +283,7 @@ public void set(final Provider<? extends Iterable<? extends T>> provider) {
                 throw new IllegalArgumentException(String.format("Cannot set the value of a property of type %s with element type %s using a provider with element type %s.", collectionType.getName(), elementType.getName(), collectionProp.getElementType().getName()));
             }
         }
-        setSupplier(new CollectingSupplier(new ElementsFromCollectionProvider<>(p)));
+        setSupplier(newSupplierOf(new ElementsFromCollectionProvider<>(p)));
     }
 
     private void unsetValueAndDefault() {
@@ -329,14 +337,14 @@ public HasMultipleValues<T> convention(@Nullable Iterable<? extends T> elements)
         if (elements == null) {
             unsetConvention();
         } else {
-            setConvention(new CollectingSupplier(new ElementsFromCollection<>(elements)));
+            setConvention(newSupplierOf(new ElementsFromCollection<>(elements)));
         }
         return this;
     }
 
     @Override
     public HasMultipleValues<T> convention(Provider<? extends Iterable<? extends T>> provider) {
-        setConvention(new CollectingSupplier(new ElementsFromCollectionProvider<>(Providers.internal(provider))));
+        setConvention(newSupplierOf(new ElementsFromCollectionProvider<>(Providers.internal(provider))));
         return this;
     }
 
@@ -355,11 +363,6 @@ public NoValueSupplier(Value<? extends C> value) {
         }
 
         @Override
-        public CollectionSupplier<T, C> absentIgnoring() {
-            return Cast.uncheckedCast(emptySupplier());
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return false;
         }
@@ -370,9 +373,9 @@ public Value<? extends C> calculateValue(ValueConsumer consumer) {
         }
 
         @Override
-        public CollectionSupplier<T, C> plus(Collector<T> collector) {
-            // No value + something = no value
-            return this;
+        public CollectionSupplier<T, C> plus(Collector<T> collector, boolean ignoreAbsent) {
+            // No value + something = no value, unless we ignoreAbsent.
+            return ignoreAbsent ? newSupplierOf(ignoreAbsentIfNeeded(collector, ignoreAbsent)) : this;
         }
 
         @Override
@@ -404,14 +407,9 @@ public Value<? extends C> calculateValue(ValueConsumer consumer) {
         }
 
         @Override
-        public CollectionSupplier<T, C> plus(Collector<T> collector) {
+        public CollectionSupplier<T, C> plus(Collector<T> collector, boolean ignoreAbsent) {
             // empty + something = something
-            return new CollectingSupplier(collector);
-        }
-
-        @Override
-        public CollectionSupplier<T, C> absentIgnoring() {
-            return this;
+            return newSupplierOf(ignoreAbsentIfNeeded(collector, ignoreAbsent));
         }
 
         @Override
@@ -440,11 +438,6 @@ public FixedSupplier(C value, @Nullable SideEffect<? super C> sideEffect) {
         }
 
         @Override
-        public CollectionSupplier<T, C> absentIgnoring() {
-            return this;
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return true;
         }
@@ -455,10 +448,8 @@ public Value<? extends C> calculateValue(ValueConsumer consumer) {
         }
 
         @Override
-        public CollectionSupplier<T, C> plus(Collector<T> collector) {
-            Collector<T> left = new FixedValueCollector<>(value, sideEffect);
-            PlusCollector<T> newCollector = new PlusCollector<>(left, collector);
-            return new CollectingSupplier(newCollector);
+        public CollectionSupplier<T, C> plus(Collector<T> collector, boolean ignoreAbsent) {
+            return newSupplierOf(new FixedValueCollector<>(value, sideEffect)).plus(collector, ignoreAbsent);
         }
 
         @Override
@@ -477,58 +468,128 @@ public String toString() {
         }
     }
 
-    private class CollectingSupplier implements CollectionSupplier<T, C> {
-        private final Collector<T> value;
-        // TODO-RC: can we get rid of this? Can we only keep this in Collectors? Changing execution time value is the only case that needs this.
-        private final boolean ignoreAbsent;
+    private CollectingSupplier<T, C> newSupplierOf(Collector<T> value) {
+        return new CollectingSupplier<>(getType(), collectionFactory, valueCollector, value);
+    }
 
-        public CollectingSupplier(Collector<T> value, boolean ignoreAbsent) {
-            this.value = value;
-            this.ignoreAbsent = ignoreAbsent;
+    private static class CollectingSupplier<T, C extends Collection<T>> extends AbstractMinimalProvider<C> implements CollectionSupplier<T, C> {
+        private final Class<C> type;
+        private final Supplier<ImmutableCollection.Builder<T>> collectionFactory;
+        private final ValueCollector<T> valueCollector;
+        // This list is shared by the collectors produced by `plus`, so we don't have to copy the collectors every time.
+        // However, this also means that you can only call plus on a given collector once.
+        private final ArrayList<Collector<T>> collectors; // TODO - Replace with PersistentList? This may make value calculation inefficient because the PersistentList can only prepend to head.
+        private final int size;
+
+        public CollectingSupplier(Class<C> type, Supplier<ImmutableCollection.Builder<T>> collectionFactory, ValueCollector<T> valueCollector, Collector<T> value) {
+            this(type, collectionFactory, valueCollector, Lists.newArrayList(value), 1);
         }
 
-        public CollectingSupplier(Collector<T> value) {
-            this(value, false);
+        // A constructor for sharing.
+        private CollectingSupplier(
+            Class<C> type,
+            Supplier<ImmutableCollection.Builder<T>> collectionFactory,
+            ValueCollector<T> valueCollector,
+            @SuppressWarnings("NonApiType") ArrayList<Collector<T>> collectors,
+            int size
+        ) {
+            this.type = type;
+            this.collectionFactory = collectionFactory;
+            this.valueCollector = valueCollector;
+            this.collectors = collectors;
+            this.size = size;
+        }
+
+        @Override
+        protected Value<? extends C> calculateOwnValue(ValueConsumer consumer) {
+            return calculateValue(consumer);
+        }
+
+        @Nullable
+        @Override
+        public Class<C> getType() {
+            return type;
         }
 
         @Override
         public boolean calculatePresence(ValueConsumer consumer) {
-            return value.calculatePresence(consumer);
+            // We're traversing the elements in reverse addition order.
+            // When determining the presence of the value, the last argument wins.
+            // See also #collectExecutionTimeValues().
+            for (Collector<T> collector : Lists.reverse(getCollectors())) {
+                if (!collector.calculatePresence(consumer)) {
+                    // We've found an argument of add/addAll that is missing.
+                    // It makes the property missing regardless of what has been added before.
+                    // Because of the reverse processing order, anything that was added after it was just add/addAll that do not change the presence.
+                    return false;
+                }
+                if (isAbsentIgnoring(collector)) {
+                    // We've found an argument of append/appendAll, and everything added before it was present.
+                    // append/appendAll recovers the value of a missing property, so the property is also definitely present.
+                    return true;
+                }
+            }
+            // Nothing caused the property to become missing. There is at least one element by design, so the property is present.
+            assert size > 0;
+            return true;
         }
 
         @Override
         public Value<C> calculateValue(ValueConsumer consumer) {
             // TODO - don't make a copy when the collector already produces an immutable collection
             ImmutableCollection.Builder<T> builder = collectionFactory.get();
-            Value<Void> result = value.collectEntries(consumer, valueCollector, builder);
-            if (result.isMissing()) {
-                return result.asType();
+            Value<Void> compositeResult = Value.present();
+            for (Collector<T> collector : getCollectors()) {
+                if (compositeResult.isMissing() && !isAbsentIgnoring(collector)) {
+                    // The property is missing so far and the argument is of add/addAll.
+                    // The property is going to be missing regardless of its value.
+                    continue;
+                }
+                Value<Void> result = collector.collectEntries(consumer, valueCollector, builder);
+                if (result.isMissing()) {
+                    // This is the argument of add/addAll and it is missing. It "poisons" the property (it becomes missing).
+                    // We discard all values and side effects gathered so far.
+                    builder = collectionFactory.get();
+                    compositeResult = result;
+                } else if (compositeResult.isMissing()) {
+                    assert isAbsentIgnoring(collector);
+                    // This is an argument of append/appendAll. It "recovers" the property from the "poisoned" state.
+                    // Entries are already in the builder.
+                    compositeResult = result;
+                } else {
+                    assert !compositeResult.isMissing();
+                    // Both the property so far and the current argument are present, just continue building the value.
+                    // Entries are already in the builder.
+                    compositeResult = compositeResult.withSideEffect(SideEffect.fixedFrom(result));
+                }
             }
-            return Value.of(Cast.<C>uncheckedNonnullCast(builder.build())).withSideEffect(SideEffect.fixedFrom(result));
+            if (compositeResult.isMissing()) {
+                return compositeResult.asType();
+            }
+            return Value.of(Cast.<C>uncheckedNonnullCast(builder.build())).withSideEffect(SideEffect.fixedFrom(compositeResult));
         }
 
         @Override
-        public CollectionSupplier<T, C> plus(Collector<T> addedCollector) {
-            Collector<T> left = value.absentIgnoringIfNeeded(ignoreAbsent);
-            Collector<T> right = addedCollector;
-            PlusCollector<T> newCollector = new PlusCollector<>(left, right);
-            return new CollectingSupplier(newCollector);
-        }
-
-        @Override
-        public CollectionSupplier<T, C> absentIgnoring() {
-            return ignoreAbsent ? this : new CollectingSupplier(value, true);
+        public CollectionSupplier<T, C> plus(Collector<T> addedCollector, boolean ignoreAbsent) {
+            Preconditions.checkState(collectors.size() == size, "Something has been appended to this collector already");
+            collectors.add(ignoreAbsentIfNeeded(addedCollector, ignoreAbsent));
+            return new CollectingSupplier<>(type, collectionFactory, valueCollector, collectors, size + 1);
         }
 
         @Override
         public ExecutionTimeValue<? extends C> calculateExecutionTimeValue() {
-            List<ExecutionTimeValue<? extends Iterable<? extends T>>> values = collectExecutionTimeValues();
+            List<Pair<Collector<T>, ExecutionTimeValue<? extends Iterable<? extends T>>>> collectorsWithValues = collectExecutionTimeValues();
+            if (collectorsWithValues.isEmpty()) {
+                return ExecutionTimeValue.missing();
+            }
+            List<ExecutionTimeValue<? extends Iterable<? extends T>>> values = collectorsWithValues.stream().map(Pair::getRight).collect(toList());
+
             boolean fixed = true;
             boolean changingContent = false;
+
             for (ExecutionTimeValue<? extends Iterable<? extends T>> value : values) {
-                if (value.isMissing()) {
-                    return ExecutionTimeValue.missing();
-                }
+                assert !value.isMissing();
+
                 if (value.isChangingValue()) {
                     fixed = false;
                 } else if (value.hasChangingContent()) {
@@ -540,20 +601,66 @@ public ExecutionTimeValue<? extends C> calculateExecutionTimeValue() {
                 return getFixedExecutionTimeValue(values, changingContent);
             }
 
-            // At least one of the values is a changing value
-            List<ProviderInternal<? extends Iterable<? extends T>>> providers = new ArrayList<>(values.size());
-            for (ExecutionTimeValue<? extends Iterable<? extends T>> value : values) {
-                providers.add(value.toProvider());
-            }
-            // TODO - CollectionSupplier could be replaced with ProviderInternal, so this type and the collection provider can be merged
-            return ExecutionTimeValue.changingValue(new CollectingProvider<>(AbstractCollectionProperty.this.getType(), providers, collectionFactory));
+            // At least one of the values is a changing value. Simplify the provider.
+            return ExecutionTimeValue.changingValue(
+                new CollectingSupplier<>(
+                    type,
+                    collectionFactory,
+                    valueCollector,
+                    collectorsWithValues.stream().map(pair -> {
+                        Collector<T> elements = toCollector(pair.getRight());
+                        return ignoreAbsentIfNeeded(elements, isAbsentIgnoring(pair.getLeft()));
+                    }).collect(toCollection(ArrayList::new)),
+                    collectorsWithValues.size()
+                )
+            );
         }
 
-        @Nonnull
-        private List<ExecutionTimeValue<? extends Iterable<? extends T>>> collectExecutionTimeValues() {
-            List<ExecutionTimeValue<? extends Iterable<? extends T>>> values = new ArrayList<>();
-            value.calculateExecutionTimeValue(values::add);
-            return values;
+        private Collector<T> toCollector(ExecutionTimeValue<? extends Iterable<? extends T>> value) {
+            Preconditions.checkArgument(!value.isMissing(), "Cannot get a collector for the missing value");
+            if (value.isChangingValue() || value.hasChangingContent() || value.getSideEffect() != null) {
+                return new ElementsFromCollectionProvider<>(value.toProvider());
+            }
+            return new ElementsFromCollection<>(value.getFixedValue());
+        }
+
+        private List<Collector<T>> getCollectors() {
+            return collectors.subList(0, size);
+        }
+
+        // Returns an empty list when the overall value is missing.
+        private List<Pair<Collector<T>, ExecutionTimeValue<? extends Iterable<? extends T>>>> collectExecutionTimeValues() {
+            // These are the values that are certainly part of the result, e.g. because of absent-ignoring append/appendAll argument.
+            List<Pair<Collector<T>, ExecutionTimeValue<? extends Iterable<? extends T>>>> executionTimeValues = new ArrayList<>();
+            // These are the values that may become part of the result if there is no missing value somewhere.
+            List<Pair<Collector<T>, ExecutionTimeValue<? extends Iterable<? extends T>>>> candidates = new ArrayList<>();
+
+            // We traverse the collectors backwards (in reverse addition order) to simplify the logic and avoid processing things that are going to be discarded.
+            // Because of that, values are collected in reverse order too.
+            // Se also #calculatePresence.
+            for (Collector<T> collector : Lists.reverse(getCollectors())) {
+                ExecutionTimeValue<? extends Iterable<? extends T>> result = collector.calculateExecutionTimeValue();
+                if (result.isMissing()) {
+                    // This is an add/addAll argument, but it is a missing provider.
+                    // Everything that was added before it isn't going to affect the result, so we stop the iteration.
+                    // All add/addAll that happened after it (thus already processed) but before any append/appendAll - the contents of candidates - are also discarded.
+                    return Lists.reverse(executionTimeValues);
+                }
+                if (isAbsentIgnoring(collector)) {
+                    // This is an argument of append/appendAll. With it the property is going to be present (though maybe empty).
+                    // As all add/addAll arguments we've processed (thus added after this one) so far weren't missing, we're sure they'll be part of the final property's value.
+                    // Move them to the executionTimeValues.
+                    executionTimeValues.addAll(candidates);
+                    executionTimeValues.add(Pair.of(collector, result));
+                    candidates.clear();
+                } else {
+                    // This is an argument of add/addAll that isn't definitely missing. It might be part of the final value.
+                    candidates.add(Pair.of(collector, result));
+                }
+            }
+            // No missing values found, so all the candidates are part of the final value.
+            executionTimeValues.addAll(candidates);
+            return Lists.reverse(executionTimeValues);
         }
 
         private ExecutionTimeValue<C> getFixedExecutionTimeValue(List<ExecutionTimeValue<? extends Iterable<? extends T>>> values, boolean changingContent) {
@@ -572,57 +679,45 @@ private ExecutionTimeValue<C> getFixedExecutionTimeValue(List<ExecutionTimeValue
             return mergedValue.withSideEffect(sideEffectBuilder.build());
         }
 
+        private Stream<ValueProducer> getProducers() {
+            return collectors.stream().map(ValueSupplier::getProducer);
+        }
+
         @Override
         public ValueProducer getProducer() {
-            return value.getProducer();
-        }
-
-        @Override
-        public String toString() {
-            return value.toString();
-        }
-    }
-
-    /**
-     * A provider for a collection type whose elements are themselves providers.
-     */
-    private static class CollectingProvider<T, C extends Collection<? extends T>> extends AbstractMinimalProvider<C> {
-        private final Class<C> type;
-        private final List<ProviderInternal<? extends Iterable<? extends T>>> providers;
-        private final Supplier<ImmutableCollection.Builder<T>> collectionFactory;
-
-        public CollectingProvider(Class<C> type, List<ProviderInternal<? extends Iterable<? extends T>>> providers, Supplier<ImmutableCollection.Builder<T>> collectionFactory) {
-            this.type = type;
-            this.providers = providers;
-            this.collectionFactory = collectionFactory;
-        }
-
-        @Nullable
-        @Override
-        public Class<C> getType() {
-            return type;
-        }
-
-        @Override
-        public ExecutionTimeValue<? extends C> calculateExecutionTimeValue() {
-            return ExecutionTimeValue.changingValue(this);
-        }
-
-        @Override
-        protected Value<? extends C> calculateOwnValue(ValueConsumer consumer) {
-            ImmutableCollection.Builder<T> builder = collectionFactory.get();
-            SideEffectBuilder<? super C> sideEffectBuilder = SideEffect.builder();
-            for (ProviderInternal<? extends Iterable<? extends T>> provider : providers) {
-                Value<? extends Iterable<? extends T>> value = provider.calculateValue(consumer);
-                if (value.isMissing()) {
-                    return Value.missing();
+            return new ValueProducer() {
+                @Override
+                public void visitProducerTasks(Action<? super Task> visitor) {
+                    getProducers().forEach(c -> c.visitProducerTasks(visitor));
                 }
-                builder.addAll(value.getWithoutSideEffect());
-                sideEffectBuilder.add(SideEffect.fixedFrom(value));
-            }
 
-            Value<? extends C> resultValue = Value.of(Cast.uncheckedNonnullCast(builder.build()));
-            return resultValue.withSideEffect(sideEffectBuilder.build());
+                @Override
+                public boolean isKnown() {
+                    return getProducers().anyMatch(ValueProducer::isKnown);
+                }
+
+                @Override
+                public void visitDependencies(TaskDependencyResolveContext context) {
+                    getProducers().forEach(c -> c.visitDependencies(context));
+                }
+
+                @Override
+                public void visitContentProducerTasks(Action<? super Task> visitor) {
+                    getProducers().forEach(c -> c.visitContentProducerTasks(visitor));
+                }
+            };
+        }
+
+        @Override
+        protected String toStringNoReentrance() {
+            StringBuilder sb = new StringBuilder();
+            getCollectors().forEach(collector -> {
+                if (sb.length() > 0) {
+                    sb.append(" + ");
+                }
+                sb.append(collector.toString());
+            });
+            return sb.toString();
         }
     }
 
@@ -653,14 +748,8 @@ public int size() {
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(collection).withSideEffect(sideEffect));
-        }
-
-        @Override
-        public Collector<T> absentIgnoring() {
-            // always present
-            return this;
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(collection).withSideEffect(sideEffect);
         }
 
         @Override
@@ -679,79 +768,44 @@ public String toString() {
         }
     }
 
-    private static abstract class AbstractPlusCollector<T> implements Collector<T> {
-        protected final Collector<T> left;
-        protected final Collector<T> right;
-
-        private AbstractPlusCollector(Collector<T> left, Collector<T> right) {
-            this.left = left;
-            this.right = right;
-        }
-
-        @Override
-        public int size() {
-            return left.size() + right.size();
-        }
-
-        @Override
-        public ValueProducer getProducer() {
-            return left.getProducer().plus(right.getProducer());
-        }
-
-        @Override
-        public String toString() {
-            return left + " + " + right;
-        }
-
+    private static boolean isAbsentIgnoring(Collector<?> collector) {
+        return collector instanceof AbsentIgnoringCollector<?>;
     }
 
-    private static class PlusCollector<T> extends AbstractPlusCollector<T> {
-
-        public PlusCollector(Collector<T> left, Collector<T> right) {
-            super(left, right);
+    private static <T> Collector<T> ignoreAbsentIfNeeded(Collector<T> collector, boolean ignoreAbsent) {
+        if (ignoreAbsent && !isAbsentIgnoring(collector)) {
+            return new AbsentIgnoringCollector<>(collector);
         }
+        return collector;
+    }
 
-        @Override
-        public Collector<T> absentIgnoring() {
-            return new AbsentIgnoringPlusCollector<>(left, right);
-        }
+    private static class AbsentIgnoringCollector<T> implements Collector<T> {
+        private final Collector<T> delegate;
 
-        @Override
-        public boolean calculatePresence(ValueConsumer consumer) {
-            return left.calculatePresence(consumer) && right.calculatePresence(consumer);
+        private AbsentIgnoringCollector(Collector<T> delegate) {
+            this.delegate = delegate;
         }
 
         @Override
         public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> collector, ImmutableCollection.Builder<T> dest) {
-            Value<Void> leftValue = left.collectEntries(consumer, collector, dest);
-            if (leftValue.isMissing()) {
-                return leftValue;
+            ImmutableList.Builder<T> candidateEntries = ImmutableList.builder();
+            Value<Void> value = delegate.collectEntries(consumer, collector, candidateEntries);
+            if (value.isMissing()) {
+                return Value.present();
             }
-            Value<Void> rightValue = right.collectEntries(consumer, collector, dest);
-            if (rightValue.isMissing()) {
-                return rightValue;
-            }
-
-            return Value.present()
-                .withSideEffect(SideEffect.fixedFrom(leftValue))
-                .withSideEffect(SideEffect.fixedFrom(rightValue));
+            dest.addAll(candidateEntries.build());
+            return Value.present().withSideEffect(SideEffect.fixedFrom(value));
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            left.calculateExecutionTimeValue(visitor);
-            right.calculateExecutionTimeValue(visitor);
+        public int size() {
+            return delegate.size();
         }
-    }
 
-    /**
-     * A plus collector that either produces a composition of both of its left and right sides,
-     * or Value.present() with empty content (if left or right side are missing).
-     */
-    private static class AbsentIgnoringPlusCollector<T> extends AbstractPlusCollector<T> {
-
-        public AbsentIgnoringPlusCollector(Collector<T> left, Collector<T> right) {
-            super(left, right);
+        @Override
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            ExecutionTimeValue<? extends Iterable<? extends T>> executionTimeValue = delegate.calculateExecutionTimeValue();
+            return executionTimeValue.isMissing() ? ExecutionTimeValue.fixedValue(ImmutableList.of()) : executionTimeValue;
         }
 
         @Override
@@ -760,44 +814,8 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            return this;
-        }
-
-        @Override
-        public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> collector, ImmutableCollection.Builder<T> dest) {
-            ImmutableList.Builder<T> candidateEntries = ImmutableList.builder();
-            // we cannot use dest directly because we don't want to emit any entries if either left or right are missing
-            Value<Void> leftValue = left.collectEntries(consumer, collector, candidateEntries);
-            if (leftValue.isMissing()) {
-                return Value.present();
-            }
-            Value<Void> rightValue = right.collectEntries(consumer, collector, candidateEntries);
-            if (rightValue.isMissing()) {
-                return Value.present();
-            }
-            dest.addAll(candidateEntries.build());
-            return Value.present()
-                .withSideEffect(SideEffect.fixedFrom(leftValue))
-                .withSideEffect(SideEffect.fixedFrom(rightValue));
-        }
-
-        @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            boolean[] anyMissing = {false};
-            ImmutableList.Builder<ExecutionTimeValue<? extends Iterable<? extends T>>> toVisit = ImmutableList.builder();
-            Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> safeVisitor = value -> {
-                if (value.isMissing()) {
-                    anyMissing[0] = true;
-                } else {
-                    toVisit.add(value);
-                }
-            };
-            left.calculateExecutionTimeValue(safeVisitor);
-            right.calculateExecutionTimeValue(safeVisitor);
-            if (!anyMissing[0]) {
-                toVisit.build().forEach(it -> visitor.execute(it));
-            }
+        public ValueProducer getProducer() {
+            return delegate.getProducer();
         }
     }
 
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/CollectionSupplier.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/CollectionSupplier.java
index 2d93b16..2e5f95a1 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/CollectionSupplier.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/CollectionSupplier.java
@@ -37,24 +37,12 @@ interface CollectionSupplier<T, C extends Collection<? extends T>> extends Value
      * elements supplied by the given collector.
      *
      * @param added a collector that represents an addition to the collection to be returned by this supplier
+     * @param ignoreAbsent if the resulting supplier should ignore absent value of this supplier or added one and fallback to empty collection
+     *
      * @return a new supplier that produces a collection that contains the
      * same elements as this supplier, plus the elements obtained via the given <code>added</code> collector
      */
-    CollectionSupplier<T, C> plus(Collector<T> added);
+    CollectionSupplier<T, C> plus(Collector<T> added, boolean ignoreAbsent);
 
     ExecutionTimeValue<? extends C> calculateExecutionTimeValue();
-
-    /**
-     * Returns a view of this supplier that will calculate its value as empty if it would be missing.
-     * If this supplier already ignores absent results, returns this supplier.
-     */
-    CollectionSupplier<T, C> absentIgnoring();
-
-    /**
-     * Returns a view of this supplier that will calculate its value as empty if it would be missing,
-     * if required. If not required, or this supplier already ignores absent results, returns this supplier.
-     */
-    default CollectionSupplier<T, C> absentIgnoringIfNeeded(boolean required) {
-        return required ? absentIgnoring() : this;
-    }
 }
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collector.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collector.java
index a3a3ca2..2be77f9 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collector.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collector.java
@@ -17,7 +17,6 @@
 package org.gradle.api.internal.provider;
 
 import com.google.common.collect.ImmutableCollection;
-import org.gradle.api.Action;
 
 
 /**
@@ -31,18 +30,5 @@ public interface Collector<T> extends ValueSupplier {
 
     int size();
 
-    void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor);
-
-    /**
-     * Returns a view of this collector that never returns a missing value.
-     */
-    Collector<T> absentIgnoring();
-
-    /**
-     * Convenience method that returns a view of this collector that never returns a missing value,
-     * if that capability is required, or this very collector.
-     */
-    default Collector<T> absentIgnoringIfNeeded(boolean required) {
-        return required ? absentIgnoring() : this;
-    }
+    ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue();
 }
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collectors.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collectors.java
index 1539d83..8a58ad3 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collectors.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/Collectors.java
@@ -20,13 +20,9 @@
 import com.google.common.collect.ImmutableCollection;
 import com.google.common.collect.ImmutableList;
 import com.google.common.collect.Iterables;
-import org.gradle.api.Action;
 import org.gradle.api.provider.Provider;
-import org.gradle.internal.Cast;
 
-import javax.annotation.Nonnull;
 import javax.annotation.Nullable;
-
 import java.util.Arrays;
 
 import static org.gradle.api.internal.lambdas.SerializableLambdas.transformer;
@@ -49,20 +45,14 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            // always present
-            return this;
-        }
-
-        @Override
         public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> collector, ImmutableCollection.Builder<T> collection) {
             collector.add(element, collection);
             return Value.present();
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(ImmutableList.of(element)));
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(ImmutableList.of(element));
         }
 
         @Override
@@ -106,11 +96,6 @@ public ElementFromProvider(ProviderInternal<? extends T> provider) {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            return new ElementsFromCollectionProvider<>(provider.map(ImmutableList::of), true);
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return provider.calculatePresence(consumer);
         }
@@ -132,9 +117,9 @@ public boolean isProvidedBy(Provider<?> provider) {
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
             ExecutionTimeValue<? extends T> value = provider.calculateExecutionTimeValue();
-            visitValue(visitor, value);
+            return visitValue(value);
         }
 
         @Override
@@ -170,14 +155,14 @@ public String toString() {
         }
     }
 
-    private static <T> void visitValue(Action<? super ValueSupplier.ExecutionTimeValue<? extends Iterable<? extends T>>> visitor, ValueSupplier.ExecutionTimeValue<? extends T> value) {
+    private static <T> ValueSupplier.ExecutionTimeValue<? extends Iterable<? extends T>> visitValue(ValueSupplier.ExecutionTimeValue<? extends T> value) {
         if (value.isMissing()) {
-            visitor.execute(ValueSupplier.ExecutionTimeValue.missing());
+            return ValueSupplier.ExecutionTimeValue.missing();
         } else if (value.hasFixedValue()) {
             // transform preserving side effects
-            visitor.execute(ValueSupplier.ExecutionTimeValue.value(value.toValue().transform(ImmutableList::of)));
+            return ValueSupplier.ExecutionTimeValue.value(value.toValue().transform(ImmutableList::of));
         } else {
-            visitor.execute(ValueSupplier.ExecutionTimeValue.changingValue(value.getChangingValue().map(transformer(ImmutableList::of))));
+            return ValueSupplier.ExecutionTimeValue.changingValue(value.getChangingValue().map(transformer(ImmutableList::of)));
         }
     }
 
@@ -194,20 +179,14 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            // always present
-            return this;
-        }
-
-        @Override
         public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> collector, ImmutableCollection.Builder<T> collection) {
             collector.addAll(value, collection);
             return Value.present();
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(value));
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(value);
         }
 
         @Override
@@ -249,30 +228,14 @@ public String toString() {
 
     public static class ElementsFromCollectionProvider<T> implements ProvidedCollector<T> {
         private final ProviderInternal<? extends Iterable<? extends T>> provider;
-        private final boolean ignoreAbsent;
 
         public ElementsFromCollectionProvider(ProviderInternal<? extends Iterable<? extends T>> provider) {
-            this(provider, false);
-        }
-
-        private ElementsFromCollectionProvider(ProviderInternal<? extends Iterable<? extends T>> provider, boolean ignoreAbsent) {
-            this.provider = ignoreAbsent ? neverMissing(Cast.uncheckedNonnullCast(provider)) : provider;
-            this.ignoreAbsent = ignoreAbsent;
-        }
-
-        @Nonnull
-        private static <T> ProviderInternal<? extends Iterable<? extends T>> neverMissing(ProviderInternal<Iterable<? extends T>> provider) {
-            return Cast.uncheckedNonnullCast(provider.orElse(ImmutableList.of()));
-        }
-
-        @Override
-        public Collector<T> absentIgnoring() {
-            return ignoreAbsent ? this : new ElementsFromCollectionProvider<>(provider, true);
+            this.provider = provider;
         }
 
         @Override
         public boolean calculatePresence(ValueConsumer consumer) {
-            return ignoreAbsent || provider.calculatePresence(consumer);
+            return provider.calculatePresence(consumer);
         }
 
         @Override
@@ -283,7 +246,7 @@ public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> coll
 
         private ValueSupplier.Value<Void> collectEntriesFromValue(ValueCollector<T> collector, ImmutableCollection.Builder<T> collection, ValueSupplier.Value<? extends Iterable<? extends T>> value) {
             if (value.isMissing()) {
-                return ignoreAbsent ? Value.present() : value.asType();
+                return value.asType();
             }
 
             collector.addAll(value.getWithoutSideEffect(), collection);
@@ -291,8 +254,8 @@ private ValueSupplier.Value<Void> collectEntriesFromValue(ValueCollector<T> coll
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            visitor.execute(provider.calculateExecutionTimeValue());
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return provider.calculateExecutionTimeValue();
         }
 
         @Override
@@ -350,12 +313,6 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            // always present
-            return this;
-        }
-
-        @Override
         public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> collector, ImmutableCollection.Builder<T> dest) {
             for (T t : value) {
                 collector.add(t, dest);
@@ -364,8 +321,8 @@ public Value<Void> collectEntries(ValueConsumer consumer, ValueCollector<T> coll
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(ImmutableList.copyOf(value)));
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(ImmutableList.copyOf(value));
         }
 
         @Override
@@ -401,11 +358,6 @@ public Class<? extends T> getType() {
         }
 
         @Override
-        public Collector<T> absentIgnoring() {
-            throw new UnsupportedOperationException();
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return delegate.calculatePresence(consumer);
         }
@@ -425,8 +377,8 @@ public boolean isProvidedBy(Provider<?> provider) {
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<? super ExecutionTimeValue<? extends Iterable<? extends T>>> visitor) {
-            delegate.calculateExecutionTimeValue(visitor);
+        public ExecutionTimeValue<? extends Iterable<? extends T>> calculateExecutionTimeValue() {
+            return delegate.calculateExecutionTimeValue();
         }
 
         @Override
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/DefaultMapProperty.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/DefaultMapProperty.java
index 3e4a254..3bea175 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/DefaultMapProperty.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/DefaultMapProperty.java
@@ -18,37 +18,43 @@
 
 import com.google.common.base.Preconditions;
 import com.google.common.collect.ImmutableCollection;
-import com.google.common.collect.ImmutableList;
 import com.google.common.collect.ImmutableMap;
 import com.google.common.collect.ImmutableSet;
+import com.google.common.collect.Lists;
 import org.gradle.api.Action;
+import org.gradle.api.Task;
 import org.gradle.api.Transformer;
 import org.gradle.api.internal.provider.MapCollectors.EntriesFromMap;
 import org.gradle.api.internal.provider.MapCollectors.EntriesFromMapProvider;
 import org.gradle.api.internal.provider.MapCollectors.EntryWithValueFromProvider;
 import org.gradle.api.internal.provider.MapCollectors.SingleEntry;
+import org.gradle.api.internal.tasks.TaskDependencyResolveContext;
 import org.gradle.api.provider.MapProperty;
 import org.gradle.api.provider.Provider;
 import org.gradle.internal.Cast;
+import org.gradle.internal.Pair;
 import org.gradle.internal.evaluation.EvaluationScopeContext;
 
+import javax.annotation.Nonnull;
 import javax.annotation.Nullable;
 import java.util.ArrayList;
 import java.util.LinkedHashMap;
 import java.util.List;
 import java.util.Map;
 import java.util.Set;
+import java.util.stream.Collectors;
 
+import static java.util.stream.Collectors.toList;
 import static org.gradle.internal.Cast.uncheckedCast;
 import static org.gradle.internal.Cast.uncheckedNonnullCast;
 
 /**
  * The implementation for {@link MapProperty}.
  * <p>
- *     Value suppliers for map properties are implementations of {@link MapSupplier}.
+ * Value suppliers for map properties are implementations of {@link MapSupplier}.
  * </p>
  * <p>
- *     Increments to map property values are implementations of {@link MapCollector}.
+ * Increments to map property values are implementations of {@link MapCollector}.
  * </p>
  *
  * This class mimics much of the behavior {@link AbstractCollectionProperty} provides for regular collections
@@ -162,13 +168,13 @@ public void set(@Nullable Map<? extends K, ? extends V> entries) {
         if (entries == null) {
             unsetValueAndDefault();
         } else {
-            setSupplier(new CollectingSupplier(new EntriesFromMap<>(entries), false));
+            setSupplier(newCollectingSupplierOf(new EntriesFromMap<>(entries)));
         }
     }
 
     @Override
     public void set(Provider<? extends Map<? extends K, ? extends V>> provider) {
-        setSupplier(new CollectingSupplier(new MapCollectors.EntriesFromMapProvider<>(checkMapProvider(provider)), false));
+        setSupplier(newCollectingSupplierOf(new MapCollectors.EntriesFromMapProvider<>(checkMapProvider(provider))));
     }
 
     @Override
@@ -225,8 +231,8 @@ public void insertAll(Map<? extends K, ? extends V> entries) {
 
     private void addExplicitCollector(MapCollector<K, V> collector, boolean ignoreAbsent) {
         assertCanMutate();
-        MapSupplier<K, V> explicitValue = getExplicitValue(defaultValue).absentIgnoringIfNeeded(ignoreAbsent);
-        setSupplier(explicitValue.plus(collector.absentIgnoringIfNeeded(ignoreAbsent)));
+        MapSupplier<K, V> explicitValue = getExplicitValue(defaultValue);
+        setSupplier(explicitValue.plus(collector, ignoreAbsent));
     }
 
     private Configurer getConfigurer() {
@@ -278,14 +284,14 @@ public MapProperty<K, V> convention(@Nullable Map<? extends K, ? extends V> valu
         if (value == null) {
             setConvention(noValueSupplier());
         } else {
-            setConvention(new CollectingSupplier(new EntriesFromMap<>(value), false));
+            setConvention(newCollectingSupplierOf(new EntriesFromMap<>(value)));
         }
         return this;
     }
 
     @Override
     public MapProperty<K, V> convention(Provider<? extends Map<? extends K, ? extends V>> valueProvider) {
-        setConvention(new CollectingSupplier(new EntriesFromMapProvider<>(Providers.internal(valueProvider)), false));
+        setConvention(newCollectingSupplierOf(new EntriesFromMapProvider<>(Providers.internal(valueProvider))));
         return this;
     }
 
@@ -312,8 +318,8 @@ public void fromState(ExecutionTimeValue<? extends Map<? extends K, ? extends V>
         } else if (value.hasFixedValue()) {
             setSupplier(new FixedSupplier(uncheckedNonnullCast(value.getFixedValue()), uncheckedCast(value.getSideEffect())));
         } else {
-            CollectingProvider<K, V> asCollectingProvider = uncheckedNonnullCast(value.getChangingValue());
-            setSupplier(new CollectingSupplier(new EntriesFromMapProvider<>(asCollectingProvider)));
+            CollectingSupplier<K, V> asCollectingProvider = uncheckedNonnullCast(value.getChangingValue());
+            setSupplier(asCollectingProvider);
         }
     }
 
@@ -408,11 +414,6 @@ public NoValueSupplier(Value<? extends Map<K, V>> value) {
         }
 
         @Override
-        public MapSupplier<K, V> absentIgnoring() {
-            return emptySupplier();
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return false;
         }
@@ -428,9 +429,9 @@ public Value<? extends Set<K>> calculateKeys(ValueConsumer consumer) {
         }
 
         @Override
-        public MapSupplier<K, V> plus(MapCollector<K, V> collector) {
-            // nothing + something = nothing
-            return this;
+        public MapSupplier<K, V> plus(MapCollector<K, V> collector, boolean ignoreAbsent) {
+            // nothing + something = nothing, unless we ignoreAbsent.
+            return ignoreAbsent ? newCollectingSupplierOf(ignoreAbsentIfNeeded(collector, ignoreAbsent)) : this;
         }
 
         @Override
@@ -451,11 +452,6 @@ public String toString() {
 
     private class EmptySupplier implements MapSupplier<K, V> {
         @Override
-        public MapSupplier<K, V> absentIgnoring() {
-            return this;
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return true;
         }
@@ -471,9 +467,9 @@ public Value<? extends Set<K>> calculateKeys(ValueConsumer consumer) {
         }
 
         @Override
-        public MapSupplier<K, V> plus(MapCollector<K, V> collector) {
+        public MapSupplier<K, V> plus(MapCollector<K, V> collector, boolean ignoreAbsent) {
             // empty + something = something
-            return new CollectingSupplier(collector);
+            return newCollectingSupplierOf(ignoreAbsentIfNeeded(collector, ignoreAbsent));
         }
 
         @Override
@@ -517,15 +513,8 @@ public Value<? extends Set<K>> calculateKeys(ValueConsumer consumer) {
         }
 
         @Override
-        public MapSupplier<K, V> plus(MapCollector<K, V> collector) {
-            MapCollector<K, V> left = new FixedValueCollector<>(entries, sideEffect);
-            PlusCollector<K, V> newCollector = new PlusCollector<>(left, collector);
-            return new CollectingSupplier(newCollector);
-        }
-
-        @Override
-        public MapSupplier<K, V> absentIgnoring() {
-            return this;
+        public MapSupplier<K, V> plus(MapCollector<K, V> collector, boolean ignoreAbsent) {
+            return newCollectingSupplierOf(new FixedValueCollector<>(entries, sideEffect)).plus(collector, ignoreAbsent);
         }
 
         @Override
@@ -544,39 +533,76 @@ public String toString() {
         }
     }
 
-    private class CollectingSupplier implements MapSupplier<K, V> {
-        private final MapCollector<K, V> collector;
-        // TODO-RC: can we get rid of this? Can we only keep this in Collectors? Changing execution time value is the only case that needs this.
-        private final boolean ignoreAbsent;
+    private CollectingSupplier<K, V> newCollectingSupplierOf(MapCollector<K, V> collector) {
+        return new CollectingSupplier<>(keyCollector, entryCollector, collector);
+    }
 
-        public CollectingSupplier(MapCollector<K, V> collector, boolean ignoreAbsent) {
-            this.collector = collector;
-            this.ignoreAbsent = ignoreAbsent;
+    private static class CollectingSupplier<K, V> extends AbstractMinimalProvider<Map<K, V>> implements MapSupplier<K, V> {
+        private final ValueCollector<K> keyCollector;
+        private final MapEntryCollector<K, V> entryCollector;
+        private final List<MapCollector<K, V>> collectors;
+        private final int size;
+
+        public CollectingSupplier(ValueCollector<K> keyCollector, MapEntryCollector<K, V> entryCollector, MapCollector<K, V> collector) {
+            this(keyCollector, entryCollector, Lists.newArrayList(collector), 1);
         }
 
-        public CollectingSupplier(MapCollector<K, V> collector) {
-            this(collector, false);
+        public CollectingSupplier(ValueCollector<K> keyCollector, MapEntryCollector<K, V> entryCollector, List<MapCollector<K, V>> collectors, int size) {
+            this.keyCollector = keyCollector;
+            this.entryCollector = entryCollector;
+            this.size = size;
+            this.collectors = collectors;
         }
 
         @Override
-        public MapSupplier<K, V> absentIgnoring() {
-            return ignoreAbsent ? this : new CollectingSupplier(collector, true);
+        protected Value<? extends Map<K, V>> calculateOwnValue(ValueConsumer consumer) {
+            return calculateValue(consumer);
         }
 
         @Override
         public boolean calculatePresence(ValueConsumer consumer) {
-            return collector.calculatePresence(consumer);
+            for (MapCollector<K, V> collector : Lists.reverse(getCollectors())) {
+                if (!collector.calculatePresence(consumer)) {
+                    return false;
+                }
+                if (collector instanceof AbsentIgnoringCollector<?, ?>) {
+                    return true;
+                }
+            }
+            return true;
+        }
+
+        @Nullable
+        @Override
+        @SuppressWarnings("unchecked")
+        public Class<Map<K, V>> getType() {
+            return (Class) Map.class;
         }
 
         @Override
         public Value<? extends Set<K>> calculateKeys(ValueConsumer consumer) {
             // TODO - don't make a copy when the collector already produces an immutable collection
             ImmutableSet.Builder<K> builder = ImmutableSet.builder();
-            Value<Void> result = collector.collectKeys(consumer, keyCollector, builder);
-            if (result.isMissing()) {
-                return result.asType();
+            Value<Void> compositeResult = Value.present();
+            for (MapCollector<K, V> collector : getCollectors()) {
+                Value<Void> result = collector.collectKeys(consumer, keyCollector, builder);
+                if (result.isMissing()) {
+                    builder = ImmutableSet.builder();
+                    compositeResult = result;
+                } else if (compositeResult.isMissing()) {
+                    if (isAbsentIgnoring(collector)) {
+                        compositeResult = result;
+                    } else {
+                        builder = ImmutableSet.builder();
+                    }
+                } else {
+                    compositeResult = compositeResult.withSideEffect(SideEffect.fixedFrom(result));
+                }
             }
-            return Value.of(ImmutableSet.copyOf(builder.build())).withSideEffect(SideEffect.fixedFrom(result));
+            if (compositeResult.isMissing()) {
+                return compositeResult.asType();
+            }
+            return Value.of(ImmutableSet.copyOf(builder.build())).withSideEffect(SideEffect.fixedFrom(compositeResult));
         }
 
         @Override
@@ -586,28 +612,66 @@ public Value<? extends Map<K, V>> calculateValue(ValueConsumer consumer) {
             // for MapProperty allows a provider to override the entries of earlier providers and so there can be multiple entries
             // with the same key
             Map<K, V> entries = new LinkedHashMap<>();
-            Value<Void> result = collector.collectEntries(consumer, entryCollector, entries);
-            if (result.isMissing()) {
-                return result.asType();
+            Value<Void> compositeResult = Value.present();
+            for (MapCollector<K, V> collector : getCollectors()) {
+                Value<Void> result = collector.collectEntries(consumer, entryCollector, entries);
+                if (result.isMissing()) {
+                    entries.clear();
+                    compositeResult = result;
+                } else if (compositeResult.isMissing()) {
+                    if (isAbsentIgnoring(collector)) {
+                        compositeResult = result;
+                    } else {
+                        entries.clear();
+                    }
+                } else {
+                    compositeResult = compositeResult.withSideEffect(SideEffect.fixedFrom(result));
+                }
             }
-            return Value.of(ImmutableMap.copyOf(entries)).withSideEffect(SideEffect.fixedFrom(result));
+            if (compositeResult.isMissing()) {
+                return compositeResult.asType();
+            }
+            return Value.of(ImmutableMap.copyOf(entries)).withSideEffect(SideEffect.fixedFrom(compositeResult));
         }
 
         @Override
-        public MapSupplier<K, V> plus(MapCollector<K, V> addedCollector) {
-            MapCollector<K, V> left = this.collector.absentIgnoringIfNeeded(ignoreAbsent);
-            MapCollector<K, V> right = addedCollector;
-            PlusCollector<K, V> newCollector = new PlusCollector<>(left, right);
-            return new CollectingSupplier(newCollector);
+        public MapSupplier<K, V> plus(MapCollector<K, V> addedCollector, boolean ignoreAbsent) {
+            Preconditions.checkState(collectors.size() == size);
+            collectors.add(ignoreAbsentIfNeeded(addedCollector, ignoreAbsent));
+            return new CollectingSupplier<>(keyCollector, entryCollector, collectors, size + 1);
         }
 
         @Override
         public ExecutionTimeValue<? extends Map<K, V>> calculateExecutionTimeValue() {
-            List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> execTimeValues = collectExecutionTimeValues();
-            ExecutionTimeValue<Map<K, V>> fixedOrMissing = fixedOrMissingValueOf(execTimeValues);
-            return fixedOrMissing != null
-                ? fixedOrMissing
-                : ExecutionTimeValue.changingValue(new CollectingProvider<>(execTimeValues));
+            List<Pair<MapCollector<K, V>, ExecutionTimeValue<? extends Map<? extends K, ? extends V>>>> collectorsWithValues = collectExecutionTimeValues();
+            if (collectorsWithValues.isEmpty()) {
+                return ExecutionTimeValue.missing();
+            }
+            List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> executionTimeValues = collectorsWithValues.stream().map(Pair::getRight).collect(Collectors.toList());
+            ExecutionTimeValue<Map<K, V>> fixedOrMissing = fixedOrMissingValueOf(executionTimeValues);
+            if (fixedOrMissing != null) {
+                return fixedOrMissing;
+            }
+
+            return ExecutionTimeValue.changingValue(new CollectingSupplier<>(
+                keyCollector,
+                entryCollector,
+                collectorsWithValues.stream().map(pair -> {
+                    MapCollector<K, V> elements = toCollector(pair.getRight());
+                    return isAbsentIgnoring(pair.getLeft())
+                        ? new AbsentIgnoringCollector<>(elements)
+                        : elements;
+                }).collect(toList()),
+                collectorsWithValues.size())
+            );
+        }
+
+        private MapCollector<K, V> toCollector(ExecutionTimeValue<? extends Map<? extends K, ? extends V>> value) {
+            Preconditions.checkArgument(!value.isMissing(), "Cannot get a collector for the missing value");
+            if (value.isChangingValue() || value.hasChangingContent() || value.getSideEffect() != null) {
+                return new EntriesFromMapProvider<>(value.toProvider());
+            }
+            return new EntriesFromMap<>(value.getFixedValue());
         }
 
         /**
@@ -636,10 +700,26 @@ private ExecutionTimeValue<Map<K, V>> fixedOrMissingValueOf(List<ExecutionTimeVa
             return null;
         }
 
-        private List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> collectExecutionTimeValues() {
-            List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> values = new ArrayList<>();
-            collector.calculateExecutionTimeValue(values::add);
-            return values;
+        @Nonnull
+        private List<Pair<MapCollector<K, V>, ExecutionTimeValue<? extends Map<? extends K, ? extends V>>>> collectExecutionTimeValues() {
+            List<Pair<MapCollector<K, V>, ExecutionTimeValue<? extends Map<? extends K, ? extends V>>>> executionTimeValues = new ArrayList<>();
+            List<Pair<MapCollector<K, V>, ExecutionTimeValue<? extends Map<? extends K, ? extends V>>>> candidates = new ArrayList<>();
+
+            for (MapCollector<K, V> collector : Lists.reverse(getCollectors())) {
+                ExecutionTimeValue<? extends Map<? extends K, ? extends V>> result = collector.calculateExecutionTimeValue();
+                if (result.isMissing()) {
+                    return Lists.reverse(executionTimeValues);
+                }
+                if (isAbsentIgnoring(collector)) {
+                    executionTimeValues.addAll(candidates);
+                    executionTimeValues.add(Pair.of(collector, result));
+                    candidates.clear();
+                } else {
+                    candidates.add(Pair.of(collector, result));
+                }
+            }
+            executionTimeValues.addAll(candidates);
+            return Lists.reverse(executionTimeValues);
         }
 
         private ImmutableMap<K, V> collectEntries(List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> values, SideEffectBuilder<? super Map<K, V>> sideEffectBuilder) {
@@ -651,54 +731,108 @@ private ImmutableMap<K, V> collectEntries(List<ExecutionTimeValue<? extends Map<
             return ImmutableMap.copyOf(entries);
         }
 
+        private List<MapCollector<K, V>> getCollectors() {
+            return collectors.subList(0, size);
+        }
+
         private ExecutionTimeValue<Map<K, V>> maybeChangingContent(ExecutionTimeValue<Map<K, V>> value, boolean changingContent) {
             return changingContent ? value.withChangingContent() : value;
         }
 
         @Override
         public ValueProducer getProducer() {
-            return collector.getProducer();
+            List<ValueProducer> producers = getCollectors().stream().map(ValueSupplier::getProducer).collect(toList());
+            return new ValueProducer() {
+                @Override
+                public void visitProducerTasks(Action<? super Task> visitor) {
+                    producers.forEach(c -> c.visitProducerTasks(visitor));
+                }
+
+                @Override
+                public boolean isKnown() {
+                    return producers.stream().anyMatch(ValueProducer::isKnown);
+                }
+
+                @Override
+                public void visitDependencies(TaskDependencyResolveContext context) {
+                    producers.forEach(c -> c.visitDependencies(context));
+                }
+
+                @Override
+                public void visitContentProducerTasks(Action<? super Task> visitor) {
+                    producers.forEach(c -> c.visitContentProducerTasks(visitor));
+                }
+            };
         }
 
         @Override
-        public String toString() {
-            return collector.toString();
+        protected String toStringNoReentrance() {
+            StringBuilder sb = new StringBuilder();
+            getCollectors().forEach(collector -> {
+                if (sb.length() > 0) {
+                    sb.append(" + ");
+                }
+                sb.append(collector.toString());
+            });
+            return sb.toString();
         }
     }
 
-    private static class CollectingProvider<K, V> extends AbstractMinimalProvider<Map<K, V>> {
-        private final List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> values;
+    private static boolean isAbsentIgnoring(MapCollector<?, ?> collector) {
+        return collector instanceof AbsentIgnoringCollector<?, ?>;
+    }
 
-        public CollectingProvider(List<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> values) {
-            this.values = values;
+    private static <K, V> MapCollector<K, V> ignoreAbsentIfNeeded(MapCollector<K, V> collector, boolean ignoreAbsent) {
+        if (ignoreAbsent && !isAbsentIgnoring(collector)) {
+            return new AbsentIgnoringCollector<>(collector);
         }
+        return collector;
+    }
 
-        @Nullable
-        @Override
-        public Class<Map<K, V>> getType() {
-            return uncheckedCast(Map.class);
+    private static class AbsentIgnoringCollector<K, V> implements MapCollector<K, V> {
+        private final MapCollector<K, V> delegate;
+
+        private AbsentIgnoringCollector(MapCollector<K, V> delegate) {
+            this.delegate = delegate;
         }
 
         @Override
-        public ExecutionTimeValue<? extends Map<K, V>> calculateExecutionTimeValue() {
-            return ExecutionTimeValue.changingValue(this);
-        }
-
-        @Override
-        protected Value<? extends Map<K, V>> calculateOwnValue(ValueConsumer consumer) {
-            Map<K, V> entries = new LinkedHashMap<>();
-            SideEffectBuilder<? super Map<K, V>> sideEffectBuilder = SideEffect.builder();
-            for (ExecutionTimeValue<? extends Map<? extends K, ? extends V>> executionTimeValue : values) {
-                Value<? extends Map<? extends K, ? extends V>> value = executionTimeValue.toProvider().calculateValue(consumer);
-                if (value.isMissing()) {
-                    return Value.missing();
-                } else {
-                    entries.putAll(value.getWithoutSideEffect());
-                    sideEffectBuilder.add(SideEffect.fixedFrom(value));
-                }
+        public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
+            Map<K, V> candidates = new LinkedHashMap<>();
+            // we cannot use dest directly because we don't want to emit any entries if either left or right are missing
+            Value<Void> value = delegate.collectEntries(consumer, collector, candidates);
+            if (value.isMissing()) {
+                return Value.present();
             }
+            dest.putAll(candidates);
+            return Value.present().withSideEffect(SideEffect.fixedFrom(value));
+        }
 
-            return Value.of(ImmutableMap.copyOf(entries)).withSideEffect(sideEffectBuilder.build());
+        @Override
+        public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collector, ImmutableCollection.Builder<K> dest) {
+            ImmutableSet.Builder<K> candidateKeys = ImmutableSet.builder();
+            Value<Void> value = delegate.collectKeys(consumer, collector, candidateKeys);
+            if (value.isMissing()) {
+                return Value.present();
+            }
+            dest.addAll(candidateKeys.build());
+            return value;
+        }
+
+        @Override
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
+            ExecutionTimeValue<? extends Map<? extends K, ? extends V>> executionTimeValue = delegate.calculateExecutionTimeValue();
+            return executionTimeValue.isMissing() ? ExecutionTimeValue.fixedValue(ImmutableMap.of()) : executionTimeValue;
+        }
+
+        @Override
+        public ValueProducer getProducer() {
+            return delegate.getProducer();
+        }
+
+        @Override
+        public boolean calculatePresence(ValueConsumer consumer) {
+            return true;
         }
     }
 
@@ -767,14 +901,8 @@ public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collect
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(entries).withSideEffect(sideEffect));
-        }
-
-        @Override
-        public MapCollector<K, V> absentIgnoring() {
-            // always present
-            return this;
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(entries).withSideEffect(sideEffect);
         }
 
         @Override
@@ -792,145 +920,4 @@ public String toString() {
             return entries.toString();
         }
     }
-
-    private static abstract class AbstractPlusCollector<K, V> implements MapCollector<K, V> {
-
-        protected final MapCollector<K, V> left;
-        protected final MapCollector<K, V> right;
-
-        private AbstractPlusCollector(MapCollector<K, V> left, MapCollector<K, V> right) {
-            this.left = left;
-            this.right = right;
-        }
-
-        @Override
-        public ValueProducer getProducer() {
-            return left.getProducer().plus(right.getProducer());
-        }
-
-        @Override
-        public String toString() {
-            return left + " + " + right;
-        }
-    }
-
-    private static class PlusCollector<K, V> extends AbstractPlusCollector<K, V> {
-
-        public PlusCollector(MapCollector<K, V> left, MapCollector<K, V> right) {
-            super(left, right);
-        }
-
-        @Override
-        public MapCollector<K, V> absentIgnoring() {
-            return new AbsentIgnoringPlusCollector<K, V>(left, right);
-        }
-
-        @Override
-        public boolean calculatePresence(ValueConsumer consumer) {
-            return left.calculatePresence(consumer) && right.calculatePresence(consumer);
-        }
-
-        @Override
-        public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
-            Value<Void> leftValue = left.collectEntries(consumer, collector, dest);
-            if (leftValue.isMissing()) {
-                return leftValue;
-            }
-            Value<Void> rightValue = right.collectEntries(consumer, collector, dest);
-            if (rightValue.isMissing()) {
-                return rightValue;
-            }
-
-            return Value.present()
-                .withSideEffect(SideEffect.fixedFrom(leftValue))
-                .withSideEffect(SideEffect.fixedFrom(rightValue));
-        }
-
-        @Override
-        public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collector, ImmutableCollection.Builder<K> dest) {
-            Value<Void> result = left.collectKeys(consumer, collector, dest);
-            if (result.isMissing()) {
-                return result;
-            }
-            return right.collectKeys(consumer, collector, dest);
-        }
-
-        @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
-            left.calculateExecutionTimeValue(visitor);
-            right.calculateExecutionTimeValue(visitor);
-        }
-    }
-
-    /**
-     * A plus collector that either produces a composition of both of its left and right sides,
-     * or Value.present() with empty content (if left or right side are missing).
-     */
-    private static class AbsentIgnoringPlusCollector<K, V> extends AbstractPlusCollector<K, V> {
-
-        private AbsentIgnoringPlusCollector(MapCollector<K, V> left, MapCollector<K, V> right) {
-            super(left, right);
-        }
-
-        @Override
-        public boolean calculatePresence(ValueConsumer consumer) {
-            return true;
-        }
-
-        @Override
-        public MapCollector<K, V> absentIgnoring() {
-            return this;
-        }
-
-        @Override
-        public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
-            Map<K, V> candidates = new LinkedHashMap<>();
-            // we cannot use dest directly because we don't want to emit any entries if either left or right are missing
-            Value<Void> leftValue = left.collectEntries(consumer, collector, candidates);
-            if (leftValue.isMissing()) {
-                return Value.present();
-            }
-            Value<Void> rightValue = right.collectEntries(consumer, collector, candidates);
-            if (rightValue.isMissing()) {
-                return Value.present();
-            }
-            dest.putAll(candidates);
-            return Value.present()
-                .withSideEffect(SideEffect.fixedFrom(leftValue))
-                .withSideEffect(SideEffect.fixedFrom(rightValue));
-        }
-
-        @Override
-        public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collector, ImmutableCollection.Builder<K> dest) {
-            ImmutableSet.Builder<K> candidateKeys = ImmutableSet.builder();
-            Value<Void> leftResult = left.collectKeys(consumer, collector, candidateKeys);
-            if (leftResult.isMissing()) {
-                return Value.present();
-            }
-            Value<Void> rightResult = right.collectKeys(consumer, collector, candidateKeys);
-            if (rightResult.isMissing()) {
-                return Value.present();
-            }
-            dest.addAll(candidateKeys.build());
-            return rightResult;
-        }
-
-        @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
-            boolean[] anyMissing = {false};
-            ImmutableList.Builder<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> toVisit = ImmutableList.builder();
-            Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> safeVisitor = value -> {
-                if (value.isMissing()) {
-                    anyMissing[0] = true;
-                } else {
-                    toVisit.add(value);
-                }
-            };
-            left.calculateExecutionTimeValue(safeVisitor);
-            right.calculateExecutionTimeValue(safeVisitor);
-            if (!anyMissing[0]) {
-                toVisit.build().forEach(it -> visitor.execute(it));
-            }
-        }
-    }
 }
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollector.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollector.java
index 09d5bfc..9414749 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollector.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollector.java
@@ -17,7 +17,6 @@
 package org.gradle.api.internal.provider;
 
 import com.google.common.collect.ImmutableCollection;
-import org.gradle.api.Action;
 
 import java.util.Map;
 
@@ -25,7 +24,7 @@
  * A supplier of zero or more mappings from value of type {@link K} to value of type {@link V}.
  *
  * <p>
- *     A <code>MapCollector</code> is for {@link DefaultMapProperty} and {@link MapSupplier} what {@link Collector} is for {@link AbstractCollectionProperty} and {@link CollectionSupplier}.
+ * A <code>MapCollector</code> is for {@link DefaultMapProperty} and {@link MapSupplier} what {@link Collector} is for {@link AbstractCollectionProperty} and {@link CollectionSupplier}.
  * </p>
  */
 public interface MapCollector<K, V> extends ValueSupplier {
@@ -34,14 +33,5 @@ public interface MapCollector<K, V> extends ValueSupplier {
 
     Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collector, ImmutableCollection.Builder<K> dest);
 
-    void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor);
-
-    MapCollector<K, V> absentIgnoring();
-
-    /**
-     * Returns a collector that may never return a missing value.
-     */
-    default MapCollector<K, V> absentIgnoringIfNeeded(boolean ignoreAbsent) {
-        return ignoreAbsent ? absentIgnoring() : this;
-    }
+    ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue();
 }
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollectors.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollectors.java
index b560041..8ba0241 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollectors.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapCollectors.java
@@ -19,11 +19,8 @@
 import com.google.common.base.Objects;
 import com.google.common.collect.ImmutableCollection;
 import com.google.common.collect.ImmutableMap;
-import org.gradle.api.Action;
 import org.gradle.api.internal.lambdas.SerializableLambdas;
-import org.gradle.internal.Cast;
 
-import javax.annotation.Nonnull;
 import java.util.Map;
 
 public class MapCollectors {
@@ -44,12 +41,6 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public MapCollector<K, V> absentIgnoring() {
-            // always present
-            return this;
-        }
-
-        @Override
         public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
             collector.add(key, value, dest);
             return Value.present();
@@ -62,8 +53,8 @@ public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collect
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
-            visitor.execute(ExecutionTimeValue.fixedValue(ImmutableMap.of(key, value)));
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(ImmutableMap.of(key, value));
         }
 
         @Override
@@ -104,11 +95,6 @@ public EntryWithValueFromProvider(K key, ProviderInternal<? extends V> providerO
         }
 
         @Override
-        public MapCollector<K, V> absentIgnoring() {
-            return new EntriesFromMapProvider<>(providerOfValue.map(value -> ImmutableMap.of(key, value)), true);
-        }
-
-        @Override
         public boolean calculatePresence(ValueConsumer consumer) {
             return providerOfValue.calculatePresence(consumer);
         }
@@ -134,16 +120,16 @@ public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collect
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
             ExecutionTimeValue<? extends V> value = providerOfValue.calculateExecutionTimeValue();
             if (value.isMissing()) {
-                visitor.execute(ExecutionTimeValue.missing());
+                return ExecutionTimeValue.missing();
             } else if (value.hasFixedValue()) {
                 // transform preserving side effects
-                visitor.execute(ExecutionTimeValue.value(value.toValue().transform(v -> ImmutableMap.of(key, v))));
+                return ExecutionTimeValue.value(value.toValue().transform(v -> ImmutableMap.of(key, v)));
             } else {
-                visitor.execute(ExecutionTimeValue.changingValue(
-                    value.getChangingValue().map(SerializableLambdas.transformer(v -> ImmutableMap.of(key, v)))));
+                return ExecutionTimeValue.changingValue(
+                    value.getChangingValue().map(SerializableLambdas.transformer(v -> ImmutableMap.of(key, v))));
             }
         }
 
@@ -172,12 +158,6 @@ public boolean calculatePresence(ValueConsumer consumer) {
         }
 
         @Override
-        public MapCollector<K, V> absentIgnoring() {
-            // always present
-            return this;
-        }
-
-        @Override
         public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
             collector.addAll(entries.entrySet(), dest);
             return Value.present();
@@ -190,8 +170,8 @@ public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collect
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> sources) {
-            sources.execute(ExecutionTimeValue.fixedValue(entries));
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
+            return ExecutionTimeValue.fixedValue(entries);
         }
 
         @Override
@@ -208,37 +188,21 @@ public String toString() {
     public static class EntriesFromMapProvider<K, V> implements MapCollector<K, V> {
 
         private final ProviderInternal<? extends Map<? extends K, ? extends V>> providerOfEntries;
-        private final boolean ignoreAbsent;
 
         public EntriesFromMapProvider(ProviderInternal<? extends Map<? extends K, ? extends V>> providerOfEntries) {
-            this(providerOfEntries, false);
-        }
-
-        private EntriesFromMapProvider(ProviderInternal<? extends Map<? extends K, ? extends V>> providerOfEntries, boolean ignoreAbsent) {
-            this.providerOfEntries = ignoreAbsent ? neverMissing(Cast.uncheckedNonnullCast(providerOfEntries)) : providerOfEntries;
-            this.ignoreAbsent = ignoreAbsent;
-        }
-
-        @Override
-        public MapCollector<K, V> absentIgnoring() {
-            return ignoreAbsent ? this : new EntriesFromMapProvider<>(providerOfEntries, true);
-        }
-
-        @Nonnull
-        private static <K, V> ProviderInternal<? extends Map<? extends K, ? extends V>> neverMissing(ProviderInternal<Map<? extends K, ? extends V>> provider) {
-            return Cast.uncheckedNonnullCast(provider.orElse(ImmutableMap.of()));
+            this.providerOfEntries = providerOfEntries;
         }
 
         @Override
         public boolean calculatePresence(ValueConsumer consumer) {
-            return ignoreAbsent || providerOfEntries.calculatePresence(consumer);
+            return providerOfEntries.calculatePresence(consumer);
         }
 
         @Override
         public Value<Void> collectEntries(ValueConsumer consumer, MapEntryCollector<K, V> collector, Map<K, V> dest) {
             Value<? extends Map<? extends K, ? extends V>> value = providerOfEntries.calculateValue(consumer);
             if (value.isMissing()) {
-                return ignoreAbsent ? Value.present() : value.asType();
+                return value.asType();
             }
             collector.addAll(value.getWithoutSideEffect().entrySet(), dest);
             return Value.present().withSideEffect(SideEffect.fixedFrom(value));
@@ -255,8 +219,8 @@ public Value<Void> collectKeys(ValueConsumer consumer, ValueCollector<K> collect
         }
 
         @Override
-        public void calculateExecutionTimeValue(Action<ExecutionTimeValue<? extends Map<? extends K, ? extends V>>> visitor) {
-            visitor.execute(providerOfEntries.calculateExecutionTimeValue());
+        public ExecutionTimeValue<? extends Map<? extends K, ? extends V>> calculateExecutionTimeValue() {
+            return providerOfEntries.calculateExecutionTimeValue();
         }
 
         @Override
diff --git a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapSupplier.java b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapSupplier.java
index ce6c982..bd4ef626 100644
--- a/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapSupplier.java
+++ b/platforms/core-configuration/model-core/src/main/java/org/gradle/api/internal/provider/MapSupplier.java
@@ -30,16 +30,7 @@ interface MapSupplier<K, V> extends ValueSupplier {
 
     Value<? extends Set<K>> calculateKeys(ValueConsumer consumer);
 
-    MapSupplier<K, V> plus(MapCollector<K, V> collector);
-
-    /**
-     * Returns a view of this supplier that may calculate its value as empty if it would be missing.
-     */
-    MapSupplier<K, V> absentIgnoring();
-
-    default MapSupplier<K, V> absentIgnoringIfNeeded(boolean required) {
-        return required ? absentIgnoring() : this;
-    }
+    MapSupplier<K, V> plus(MapCollector<K, V> collector, boolean ignoreAbsent);
 
     ExecutionTimeValue<? extends Map<K, V>> calculateExecutionTimeValue();
 }
diff --git a/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/CollectionPropertySpec.groovy b/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/CollectionPropertySpec.groovy
index a891904..4a7bddf 100644
--- a/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/CollectionPropertySpec.groovy
+++ b/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/CollectionPropertySpec.groovy
@@ -29,6 +29,7 @@
 import java.util.function.Consumer
 
 import static org.gradle.api.internal.provider.CircularEvaluationSpec.ProviderConsumer.GET_PRODUCER
+import static org.gradle.api.internal.provider.CircularEvaluationSpec.ProviderConsumer.TO_STRING
 import static org.gradle.api.internal.provider.Providers.notDefined
 
 abstract class CollectionPropertySpec<C extends Collection<String>> extends PropertySpec<C> {
@@ -1165,6 +1166,11 @@
 
     static abstract class CollectionPropertyCircularChainEvaluationTest<T, C extends Collection<T>> extends PropertySpec.PropertyCircularChainEvaluationSpec<C> {
         @Override
+        List<Consumer<ProviderInternal<?>>> safeConsumers() {
+            return [TO_STRING, GET_PRODUCER]
+        }
+
+        @Override
         abstract AbstractCollectionProperty<T, C> property()
 
         def "calling #consumer throws exception if added item provider references the property"(
@@ -1385,48 +1391,48 @@
         }
 
         when:
-        operations.each {operation -> operation.call(property) }
+        operations.each { operation -> operation.call(property) }
 
         then:
         expected == null || property.getOrNull() == toImmutable(expected)
         expected != null || !property.present
 
         where:
-        expected    | explicit      | convention    | label                                             | operations
-        ["1"]       | _             | _             | "add"                                             | { it.add("1") }
-        ["1"]       | _             | _             | "append"                                          | { it.append("1") }
-        ["1"]       | []            | _             | "add to empty"                                    | { it.add("1") }
-        ["1"]       | []            | _             | "append to empty"                                 | { it.append("1") }
-        ["1"]       | _             | []            | "add to empty convention"                         | { it.add("1") }
-        ["1"]       | _             | []            | "append to empty convention"                      | { it.append("1") }
-        null        | null          | []            | "add to unset value w/ empty convention"          | { it.add("1") }
-        ["1"]       | null          | []            | "append to unset value w/ empty convention"       | { it.append("1") }
-        ["1"]       | _             | ["0"]         | "add to non-empty convention"                     | { it.add("1") }
-        ["0", "1"]  | _             | ["0"]         | "append to non-empty convention"                  | { it.append("1") }
-        null        | null          | ["0"]         | "add to unset value w/ non-empty convention"      | { it.add("1") }
-        ["0", "1"]  | null          | ["0"]         | "append to unset value w/ non-empty convention"   | { it.append("1") }
-        null        | notDefined()  | _             | "add to missing"                                  | { it.add("1") }
-        ["1"]       | notDefined()  | _             | "append to missing"                               | { it.append("1") }
-        null        | notDefined()  | ["0"]         | "add to missing w/ non-empty convention"          | { it.add("1") }
-        ["1"]       | notDefined()  | ["0"]         | "append to missing w/ non-empty convention"       | { it.append("1") }
-        null        | []            | _             | "add missing to empty value"                      | { it.add(notDefined()) }
-        []          | []            | _             | "append missing to empty value"                   | { it.append(notDefined()) }
-        null        | _             | _             | "add missing"                                     | { it.add(notDefined()) }
-        []          | _             | _             | "append missing"                                  | { it.append(notDefined()) }
-        ["1"]       | _             | _             | "add missing, then append"                        | { it.add(notDefined()) ; it.append("1") }
-        ["1"]       | _             | _             | "append missing, then add"                        | { it.append(notDefined()) ; it.add("1") }
-        ["1"]       | ["0"]         | _             | "add missing to non-empty value, then append"     | { it.add(notDefined()) ; it.append("1") }
-        ["0", "1"]  | ["0"]         | _             | "append missing to non-empty value, then add"     | { it.append(notDefined()) ; it.add("1") }
-        ["1"]       | _             | ["0"]         | "add missing to non-empty convention, then append"| { it.add(notDefined()) ; it.append("1") }
-        ["0", "1"]  | _             | ["0"]         | "append missing to non-empty convention, then add"| { it.append(notDefined()) ; it.add("1") }
-        ["1"]       | _             | _             | "add, then append missing"                        | { it.add("1") ; it.append(notDefined()) }
-        null        | _             | _             | "append, then add missing"                        | { it.append("1") ; it.add(notDefined()) }
-        ["0", "1"]  | ["0"]         | _             | "add to non-empty value, then append missing"     | { it.add("1") ; it.append(notDefined()) }
-        null        | ["0"]         | _             | "append to non-empty value, then add missing"     | { it.append("1") ; it.add(notDefined()) }
-        ["1"]       | _             | ["0"]         | "add to non-empty convention, then append missing"| { it.add("1") ; it.append(notDefined()) }
-        null        | _             | ["0"]         | "append to non-empty conventio, then add missing" | { it.append("1") ; it.add(notDefined()) }
-        ["1"]       | _             | _             | "add, then add missing, then append"              | { it.add("0") ; it.add(notDefined()) ; it.append("1") }
-        ["0", "1"]  | _             | _             | "add, then append missing, then add"              | { it.add("0") ; it.append(notDefined()) ; it.add("1") }
+        expected   | explicit     | convention | label                                              | operations
+        ["1"]      | _            | _          | "add"                                              | { it.add("1") }
+        ["1"]      | _            | _          | "append"                                           | { it.append("1") }
+        ["1"]      | []           | _          | "add to empty"                                     | { it.add("1") }
+        ["1"]      | []           | _          | "append to empty"                                  | { it.append("1") }
+        ["1"]      | _            | []         | "add to empty convention"                          | { it.add("1") }
+        ["1"]      | _            | []         | "append to empty convention"                       | { it.append("1") }
+        null       | null         | []         | "add to unset value w/ empty convention"           | { it.add("1") }
+        ["1"]      | null         | []         | "append to unset value w/ empty convention"        | { it.append("1") }
+        ["1"]      | _            | ["0"]      | "add to non-empty convention"                      | { it.add("1") }
+        ["0", "1"] | _            | ["0"]      | "append to non-empty convention"                   | { it.append("1") }
+        null       | null         | ["0"]      | "add to unset value w/ non-empty convention"       | { it.add("1") }
+        ["0", "1"] | null         | ["0"]      | "append to unset value w/ non-empty convention"    | { it.append("1") }
+        null       | notDefined() | _          | "add to missing"                                   | { it.add("1") }
+        ["1"]      | notDefined() | _          | "append to missing"                                | { it.append("1") }
+        null       | notDefined() | ["0"]      | "add to missing w/ non-empty convention"           | { it.add("1") }
+        ["1"]      | notDefined() | ["0"]      | "append to missing w/ non-empty convention"        | { it.append("1") }
+        null       | []           | _          | "add missing to empty value"                       | { it.add(notDefined()) }
+        []         | []           | _          | "append missing to empty value"                    | { it.append(notDefined()) }
+        null       | _            | _          | "add missing"                                      | { it.add(notDefined()) }
+        []         | _            | _          | "append missing"                                   | { it.append(notDefined()) }
+        ["1"]      | _            | _          | "add missing, then append"                         | { it.add(notDefined()); it.append("1") }
+        ["1"]      | _            | _          | "append missing, then add"                         | { it.append(notDefined()); it.add("1") }
+        ["1"]      | ["0"]        | _          | "add missing to non-empty value, then append"      | { it.add(notDefined()); it.append("1") }
+        ["0", "1"] | ["0"]        | _          | "append missing to non-empty value, then add"      | { it.append(notDefined()); it.add("1") }
+        ["1"]      | _            | ["0"]      | "add missing to non-empty convention, then append" | { it.add(notDefined()); it.append("1") }
+        ["0", "1"] | _            | ["0"]      | "append missing to non-empty convention, then add" | { it.append(notDefined()); it.add("1") }
+        ["1"]      | _            | _          | "add, then append missing"                         | { it.add("1"); it.append(notDefined()) }
+        null       | _            | _          | "append, then add missing"                         | { it.append("1"); it.add(notDefined()) }
+        ["0", "1"] | ["0"]        | _          | "add to non-empty value, then append missing"      | { it.add("1"); it.append(notDefined()) }
+        null       | ["0"]        | _          | "append to non-empty value, then add missing"      | { it.append("1"); it.add(notDefined()) }
+        ["1"]      | _            | ["0"]      | "add to non-empty convention, then append missing" | { it.add("1"); it.append(notDefined()) }
+        null       | _            | ["0"]      | "append to non-empty convention, then add missing" | { it.append("1"); it.add(notDefined()) }
+        ["1"]      | _            | _          | "add, then add missing, then append"               | { it.add("0"); it.add(notDefined()); it.append("1") }
+        ["0", "1"] | _            | _          | "add, then append missing, then add"               | { it.add("0"); it.append(notDefined()); it.add("1") }
     }
 
     def "execution time value is present if only undefined-safe operations are performed"() {
@@ -1445,7 +1451,7 @@
         def execTimeValue = property.calculateExecutionTimeValue()
 
         then:
-        assertCollectionIs(toImmutable(['2', '3', '4']), execTimeValue.toValue().get())
+        assertCollectionIs(execTimeValue.toValue().get(), toImmutable(['2', '3', '4']),)
     }
 
     def "property restores undefined-safe items"() {
@@ -1468,13 +1474,53 @@
         null  | ["1", "3"]
     }
 
+    def "property restores undefined-safe items that have #numProviders changing providers without values in front"() {
+        given:
+        numProviders.times {
+            property.add(Providers.changing { null as String })
+        }
+        property.append(Providers.changing { "1" })
+        property.add("2")
+
+        when:
+        def property2 = property()
+        property2.fromState(property.calculateExecutionTimeValue())
+
+        then:
+        assertValueIs(toImmutable(["1", "2"]), property2)
+
+        where:
+        // TODO(mlopatkin): Original implementation works only with numProviders == 1
+        numProviders << [1, 2]
+    }
+
+    def "#opName to empty property is undefined-safe"() {
+        given:
+        property.set(null as Iterable)
+
+        when:
+        op(property)
+
+        then:
+        assertValueIs(toImmutable(expected), property)
+
+        where:
+        opName                      | op                                         | expected
+        "append(1)"                 | { it.append("1") }                         | ["1"]
+        "appendAll(1, 2)"           | { it.appendAll(["1", "2"]) }               | ["1", "2"]
+        "append(provider(1))"       | { it.append(Providers.of("1")) }           | ["1"]
+        "appendAll(provider(1, 2))" | { it.appendAll(Providers.of(["1", "2"])) } | ["1", "2"]
+        "append(notDefined())"      | { it.append(notDefined()) }                | []
+        "appendAll(notDefined())"   | { it.appendAll(notDefined()) }             | []
+    }
+
     def "property remains undefined-safe after restored"() {
         given:
         property.append(notDefined())
         property.add("2")
         property.append(notDefined())
         property.append(notDefined())
-        property.addAll(supplierWithChangingExecutionTimeValues(['3'], ['3a'], ['3b'], ['3c'], ['3d']))
+        property.addAll(supplierWithChangingExecutionTimeValues(['3a'], ['3b'], ['3c'], ['3d']))
         property.addAll(supplierWithValues(['4']))
         property.append(notDefined())
 
@@ -1500,7 +1546,7 @@
         property3.fromState(execTimeValue2)
 
         then:
-        assertValueIs(['2', '3d', '4', '5', '6'], property3)
+        assertValueIs(['2', '3c', '4', '5', '6'], property3)
     }
 
     def "can alternate append and add"() {
@@ -1599,4 +1645,15 @@
         assertValueIs([])
         !property.explicit
     }
+
+    def "can add a lot of providers"() {
+        given:
+        (0..<100000).each {
+            property.addAll(supplierWithProducer(Mock(Task), toImmutable([it.toString()])))
+        }
+
+        expect:
+        property.get().size() == 100000
+        property.getProducer().visitProducerTasks {}
+    }
 }
diff --git a/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/MapPropertySpec.groovy b/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/MapPropertySpec.groovy
index 9ca2066..d38716d 100644
--- a/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/MapPropertySpec.groovy
+++ b/platforms/core-configuration/model-core/src/test/groovy/org/gradle/api/internal/provider/MapPropertySpec.groovy
@@ -1536,7 +1536,7 @@
         property.put("a", notDefined())
         property.insert("b", "2")
         // changing execution time values or else we would end up with fixed values
-        property.putAll(supplierWithChangingExecutionTimeValues([c: '3'], [c: '3a'], [c: '3b'], [c: '3c'], [c: '3d'], [c: '3e']))
+        property.putAll(supplierWithChangingExecutionTimeValues([c: '3a'], [c: '3b'], [c: '3c'], [c: '3d'], [c: '3e']))
         property.putAll(supplierWithValues([d: '4']))
         property.insert("e", notDefined())
 
@@ -1562,7 +1562,7 @@
         property3.fromState(execTimeValue2)
 
         then:
-        assertValueIs([b: '2', c: '3d', d: '4', f: '6', g: '7'], property3)
+        assertValueIs([b: '2', c: '3c', d: '4', f: '6', g: '7'], property3)
     }
 
     def "keySet provider has some values when property with no value is added via insert"() {
@@ -1975,4 +1975,15 @@
         // The following case abuses Groovy lax type-checking to put an invalid value into the property.
         "[k: (Object) provider {v}]" | { property().value(k: Providers.of("v")) }              || "Map(String->String, {k=fixed(class ${String.name}, v)})"
     }
+
+    def "can add a lot of providers"() {
+        given:
+        (0..<100000).each {
+            property.putAll(supplierWithProducer(Mock(Task), ImmutableMap.of(it.toString(), it.toString())))
+        }
+
+        expect:
+        property.get().size() == 100000
+        property.getProducer().visitProducerTasks {}
+    }
 }
diff --git a/platforms/core-execution/persistent-cache/src/integTest/groovy/org/gradle/cache/internal/DefaultFileLockManagerContentionIntegrationTest.groovy b/platforms/core-execution/persistent-cache/src/integTest/groovy/org/gradle/cache/internal/DefaultFileLockManagerContentionIntegrationTest.groovy
index cf6403c..a840b80 100644
--- a/platforms/core-execution/persistent-cache/src/integTest/groovy/org/gradle/cache/internal/DefaultFileLockManagerContentionIntegrationTest.groovy
+++ b/platforms/core-execution/persistent-cache/src/integTest/groovy/org/gradle/cache/internal/DefaultFileLockManagerContentionIntegrationTest.groovy
@@ -28,7 +28,6 @@
 import org.gradle.internal.concurrent.DefaultExecutorFactory
 import org.gradle.internal.remote.internal.inet.InetAddressFactory
 import org.gradle.internal.time.Time
-import org.gradle.test.fixtures.Flaky
 
 import java.util.concurrent.Executors
 import java.util.concurrent.TimeUnit
@@ -38,7 +37,6 @@
 import static org.gradle.test.fixtures.ConcurrentTestUtil.poll
 import static org.gradle.util.internal.TextUtil.escapeString
 
-@Flaky(because = "https://github.com/gradle/gradle-private/issues/4441")
 class DefaultFileLockManagerContentionIntegrationTest extends AbstractIntegrationSpec {
     def addressFactory = new InetAddressFactory()
 
@@ -90,13 +88,16 @@
         when:
         def build = executer.withTasks("lock").start()
         def timer = Time.startTimer()
+
+        def pingCountInOutput = 0
         poll(120) {
-            assert (build.standardOutput =~ 'Pinged owner at port').count == 3
+            pingCountInOutput = (build.standardOutput =~ 'Pinged owner at port').count
+            assert pingCountInOutput >= 3
         }
         receivingLock.close()
         then:
         build.waitForFinish()
-        pingRequestCount == 3 || pingRequestCount == 4
+        pingRequestCount in [pingCountInOutput, pingCountInOutput + 1]
         timer.elapsedMillis > 3000 // See: DefaultFileLockContentionHandler.PING_DELAY
     }
 
diff --git a/platforms/core-runtime/launcher/src/main/resources/release-features.txt b/platforms/core-runtime/launcher/src/main/resources/release-features.txt
index e69de29..abb8366 100644
--- a/platforms/core-runtime/launcher/src/main/resources/release-features.txt
+++ b/platforms/core-runtime/launcher/src/main/resources/release-features.txt
@@ -0,0 +1,3 @@
+ - Enhanced error and warning reporting with the Problems API
+ - File-system watching support on Alpine Linux
+ - Build and test Swift 6 libraries and apps
diff --git a/platforms/core-runtime/native/src/main/java/org/gradle/internal/nativeintegration/services/NativeServices.java b/platforms/core-runtime/native/src/main/java/org/gradle/internal/nativeintegration/services/NativeServices.java
index 629e6d4..3eb5309 100644
--- a/platforms/core-runtime/native/src/main/java/org/gradle/internal/nativeintegration/services/NativeServices.java
+++ b/platforms/core-runtime/native/src/main/java/org/gradle/internal/nativeintegration/services/NativeServices.java
@@ -99,6 +99,14 @@ public enum NativeFeatures {
             @Override
             public boolean initialize(File nativeBaseDir, boolean useNativeIntegrations) {
                 if (useNativeIntegrations) {
+                    OperatingSystem operatingSystem = OperatingSystem.current();
+                    if (operatingSystem.isMacOsX()) {
+                        String version = operatingSystem.getVersion();
+                        if (VersionNumber.parse(version).getMajor() < 12) {
+                            LOGGER.info("Disabling file system watching on macOS {}, as it is only supported for macOS 12+", version);
+                            return false;
+                        }
+                    }
                     try {
                         FileEvents.init(nativeBaseDir);
                         LOGGER.info("Initialized file system watching services in: {}", nativeBaseDir);
@@ -475,14 +483,6 @@ public boolean useNativeIntegrations() {
 
             @Override
             public boolean useFileSystemWatching() {
-                OperatingSystem operatingSystem = OperatingSystem.current();
-                if (operatingSystem.isMacOsX()) {
-                    String version = operatingSystem.getVersion();
-                    if (VersionNumber.parse(version).getMajor() < 12) {
-                        LOGGER.info("Disabling file system watching on macOS {}, as it is only supported for macOS 12+", version);
-                        return false;
-                    }
-                }
                 return isFeatureEnabled(NativeFeatures.FILE_SYSTEM_WATCHING);
             }
         };
diff --git a/platforms/documentation/docs/src/docs/release/notes-template.md b/platforms/documentation/docs/src/docs/release/notes-template.md
index 6382a3a..9686095 100644
--- a/platforms/documentation/docs/src/docs/release/notes-template.md
+++ b/platforms/documentation/docs/src/docs/release/notes-template.md
@@ -1,3 +1,7 @@
+<meta property="og:image" content="" />
+<meta property="og:type"  content="website" />
+<meta property="og:title" content="" />
+
 The Gradle team is excited to announce Gradle @version@.
 
 This release features [1](), [2](), ... [n](), and more.
@@ -96,6 +100,6 @@
 ## Reporting problems
 
 If you find a problem with this release, please file a bug on [GitHub Issues](https://github.com/gradle/gradle/issues) adhering to our issue guidelines.
-If you're not sure you're encountering a bug, please use the [forum](https://discuss.gradle.org/c/help-discuss).
+If you're not sure if you're encountering a bug, please use the [forum](https://discuss.gradle.org/c/help-discuss).
 
 We hope you will build happiness with Gradle, and we look forward to your feedback via [Twitter](https://twitter.com/gradle) or on [GitHub](https://github.com/gradle).
diff --git a/platforms/documentation/docs/src/docs/release/notes.md b/platforms/documentation/docs/src/docs/release/notes.md
index 98c3b20..2f93ef3 100644
--- a/platforms/documentation/docs/src/docs/release/notes.md
+++ b/platforms/documentation/docs/src/docs/release/notes.md
@@ -1,6 +1,14 @@
+<meta property="og:image" content="https://gradle.org/images/releases/gradle-8.12" />
+<meta property="og:type"  content="website" />
+<meta property="og:title" content="Gradle 8.12 Release Notes" />
+
 The Gradle team is excited to announce Gradle @version@.
 
-This release features [1](), [2](), ... [n](), and more.
+This release improves [error and warning reporting](#error-warning) by summarizing duplicate entries in the Problems API's generated problems report for better readability. The [console output](#build-authoring) is also enhanced when the Problems API is used to fail the build.
+
+Gradle @version@ introduces [platform enhancements](#platform), including file-system watching support on the Alpine Linux distribution and support for building and testing Swift 6 applications.
+
+Additionally, artifact transform ambiguities now produce a [deprecation warning](#error-warning) with clearer, more actionable information and [new methods](#build-authoring) are available in the DependencyConstraint API.
 
 <!--
 Include only their name, impactful features should be called out separately below.
@@ -10,6 +18,25 @@
 -->
 
 We would like to thank the following community members for their contributions to this release of Gradle:
+[Abhiraj Adhikary](https://github.com/abhirajadhikary06),
+[Ayush Saxena](https://github.com/Ayushcode10),
+[Björn Kautler](https://github.com/Vampire),
+[davidburstrom](https://github.com/davidburstrom),
+[Dominic Fellbaum](https://github.com/felldo),
+[Emmanuel Ferdman](https://github.com/emmanuel-ferdman),
+[Finn Petersen](https://github.com/fp7),
+[Johnny Lim](https://github.com/izeye),
+[Mahdi Hosseinzadeh](https://github.com/mahozad),
+[Martin Bonnin](https://github.com/martinbonnin),
+[Paint_Ninja](https://github.com/PaintNinja),
+[Petter Måhlén](https://github.com/pettermahlen),
+[Philip Wedemann](https://github.com/hfhbd),
+[stegeto22](https://github.com/stegeto22),
+[Tanish](https://github.com/Taz03),
+[TheGoesen](https://github.com/TheGoesen),
+[Tim Nielens](https://github.com/tnielens),
+[Trout Zhang](https://github.com/TroutZhang),
+[Victor Merkulov](https://github.com/urdak)
 
 Be sure to check out the [public roadmap](https://blog.gradle.org/roadmap-announcement) for insight into what's planned for future releases.
 
@@ -25,11 +52,24 @@
 
 ## New features and usability improvements
 
+<a name="error-warning"></a>
 ### Error and warning reporting improvements
 
+Gradle provides a rich set of [error and warning messages](userguide/logging.html) to help you understand and resolve problems in your build.
+
+#### Summarization in the HTML report for problems
+
+The [Problems API](userguide/reporting_problems.html) provides structured feedback on build issues, helping developers and tools like IDEs identify and resolve warnings, errors, or deprecations during configuration or runtime.
+
+This release introduces a new problem summarization mechanism that reduces redundancy in the generated [HTML Problems Report](userguide/reporting_problems.html#sec:generated_html_report).
+
+The feature limits the number of identical problems reported and provides a summarized count of additional occurrences in the summary report:
+
+<img alt="HTML Problems report" src="">
+
 #### Ambiguous Artifact Transformation chains are detected and reported
 
-Previously, when two or more equal-length chains of <<artifact_transforms.adoc#sec,artifact transforms>> produced compatible variants to satisfy a resolution request, Gradle would arbitrarily and silently select one.
+Previously, when two or more equal-length chains of [artifact transforms](userguide/artifact_transforms.html) produced compatible variants to satisfy a resolution request, Gradle would arbitrarily and silently select one.
 Gradle now emits a warning for this case.
 
 This deprecation warning is the same failure message that now appears when multiple equal-length chains are available, producing incompatible variants that could each satisfy a resolution request.
@@ -85,108 +125,91 @@
 ```
 
 The formatting of this message has been improved to comprehensively display information about each complete chain of transformations that produces the candidates that would satisfy the request.
-This allows authors to better analyze and understand their builds, allowing them to remove the ambiguity.
+This allows authors to better analyze and understand their builds, allowing them to remove ambiguity.
 
-<!-- Do not add breaking changes or deprecations here! Add them to the upgrade guide instead. -->
+<a name="platform"></a>
+### Platform enhancements
 
+Gradle provides many features for specific platforms and languages.
 
-<a name="Problems API"></a>
+#### File-system watching and continuous mode support on Alpine Linux
 
-### Problems API improvements
+[File-system watching](userguide/file_system_watching.html) is now supported on [Alpine Linux](https://alpinelinux.org), a popular choice for container-based images and the default distribution for Docker.
 
-#### Enhanced Problem Summarization for Improved Usability
+The feature is enabled by default, as is the case with all other supported platforms.
 
-This release introduces a new problem summarization mechanism that reduces redundancy in problem reporting during builds. The feature limits the number of identical problems reported for each group (
-default threshold: 15) and provides a summarized count of additional occurrences at the end of the build.
+Additionally, it is now possible to [run builds in continuous mode](userguide/continuous_builds.html) on Alpine Linux.
 
-##### Key Improvements
+<a name="swift-support"></a>
+#### Swift 6 support
 
-- **Optimized Event Reporting**: Summarized problems minimize the data sent to clients, improving performance and reducing resource consumption.
-- **Simplified Developer Experience**: Cleaner problem reports with less noise, making it easier to identify and address critical issues.
-- **Enhanced Reporting**: Summarized problem details are reflected in the Messages, Group, and Locations tabs, maintaining clarity while reducing verbosity.
+Gradle’s [Swift support](userguide/building_swift_projects.html) allows you to build and test native Swift libraries and applications.
 
-This change ensures a smoother experience when using Gradle with repetitive problems.
-
-To learn more, check out our [sample project](samples/sample_problems_api_usage.html)
-<!--
-================== TEMPLATE ==============================
-
-<a name="FILL-IN-KEY-AREA"></a>
-### FILL-IN-KEY-AREA improvements
-
-<<<FILL IN CONTEXT FOR KEY AREA>>>
-Example:
-> The [configuration cache](userguide/configuration_cache.html) improves build performance by caching the result of
-> the configuration phase. Using the configuration cache, Gradle can skip the configuration phase entirely when
-> nothing that affects the build configuration has changed.
-
-#### FILL-IN-FEATURE
-> HIGHLIGHT the use case or existing problem the feature solves
-> EXPLAIN how the new release addresses that problem or use case
-> PROVIDE a screenshot or snippet illustrating the new feature, if applicable
-> LINK to the full documentation for more details
-
-================== END TEMPLATE ==========================
-
-
-==========================================================
-ADD RELEASE FEATURES BELOW
-vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv -->
+Gradle now supports [Swift 6](https://www.swift.org/blog/announcing-swift-6/), introduced with [Xcode 16.0](https://developer.apple.com/documentation/xcode-release-notes/xcode-16-release-notes), extending its capabilities to the latest major version of Swift.
 
 <a name="build-authoring"></a>
 ### Build authoring improvements
 
-Gradle provides rich APIs for plugin authors and build engineers to develop custom build logic.
+Gradle provides [rich APIs](userguide/getting_started_dev.html) for plugin authors and build engineers to develop custom build logic.
+
+#### Richer console output for failures using the Problems API
+
+The [Problems API](userguide/reporting_problems.html) provides structured feedback on build issues, helping developers and tools like IDEs identify and resolve warnings, errors, or deprecations during configuration or runtime.
+
+With this release, problems that are the source of a build failure have all of their information displayed on the console output at the end of the build:
+
+```text
+FAILURE: Build failed with an exception.
+
+* What went wrong:
+Execution failed for task ':sample-project:myFailingTask'.
+> Message from runtime exception
+    This happened because ProblemReporter.throwing() was called
+      This is a demonstration of how to add
+      detailed information to a build failure
+
+* Try:
+> Remove the Problems.throwing() method call from the task action
+> Run with --scan to get full insights.
+
+BUILD FAILED in 10s
+```
+
+This example output was obtained by using the Problems API as shown below:
+
+```java
+public abstract class FailingTask extends DefaultTask {
+
+
+    @Inject public abstract Problems getProblems();
+
+
+    @TaskAction public void run() {
+        throw getProblems().getReporter().throwing(problemSpec -> {
+            problemSpec.contextualLabel("This happened because ProblemReporter.throwing() was called");
+            problemSpec.details("This is a demonstration of how to add\ndetailed information to a build failure");
+            problemSpec.withException(new RuntimeException("Message from runtime exception"));
+            problemSpec.solution("Remove the Problems.throwing() method call from the task action");
+        });
+    }
+}
+```
+
+Check out our [sample project](samples/sample_problems_api_usage.html) for the complete code.
+
+This will enable plugin authors to fully leverage the Problems API to enhance any error with additional details, documentation links, and possible resolution steps.
+
+See the [Problems API](userguide/reporting_problems.html#command_line_interface) for more information.
 
 #### `DependencyConstraintHandler` now has `addProvider` methods
 
 The [`DependencyConstraintHandler`](javadoc/org/gradle/api/artifacts/dsl/DependencyConstraintHandler.html) now has `addProvider` methods, similar to the
 [`DependencyHandler`](javadoc/org/gradle/api/artifacts/dsl/DependencyHandler.html).
 
-```kotlin
-dependencies {
-    constraints {
-        // Existing API:
-        add("implementation", provider { "org.foo:bar:1.0" })
-        add("implementation", provider { "org.foo:bar:1.0" }) {
-            because("newer versions have bugs")
-        }
-        // New methods:
-        addProvider("implementation", provider { "org.foo:bar:1.0" })
-        addProvider("implementation", provider { "org.foo:bar:1.0" }) {
-            because("newer versions have bugs")
-        }
-    }
-}
-```
-
-This clarifies that adding a provider is possible, and that there is no immediately usable return value. The ability to pass a provider to `DependencyConstraintHandler.add` is unaffected.
-
-### Other improvements
-
-#### File-system watching and continuous mode support on Alpine Linux
-
-[File-system watching](userguide/file_system_watching.html) is now supported on Alpine Linux.
-The feature is enabled by default, as on all other supported platforms.
-
-It is now also possible to [run builds in continuous mode](userguide/continuous_builds.html) on Alpine.
-
-<a name="swift-support"></a>
-### Swift support
-
-Gradle’s [Swift support](userguide/building_swift_projects.html) allows building and testing native Swift libraries and applications.
-
-#### Basic Swift 6 support
-
-Gradle now supports [Swift 6](https://www.swift.org/blog/announcing-swift-6/), introduced with [Xcode 16.0](https://developer.apple.com/documentation/xcode-release-notes/xcode-16-release-notes), extending its capabilities to the latest major version of Swift.
-
-<!-- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
-ADD RELEASE FEATURES ABOVE
-==========================================================
-
--->
+These are useful in plugin code to bring attention to where inputs should and should not be lazily evaluated by preventing eager results from being passed in.
 
 ## Promoted features
+
 Promoted features are features that were incubating in previous versions of Gradle but are now supported and subject to backward compatibility.
 See the User Manual section on the “[Feature Lifecycle](userguide/feature_lifecycle.html)” for more information.
 
@@ -194,8 +217,9 @@
 
 ### Service reference properties are now stable
 
-Service references are task properties meant for easier consumption of [shared build services](userguide/build_services.html#sec:service_references).
-[`ServiceReference`](/javadoc/org/gradle/api/services/ServiceReference.html) is now stable.
+Service references are task properties meant to facilitate the consumption of [shared build services](userguide/build_services.html#sec:service_references).
+
+[`ServiceReference`](javadoc/org/gradle/api/services/ServiceReference.html) is now stable.
 
 ## Fixed issues
 
@@ -218,6 +242,6 @@
 ## Reporting problems
 
 If you find a problem with this release, please file a bug on [GitHub Issues](https://github.com/gradle/gradle/issues) adhering to our issue guidelines.
-If you're not sure you're encountering a bug, please use the [forum](https://discuss.gradle.org/c/help-discuss).
+If you're not sure if you're encountering a bug, please use the [forum](https://discuss.gradle.org/c/help-discuss).
 
 We hope you will build happiness with Gradle, and we look forward to your feedback via [Twitter](https://twitter.com/gradle) or on [GitHub](https://github.com/gradle).
diff --git a/platforms/documentation/docs/src/docs/userguide/authoring-builds/plugins/reporting_problems.adoc b/platforms/documentation/docs/src/docs/userguide/authoring-builds/plugins/reporting_problems.adoc
index ee8ae75..be02ad8 100644
--- a/platforms/documentation/docs/src/docs/userguide/authoring-builds/plugins/reporting_problems.adoc
+++ b/platforms/documentation/docs/src/docs/userguide/authoring-builds/plugins/reporting_problems.adoc
@@ -95,3 +95,26 @@
 
 In addition, there's a more convenient way to access the failure details.
 If clients configure the project connection with `LongRunningOperation.withFailureDetails()`, the Tooling API implicitly subscribes to the `ROOT` operation type and provides failure details via the `GradleConnectionException.getFailures()` method.
+
+[[sec:generated_html_report]]
+=== Generated HTML report
+
+The output of the problems generated by the Problems API is also provided as a rich HTML report generated at the end of the build.
+This report serves as a central location for users to review problems that occurred during a build.
+
+Plugin authors can use the Problems API to log events specific to their plugins, adding to the Gradle-generated ones.
+
+The report is not generated if no issues have been reported.
+Also, if you do not want to generate this report, you can disable it with the `--no-problems-report` flag.
+The console output provides a link to this report, as shown below:
+
+[source,text]
+----
+[Incubating] Problem report is available at: <project-dir>/build/reports/problems/problems-report.html
+
+BUILD SUCCESSFUL in 1s
+----
+
+The rendered report link directs you to a detailed HTML view of the problems:
+
+image::problems-report-html.png[]
diff --git a/platforms/documentation/docs/src/docs/userguide/img/problems-report-html.png b/platforms/documentation/docs/src/docs/userguide/img/problems-report-html.png
new file mode 100644
index 0000000..68fecb6
--- /dev/null
+++ b/platforms/documentation/docs/src/docs/userguide/img/problems-report-html.png
Binary files differ
diff --git a/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/KotlinApplicationInitIntegrationTest.groovy b/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/KotlinApplicationInitIntegrationTest.groovy
index a0f6dc6..1ddb8dc 100644
--- a/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/KotlinApplicationInitIntegrationTest.groovy
+++ b/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/KotlinApplicationInitIntegrationTest.groovy
@@ -18,11 +18,13 @@
 
 import org.gradle.api.JavaVersion
 import org.gradle.buildinit.plugins.fixtures.ScriptDslFixture
+import org.gradle.integtests.fixtures.ToBeFixedForIsolatedProjects
 import org.gradle.test.fixtures.file.LeaksFileHandles
 import org.gradle.test.precondition.Requires
 import org.gradle.test.preconditions.UnitTestPreconditions
 
 import static org.gradle.buildinit.plugins.internal.modifiers.BuildInitDsl.KOTLIN
+import static org.gradle.integtests.fixtures.ToBeFixedForIsolatedProjects.Skip.FLAKY
 import static org.hamcrest.CoreMatchers.containsString
 
 @LeaksFileHandles
@@ -194,6 +196,7 @@
     }
 
     @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+    @ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
     def "initializes Kotlin application with JUnit Jupiter test framework with --split-project"() {
         when:
         run('init', '--type', 'kotlin-application', '--test-framework', 'junit-jupiter', "--split-project", '--java-version', JavaVersion.current().majorVersion)
diff --git a/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/MultiProjectJvmApplicationInitIntegrationTest.groovy b/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/MultiProjectJvmApplicationInitIntegrationTest.groovy
index 65b3234..5cdcd87 100644
--- a/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/MultiProjectJvmApplicationInitIntegrationTest.groovy
+++ b/platforms/software/build-init/src/integTest/groovy/org/gradle/buildinit/plugins/MultiProjectJvmApplicationInitIntegrationTest.groovy
@@ -21,6 +21,7 @@
 import org.gradle.buildinit.plugins.internal.modifiers.BuildInitDsl
 import org.gradle.buildinit.plugins.internal.modifiers.Language
 import org.gradle.integtests.fixtures.DefaultTestExecutionResult
+import org.gradle.integtests.fixtures.ToBeFixedForIsolatedProjects
 import org.gradle.test.fixtures.file.TestFile
 import org.gradle.test.precondition.Requires
 import org.gradle.test.preconditions.UnitTestPreconditions
@@ -29,6 +30,7 @@
 import static org.gradle.buildinit.plugins.internal.modifiers.Language.JAVA
 import static org.gradle.buildinit.plugins.internal.modifiers.Language.KOTLIN
 import static org.gradle.buildinit.plugins.internal.modifiers.Language.SCALA
+import static org.gradle.integtests.fixtures.ToBeFixedForIsolatedProjects.Skip.FLAKY
 import static org.gradle.util.Matchers.containsLine
 import static org.gradle.util.Matchers.containsText
 import static org.hamcrest.core.AllOf.allOf
@@ -303,6 +305,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class GroovyDslMultiProjectKotlinApplicationInitIntegrationTest1 extends AbstractMultiProjectJvmApplicationInitIntegrationTest1 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.GROOVY, KOTLIN)
@@ -310,6 +313,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class GroovyDslMultiProjectKotlinApplicationInitIntegrationTest2 extends AbstractMultiProjectJvmApplicationInitIntegrationTest2 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.GROOVY, KOTLIN)
@@ -317,6 +321,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class GroovyDslMultiProjectKotlinApplicationInitIntegrationTest3 extends AbstractMultiProjectJvmApplicationInitIntegrationTest3 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.GROOVY, KOTLIN)
@@ -381,6 +386,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class KotlinDslMultiProjectKotlinApplicationInitIntegrationTest1 extends AbstractMultiProjectJvmApplicationInitIntegrationTest1 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.KOTLIN, KOTLIN)
@@ -388,6 +394,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class KotlinDslMultiProjectKotlinApplicationInitIntegrationTest2 extends AbstractMultiProjectJvmApplicationInitIntegrationTest2 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.KOTLIN, KOTLIN)
@@ -395,6 +402,7 @@
 }
 
 @Requires(value = UnitTestPreconditions.KotlinSupportedJdk.class)
+@ToBeFixedForIsolatedProjects(skip = FLAKY, because = "KGP modifies service parameter properties concurrently")
 class KotlinDslMultiProjectKotlinApplicationInitIntegrationTest3 extends AbstractMultiProjectJvmApplicationInitIntegrationTest3 {
     def setup() {
         setupDslAndLanguage(BuildInitDsl.KOTLIN, KOTLIN)
diff --git a/subprojects/core/src/integTest/groovy/org/gradle/process/internal/CancellationIntegrationTest.groovy b/subprojects/core/src/integTest/groovy/org/gradle/process/internal/CancellationIntegrationTest.groovy
index 11f62ec..7dcfa35 100644
--- a/subprojects/core/src/integTest/groovy/org/gradle/process/internal/CancellationIntegrationTest.groovy
+++ b/subprojects/core/src/integTest/groovy/org/gradle/process/internal/CancellationIntegrationTest.groovy
@@ -45,7 +45,9 @@
             task projectExecTask {
                 dependsOn 'compileJava'
                 doLast {
-                    def result = exec { commandLine '${fileToPath(Jvm.current().javaExecutable)}', '-cp', '${fileToPath(file('build/classes/java/main'))}', 'Block' }
+                    def result = services.get(ExecOperations).exec {
+                        commandLine '${fileToPath(Jvm.current().javaExecutable)}', '-cp', '${fileToPath(file('build/classes/java/main'))}', 'Block'
+                    }
                     assert result.exitValue == 0
                 }
             }
diff --git a/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjects.groovy b/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjects.groovy
index 0a3cdb1..b70de38 100644
--- a/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjects.groovy
+++ b/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjects.groovy
@@ -26,7 +26,33 @@
 @Retention(RetentionPolicy.RUNTIME)
 @Target([ElementType.METHOD, ElementType.TYPE])
 @ExtensionAnnotation(ToBeFixedForIsolatedProjectsExtension.class)
-public @interface ToBeFixedForIsolatedProjects {
+@interface ToBeFixedForIsolatedProjects {
+    /**
+     * Set to some {@link Skip} to skip the annotated test.
+     */
+    Skip skip() default Skip.DO_NOT_SKIP;
 
     String because() default "";
+
+    /**
+     * Reason for skipping a test with isolated projects.
+     */
+    enum Skip {
+
+        /**
+         * Do not skip this test, this is the default.
+         */
+        DO_NOT_SKIP {
+            @Override String getReason() { throw new UnsupportedOperationException("Must not be skipped") }
+        },
+
+        /**
+         * Use this reason on tests that intermittently fail with isolated projects.
+         */
+        FLAKY {
+            @Override String getReason() { "flaky" }
+        };
+
+        abstract String getReason();
+    }
 }
diff --git a/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjectsExtension.groovy b/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjectsExtension.groovy
index 95d9a54..86f44d2 100644
--- a/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjectsExtension.groovy
+++ b/testing/internal-integ-testing/src/main/groovy/org/gradle/integtests/fixtures/ToBeFixedForIsolatedProjectsExtension.groovy
@@ -22,25 +22,32 @@
 import org.spockframework.runtime.model.SpecElementInfo
 import org.spockframework.runtime.model.SpecInfo
 
+import static org.gradle.integtests.fixtures.ToBeFixedForIsolatedProjects.Skip.DO_NOT_SKIP
+
 class ToBeFixedForIsolatedProjectsExtension implements IAnnotationDrivenExtension<ToBeFixedForIsolatedProjects> {
 
     private final ToBeFixedSpecInterceptor toBeFixedSpecInterceptor = new ToBeFixedSpecInterceptor("Isolated Projects")
 
     @Override
     void visitSpecAnnotation(ToBeFixedForIsolatedProjects annotation, SpecInfo spec) {
-        visitAnnotation(spec)
+        visitAnnotation(annotation, spec)
     }
 
     @Override
     void visitFeatureAnnotation(ToBeFixedForIsolatedProjects annotation, FeatureInfo feature) {
-        visitAnnotation(feature)
+        visitAnnotation(annotation, feature)
     }
 
-    private void visitAnnotation(SpecElementInfo specElementInfo) {
+    private void visitAnnotation(ToBeFixedForIsolatedProjects annotation, SpecElementInfo specElementInfo) {
         if (GradleContextualExecuter.isNotIsolatedProjects()) {
             return
         }
 
+        if (annotation.skip() != DO_NOT_SKIP) {
+            specElementInfo.skip(annotation.skip().reason)
+            return
+        }
+
         toBeFixedSpecInterceptor.intercept(specElementInfo, new String[0])
     }
 }