// Copyright 2015 Google Inc. 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 static com.google.devtools.build.lib.packages.ImplicitOutputsFunction.fromTemplates; import com.google.common.annotations.VisibleForTesting; import com.google.common.collect.ImmutableList; 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.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.packages.ImplicitOutputsFunction.SafeImplicitOutputsFunction; import com.google.devtools.build.lib.rules.objc.ReleaseBundlingSupport.SplitArchTransition.ConfigurationDistinguisher; import com.google.devtools.build.lib.rules.objc.XcodeProvider.Builder; import com.google.devtools.build.xcode.xcodegen.proto.XcodeGenProtos.XcodeprojBuildSetting; /** * Support for Objc rule types that export an Xcode provider or generate xcode project files. * *

Methods on this class can be called in any order without impacting the result. */ public final class XcodeSupport { /** * Template for a target's xcode project. */ public static final SafeImplicitOutputsFunction PBXPROJ = fromTemplates("%{name}.xcodeproj/project.pbxproj"); private final RuleContext ruleContext; /** * Creates a new xcode support for the given context. */ XcodeSupport(RuleContext ruleContext) { this.ruleContext = ruleContext; } /** * Adds xcode project files to the given builder. * * @return this xcode support */ XcodeSupport addFilesToBuild(NestedSetBuilder filesToBuild) { filesToBuild.add(ruleContext.getImplicitOutputArtifact(PBXPROJ)); return this; } /** * Adds a dummy source file to the Xcode target. This is needed if the target does not have any * source files but Xcode requires one. * * @return this xcode support */ XcodeSupport addDummySource(XcodeProvider.Builder xcodeProviderBuilder) { xcodeProviderBuilder.addAdditionalSources( ruleContext.getPrerequisiteArtifact("$dummy_source", Mode.TARGET)); return this; } /** * Registers actions that generate the rule's Xcode project. * * @param xcodeProvider information about this rule's xcode settings and that of its dependencies * @return this xcode support */ XcodeSupport registerActions(XcodeProvider xcodeProvider) { ObjcActionsBuilder actionsBuilder = ObjcRuleClasses.actionsBuilder(ruleContext); actionsBuilder.registerXcodegenActions( new ObjcRuleClasses.Tools(ruleContext), ruleContext.getImplicitOutputArtifact(XcodeSupport.PBXPROJ), XcodeProvider.Project.fromTopLevelTarget(xcodeProvider)); return this; } /** * Adds common xcode settings to the given provider builder. * * @param objcProvider provider containing all dependencies' information as well as some of this * rule's * @param productType type of this rule's Xcode target * * @return this xcode support */ XcodeSupport addXcodeSettings(XcodeProvider.Builder xcodeProviderBuilder, ObjcProvider objcProvider, XcodeProductType productType) { ObjcConfiguration objcConfiguration = ObjcRuleClasses.objcConfiguration(ruleContext); return addXcodeSettings(xcodeProviderBuilder, objcProvider, productType, objcConfiguration.getIosCpu(), objcConfiguration.getConfigurationDistinguisher()); } /** * Adds common xcode settings to the given provider builder, explicitly specifying architecture * to use. * * @param objcProvider provider containing all dependencies' information as well as some of this * rule's * @param productType type of this rule's Xcode target * @param architecture architecture to filter all dependencies with (only matching ones will be * included in the final targets generated) * @param configurationDistinguisher distinguisher that will cause this target's xcode provider to * discard any dependencies from sources that are tagged with a different distinguisher * @return this xcode support */ XcodeSupport addXcodeSettings(Builder xcodeProviderBuilder, ObjcProvider objcProvider, XcodeProductType productType, String architecture, ConfigurationDistinguisher configurationDistinguisher) { xcodeProviderBuilder .setLabel(ruleContext.getLabel()) .setArchitecture(architecture) .setConfigurationDistinguisher(configurationDistinguisher) .setObjcProvider(objcProvider) .setProductType(productType) .addXcodeprojBuildSettings(XcodeSupport.defaultXcodeSettings()); return this; } /** * Adds dependencies to the given provider builder from the given attribute. * * @return this xcode support */ XcodeSupport addDependencies(Builder xcodeProviderBuilder, Attribute attribute) { xcodeProviderBuilder.addPropagatedDependencies( ruleContext.getPrerequisites( attribute.getName(), attribute.getAccessMode(), XcodeProvider.class)); return this; } /** * Adds non-propagated dependencies to the given provider builder from the given attribute. * *

A non-propagated dependency will not be linked into the final app bundle and can only serve * as a compile-only dependency for its direct dependent. * * @return this xcode support */ XcodeSupport addNonPropagatedDependencies(Builder xcodeProviderBuilder, Attribute attribute) { xcodeProviderBuilder.addNonPropagatedDependencies( ruleContext.getPrerequisites( attribute.getName(), attribute.getAccessMode(), XcodeProvider.class)); return this; } /** * Generates an extra {@link XcodeProductType#LIBRARY_STATIC} Xcode target with the same * compilation artifacts as the main Xcode target associated with this Xcode support. The extra * Xcode library target, instead of the main Xcode target, will act as a dependency for all * dependent Xcode targets. * *

This is needed to build the Xcode binary target generated by ios_application in XCode. * Currently there is an Xcode target dependency between the binary target from ios_application * and the binary target from objc_binary. But Xcode does not link in compiled artifacts from * binary dependencies, so any sources specified on objc_binary rules will not be compiled and * linked into the app bundle in dependent binary targets associated with ios_application in * XCode. */ // TODO(bazel-team): Remove this when the binary rule types and bundling rule types are merged. XcodeSupport generateCompanionLibXcodeTarget(Builder xcodeProviderBuilder) { xcodeProviderBuilder.generateCompanionLibTarget(); return this; } /** * Returns a list of default XCode build settings for Bazel-generated XCode projects. */ @VisibleForTesting static Iterable defaultXcodeSettings() { // Do not use XCode headermap because Bazel-generated header search paths are sufficient for // resolving header imports. return ImmutableList.of( XcodeprojBuildSetting.newBuilder() .setName("USE_HEADERMAP") .setValue("NO") .build()); } }