|  | // Copyright 2014 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 com.google.common.collect.ImmutableSet; | 
|  | import com.google.devtools.build.lib.analysis.configuredtargets.RuleConfiguredTarget.Mode; | 
|  | import com.google.devtools.build.lib.cmdline.Label; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; | 
|  | import java.util.Set; | 
|  |  | 
|  | /** | 
|  | * Transitive info provider for rules that behave differently when used from | 
|  | * different languages. | 
|  | * | 
|  | * <p>Most rules generate code for a particular language or are totally language independent. | 
|  | * Some rules, however, behave differently when depended upon from different languages. | 
|  | * They might generate different libraries when used from different languages (and with | 
|  | * different API versions). This interface allows code sharing between implementations. | 
|  | * | 
|  | * <p>This provider is not really a roll-up of transitive information. | 
|  | */ | 
|  | @Immutable | 
|  | public final class LanguageDependentFragment implements TransitiveInfoProvider { | 
|  | /** | 
|  | * A language that can be supported by a multi-language configured target. | 
|  | * | 
|  | * <p>Note that no {@code hashCode}/{@code equals} methods are provided, because these | 
|  | * objects are expected to be compared for object identity, which is the default. | 
|  | */ | 
|  | public static final class LibraryLanguage { | 
|  | private final String displayName; | 
|  |  | 
|  | public LibraryLanguage(String displayName) { | 
|  | this.displayName = displayName; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | return displayName; | 
|  | } | 
|  | } | 
|  |  | 
|  | private final Label label; | 
|  | private final ImmutableSet<LibraryLanguage> languages; | 
|  |  | 
|  | public LanguageDependentFragment(Label label, Set<LibraryLanguage> languages) { | 
|  | this.label = label; | 
|  | this.languages = ImmutableSet.copyOf(languages); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the label that is associated with this piece of information. | 
|  | * | 
|  | * <p>This is usually the label of the target that provides the information. | 
|  | */ | 
|  | public Label getLabel() { | 
|  | return label; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a set of the languages the ConfiguredTarget generates output for. | 
|  | * For use only by rules that directly depend on this library via a "deps" attribute. | 
|  | */ | 
|  | public ImmutableSet<LibraryLanguage> getSupportedLanguages() { | 
|  | return languages; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Routines for verifying that dependency provide the right output. | 
|  | */ | 
|  | public static final class Checker { | 
|  | /** Checks that given dep supports the given language. */ | 
|  | public static boolean depSupportsLanguage( | 
|  | RuleContext context, | 
|  | LanguageDependentFragment dep, | 
|  | LibraryLanguage language, | 
|  | String attrName) { | 
|  | if (dep.getSupportedLanguages().contains(language)) { | 
|  | return true; | 
|  | } else { | 
|  | context.attributeError( | 
|  | attrName, | 
|  | String.format("'%s' does not produce output for %s", dep.getLabel(), language)); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Checks that all LanguageDependentFragment support the given language. | 
|  | * | 
|  | * @param attrNames names of attributes whose contents should be checked for supported | 
|  | *     languages, e.g., "deps". | 
|  | */ | 
|  | public static void depsSupportsLanguage( | 
|  | RuleContext context, LibraryLanguage language, Iterable<String> attrNames) { | 
|  | for (String attrName : attrNames) { | 
|  | for (LanguageDependentFragment dep : | 
|  | context.getPrerequisites(attrName, Mode.TARGET, LanguageDependentFragment.class)) { | 
|  | depSupportsLanguage(context, dep, language, attrName); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } |