blob: 176aae769c51ed08c6480a3d34668c22b364d72e [file] [log] [blame]
// Copyright 2020 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.analysis;
import static com.google.common.collect.ImmutableList.toImmutableList;
import com.google.auto.value.AutoValue;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.analysis.platform.ConstraintValueInfo;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable;
import com.google.devtools.build.lib.packages.BuiltinProvider;
import com.google.devtools.build.lib.packages.Info;
import com.google.devtools.build.lib.starlarkbuildapi.platform.IncompatiblePlatformProviderApi;
import java.util.Comparator;
import javax.annotation.Nullable;
/**
* Provider instance for the {@code target_compatible_with} attribute.
*
* <p>The presence of this provider is used to indicate that a target is incompatible with the
* current platform. Any target that provides this will automatically be excluded from {@link
* SkyframeAnalysisResult}'s list of configured targets.
*
* <p>This provider is able to keep track of _why_ the corresponding target is considered
* incompatible. If the target is incompatible because the target platform didn't satisfy one of the
* constraints in target_compatible_with, then all the relevant constraints are accessible via
* {@code getConstraintsResponsibleForIncompatibility()}. On the other hand, if the corresponding
* target is incompatible because one of its dependencies is incompatible, then all the incompatible
* dependencies are available via {@code getTargetResponsibleForIncompatibility()}.
*/
@Immutable
@AutoValue
public abstract class IncompatiblePlatformProvider
implements Info, IncompatiblePlatformProviderApi {
/** Name used in Starlark for accessing this provider. */
public static final String STARLARK_NAME = "IncompatiblePlatformProvider";
/** Provider singleton constant. */
public static final BuiltinProvider<IncompatiblePlatformProvider> PROVIDER =
new BuiltinProvider<IncompatiblePlatformProvider>(
STARLARK_NAME, IncompatiblePlatformProvider.class) {};
@Override
public BuiltinProvider<IncompatiblePlatformProvider> getProvider() {
return PROVIDER;
}
public static IncompatiblePlatformProvider incompatibleDueToTargets(
@Nullable Label targetPlatform,
ImmutableList<ConfiguredTarget> targetsResponsibleForIncompatibility) {
Preconditions.checkNotNull(targetsResponsibleForIncompatibility);
Preconditions.checkArgument(!targetsResponsibleForIncompatibility.isEmpty());
return new AutoValue_IncompatiblePlatformProvider(
targetPlatform, targetsResponsibleForIncompatibility, null);
}
public static IncompatiblePlatformProvider incompatibleDueToConstraints(
@Nullable Label targetPlatform, ImmutableList<ConstraintValueInfo> constraints) {
Preconditions.checkNotNull(constraints);
Preconditions.checkArgument(!constraints.isEmpty());
// Deduplicate and sort the list of incompatible constraints. Doing it here means that everyone
// inspecting this provider doesn't have to deal with it.
constraints =
constraints.stream()
.sorted(Comparator.comparing(ConstraintValueInfo::label))
.distinct()
.collect(toImmutableList());
return new AutoValue_IncompatiblePlatformProvider(targetPlatform, null, constraints);
}
@Override
public boolean isImmutable() {
return true; // immutable and Starlark-hashable
}
/** Returns the target platform of the target that was incompatible. */
@Nullable
public abstract Label targetPlatform();
/**
* Returns the incompatible dependencies that caused this provider to be present.
*
* <p>This may be null. If it is null, then {@code getConstraintsResponsibleForIncompatibility()}
* is guaranteed to be non-null. It will have at least one element in it if it is not null.
*/
@Nullable
public abstract ImmutableList<ConfiguredTarget> targetsResponsibleForIncompatibility();
/**
* Returns the constraints that the target platform didn't satisfy.
*
* <p>This may be null. If it is null, then {@code getTargetsResponsibleForIncompatibility()} is
* guaranteed to be non-null. It will have at least one element in it if it is not null.
*
* <p>The list is sorted based on the stringified label of each constraint.
*/
@Nullable
public abstract ImmutableList<ConstraintValueInfo> constraintsResponsibleForIncompatibility();
}