// Copyright 2015 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.cpp; import com.google.common.collect.ImmutableList; import com.google.devtools.build.lib.actions.Artifact; import com.google.devtools.build.lib.analysis.skylark.SkylarkApiProvider; import com.google.devtools.build.lib.collect.nestedset.NestedSet; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.skylarkinterface.SkylarkCallable; import com.google.devtools.build.lib.skylarkinterface.SkylarkModule; import com.google.devtools.build.lib.skylarkinterface.SkylarkModuleCategory; import com.google.devtools.build.lib.vfs.PathFragment; /** * A class that exposes the C++ providers to Skylark. It is intended to provide a simple and stable * interface for Skylark users. */ @SkylarkModule( name = "CcSkylarkApiProvider", category = SkylarkModuleCategory.PROVIDER, doc = "Provides access to information about C++ rules. " + "Every C++-related target provides this struct, accessible as a cc field " + "on target." ) public final class CcSkylarkApiProvider extends SkylarkApiProvider { /** The name of the field in Skylark used to access this class. */ public static final String NAME = "cc"; @SkylarkCallable( name = "transitive_headers", structField = true, doc = "Returns a depset of headers that have been declared in the " + " src or headers attribute" + "(possibly empty but never None).") public NestedSet getTransitiveHeaders() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); return ccContext.getDeclaredIncludeSrcs(); } @SkylarkCallable( name = "libs", structField = true, doc = "Returns the depset of libraries for either " + "FULLY STATIC mode (linkopts=[\"-static\"]) or " + "MOSTLY STATIC mode (linkstatic=1) " + "(possibly empty but never None)") public NestedSet getLibraries() { NestedSetBuilder libs = NestedSetBuilder.linkOrder(); CcLinkParamsInfo ccLinkParams = getInfo().get(CcLinkParamsInfo.PROVIDER); if (ccLinkParams == null) { return libs.build(); } for (LinkerInput lib : ccLinkParams.getCcLinkParams(true, false).getLibraries()) { libs.add(lib.getArtifact()); } return libs.build(); } @SkylarkCallable( name = "link_flags", structField = true, doc = "Returns the list of flags given to the C++ linker command for either " + "FULLY STATIC mode (linkopts=[\"-static\"]) or " + "MOSTLY STATIC mode (linkstatic=1) " + "(possibly empty but never None)") public ImmutableList getLinkopts() { CcLinkParamsInfo ccLinkParams = getInfo().get(CcLinkParamsInfo.PROVIDER); if (ccLinkParams == null) { return ImmutableList.of(); } return ccLinkParams.getCcLinkParams(true, false).flattenedLinkopts(); } @SkylarkCallable( name = "defines", structField = true, doc = "Returns the list of defines used to compile this target " + "(possibly empty but never None).") public ImmutableList getDefines() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); return ccContext == null ? ImmutableList.of() : ccContext.getDefines(); } @SkylarkCallable( name = "system_include_directories", structField = true, doc = "Returns the list of system include directories used to compile this target " + "(possibly empty but never None).") public ImmutableList getSystemIncludeDirs() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); if (ccContext == null) { return ImmutableList.of(); } ImmutableList.Builder builder = ImmutableList.builder(); for (PathFragment path : ccContext.getSystemIncludeDirs()) { builder.add(path.getSafePathString()); } return builder.build(); } @SkylarkCallable( name = "include_directories", structField = true, doc = "Returns the list of include directories used to compile this target " + "(possibly empty but never None).") public ImmutableList getIncludeDirs() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); if (ccContext == null) { return ImmutableList.of(); } ImmutableList.Builder builder = ImmutableList.builder(); for (PathFragment path : ccContext.getIncludeDirs()) { builder.add(path.getSafePathString()); } return builder.build(); } @SkylarkCallable( name = "quote_include_directories", structField = true, doc = "Returns the list of quote include directories used to compile this target " + "(possibly empty but never None).") public ImmutableList getQuoteIncludeDirs() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); if (ccContext == null) { return ImmutableList.of(); } ImmutableList.Builder builder = ImmutableList.builder(); for (PathFragment path : ccContext.getQuoteIncludeDirs()) { builder.add(path.getSafePathString()); } return builder.build(); } @SkylarkCallable( name = "compile_flags", structField = true, doc = "Returns the list of flags used to compile this target " + "(possibly empty but never None).") public ImmutableList getCcFlags() { CppCompilationContext ccContext = getInfo().getProvider(CppCompilationContext.class); ImmutableList.Builder options = ImmutableList.builder(); for (String define : ccContext.getDefines()) { options.add("-D" + define); } for (PathFragment path : ccContext.getSystemIncludeDirs()) { options.add("-isystem " + path.getSafePathString()); } for (PathFragment path : ccContext.getIncludeDirs()) { options.add("-I " + path.getSafePathString()); } for (PathFragment path : ccContext.getQuoteIncludeDirs()) { options.add("-iquote " + path.getSafePathString()); } return options.build(); } }