// Copyright 2016 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.rules.objc; import com.google.auto.value.AutoValue; import com.google.common.base.Optional; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableListMultimap; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterables; import com.google.devtools.build.lib.actions.Artifact; import com.google.devtools.build.lib.analysis.RuleConfiguredTarget.Mode; import com.google.devtools.build.lib.analysis.RuleContext; import com.google.devtools.build.lib.analysis.TransitiveInfoCollection; import com.google.devtools.build.lib.analysis.config.BuildConfiguration; import com.google.devtools.build.lib.collect.nestedset.NestedSet; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.packages.RuleClass.ConfiguredTargetFactory.RuleErrorException; import com.google.devtools.build.lib.rules.apple.Platform; import com.google.devtools.build.lib.rules.cpp.CcLinkParamsProvider; import com.google.devtools.build.lib.rules.cpp.CcToolchainProvider; import com.google.devtools.build.lib.rules.objc.CompilationSupport.ExtraLinkArgs; import com.google.devtools.build.lib.rules.objc.ObjcCommon.ResourceAttributes; import java.util.List; import java.util.Map; import java.util.Set; /** * Support utility for creating multi-arch Apple binaries. */ public class MultiArchBinarySupport { private final RuleContext ruleContext; /** * Configuration, toolchain, and provider for for single-arch dependency configurations of a * multi-arch target. */ @AutoValue abstract static class DependencySpecificConfiguration { static DependencySpecificConfiguration create( BuildConfiguration config, CcToolchainProvider toolchain, ObjcProvider objcProvider) { return new AutoValue_MultiArchBinarySupport_DependencySpecificConfiguration( config, toolchain, objcProvider); } abstract BuildConfiguration config(); abstract CcToolchainProvider toolchain(); abstract ObjcProvider objcProvider(); } /** * @param ruleContext the current rule context */ public MultiArchBinarySupport(RuleContext ruleContext) { this.ruleContext = ruleContext; } /** * Registers actions to create a multi-arch Apple binary. * * @param platform the platform for which the binary is targeted * @param extraLinkArgs the extra linker args to add to link actions linking single-architecture * binaries together * @param dependencySpecificConfigurations a set of {@link DependencySpecificConfiguration} that * corresponds to child configurations for this target. Can be obtained via {@link * #getDependencySpecificConfigurations} * @param extraLinkInputs the extra linker inputs to be made available during link actions * @param configToDepsCollectionMap a multimap from dependency configuration to the list of * provider collections which are propagated from the dependencies of that configuration * @param outputLipoBinary the artifact (lipo'ed binary) which should be output as a result of * this support * @throws RuleErrorException if there are attribute errors in the current rule context */ public void registerActions( Platform platform, ExtraLinkArgs extraLinkArgs, Set dependencySpecificConfigurations, Iterable extraLinkInputs, ImmutableListMultimap configToDepsCollectionMap, Artifact outputLipoBinary) throws RuleErrorException, InterruptedException { NestedSetBuilder binariesToLipo = NestedSetBuilder.stableOrder(); for (DependencySpecificConfiguration dependencySpecificConfiguration : dependencySpecificConfigurations) { IntermediateArtifacts intermediateArtifacts = ObjcRuleClasses.intermediateArtifacts( ruleContext, dependencySpecificConfiguration.config()); ImmutableList.Builder j2ObjcMappingFileProviders = ImmutableList.builder(); J2ObjcEntryClassProvider.Builder j2ObjcEntryClassProviderBuilder = new J2ObjcEntryClassProvider.Builder(); for (TransitiveInfoCollection dep : configToDepsCollectionMap.get(dependencySpecificConfiguration.config())) { if (dep.getProvider(J2ObjcMappingFileProvider.class) != null) { j2ObjcMappingFileProviders.add(dep.getProvider(J2ObjcMappingFileProvider.class)); } if (dep.getProvider(J2ObjcEntryClassProvider.class) != null) { j2ObjcEntryClassProviderBuilder.addTransitive( dep.getProvider(J2ObjcEntryClassProvider.class)); } } J2ObjcMappingFileProvider j2ObjcMappingFileProvider = J2ObjcMappingFileProvider.union(j2ObjcMappingFileProviders.build()); J2ObjcEntryClassProvider j2ObjcEntryClassProvider = j2ObjcEntryClassProviderBuilder.build(); binariesToLipo.add(intermediateArtifacts.strippedSingleArchitectureBinary()); ObjcProvider objcProvider = dependencySpecificConfiguration.objcProvider(); CompilationArtifacts compilationArtifacts = CompilationSupport.compilationArtifacts( ruleContext, ObjcRuleClasses.intermediateArtifacts( ruleContext, dependencySpecificConfiguration.config())); CompilationSupport.createForConfig(ruleContext, dependencySpecificConfiguration.config()) .registerCompileAndArchiveActions( compilationArtifacts, objcProvider, dependencySpecificConfiguration.toolchain()) .registerLinkActions( objcProvider, j2ObjcMappingFileProvider, j2ObjcEntryClassProvider, extraLinkArgs, extraLinkInputs, DsymOutputType.APP, dependencySpecificConfiguration.toolchain()) .validateAttributes(); ruleContext.assertNoErrors(); } new LipoSupport(ruleContext) .registerCombineArchitecturesAction( binariesToLipo.build(), outputLipoBinary, platform); } /** * Returns a set of {@link DependencySpecificConfiguration} instances that comprise all * information about the dependencies for each child configuration. This can be used both to * register actions in {@link #registerActions} and collect provider information to be propagated * upstream. * * @param childConfigurationsAndToolchains the set of configurations and toolchains for which * dependencies of the current rule are built * @param configToDepsCollectionMap a map from child configuration to providers that "deps" of the * current rule have propagated in that configuration * @param configurationToNonPropagatedObjcMap a map from child configuration to providers that * "non_propagated_deps" of the current rule have propagated in that configuration * @param dylibObjcProviders {@link ObjcProvider}s that dynamic library dependencies of the * current rule have propagated * @param dylibProtoProviders {@link ObjcProtoProvider} providers that dynamic library * dependencies of the current rule have propagated * @param bundleLoaderObjcProvider Optional ObjcProvider containing artifacts and paths to be * included in this binary's compilation actions * @throws RuleErrorException if there are attribute errors in the current rule context */ public ImmutableSet getDependencySpecificConfigurations( Map childConfigurationsAndToolchains, ImmutableListMultimap configToDepsCollectionMap, ImmutableListMultimap configurationToNonPropagatedObjcMap, Iterable dylibObjcProviders, Iterable dylibProtoProviders) throws RuleErrorException, InterruptedException { ImmutableSet.Builder childInfoBuilder = ImmutableSet.builder(); for (BuildConfiguration childConfig : childConfigurationsAndToolchains.keySet()) { Optional protosObjcProvider; if (ObjcRuleClasses.objcConfiguration(ruleContext).enableAppleBinaryNativeProtos()) { ProtobufSupport protoSupport = new ProtobufSupport(ruleContext, childConfig, protoArtifactsToAvoid(dylibProtoProviders)) .registerGenerationActions() .registerCompilationActions(); protosObjcProvider = protoSupport.getObjcProvider(); } else { protosObjcProvider = Optional.absent(); } IntermediateArtifacts intermediateArtifacts = ObjcRuleClasses.intermediateArtifacts(ruleContext, childConfig); Iterable additionalDepProviders = Iterables.concat( dylibObjcProviders, ruleContext.getPrerequisites("bundles", Mode.TARGET, ObjcProvider.class), protosObjcProvider.asSet()); ObjcCommon common = common( ruleContext, childConfig, intermediateArtifacts, nullToEmptyList(configToDepsCollectionMap.get(childConfig)), nullToEmptyList(configurationToNonPropagatedObjcMap.get(childConfig)), additionalDepProviders); ObjcProvider objcProvider = common.getObjcProvider().subtractSubtrees(dylibObjcProviders, ImmutableList.of()); childInfoBuilder.add( DependencySpecificConfiguration.create( childConfig, childConfigurationsAndToolchains.get(childConfig), objcProvider)); } return childInfoBuilder.build(); } private ObjcCommon common( RuleContext ruleContext, BuildConfiguration buildConfiguration, IntermediateArtifacts intermediateArtifacts, List propagatedDeps, List nonPropagatedObjcDeps, Iterable additionalDepProviders) { CompilationArtifacts compilationArtifacts = CompilationSupport.compilationArtifacts(ruleContext, intermediateArtifacts); ObjcCommon.Builder commonBuilder = new ObjcCommon.Builder(ruleContext, buildConfiguration) .setCompilationAttributes( CompilationAttributes.Builder.fromRuleContext(ruleContext).build()) .setCompilationArtifacts(compilationArtifacts) .setResourceAttributes(new ResourceAttributes(ruleContext)) .addDefines(ruleContext.getTokenizedStringListAttr("defines")) .addDeps(propagatedDeps) .addDepObjcProviders(additionalDepProviders) .addNonPropagatedDepObjcProviders(nonPropagatedObjcDeps) .setIntermediateArtifacts(intermediateArtifacts) .setAlwayslink(false) // TODO(b/29152500): Enable module map generation. .setLinkedBinary(intermediateArtifacts.strippedSingleArchitectureBinary()); if (ObjcRuleClasses.objcConfiguration(ruleContext).generateDsym()) { commonBuilder.addDebugArtifacts(DsymOutputType.APP); } return commonBuilder.build(); } private List nullToEmptyList(List inputList) { return inputList != null ? inputList : ImmutableList.of(); } private static NestedSet protoArtifactsToAvoid( Iterable avoidedProviders) { NestedSetBuilder avoidArtifacts = NestedSetBuilder.stableOrder(); for (ObjcProtoProvider avoidProvider : avoidedProviders) { for (NestedSet avoidProviderOutputGroup : avoidProvider.getProtoGroups()) { avoidArtifacts.addTransitive(avoidProviderOutputGroup); } } return avoidArtifacts.build(); } }