| // Copyright 2018 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.includescanning; |
| |
| import static com.google.common.collect.ImmutableList.toImmutableList; |
| |
| import com.google.common.base.Preconditions; |
| import com.google.common.collect.ImmutableList; |
| import com.google.common.collect.Sets; |
| import com.google.devtools.build.lib.actions.ActionExecutionContext; |
| import com.google.devtools.build.lib.actions.Artifact; |
| import com.google.devtools.build.lib.actions.Artifact.TreeFileArtifact; |
| import com.google.devtools.build.lib.actions.EnvironmentalExecException; |
| import com.google.devtools.build.lib.actions.ExecException; |
| import com.google.devtools.build.lib.actions.UserExecException; |
| import com.google.devtools.build.lib.packages.NoSuchPackageException; |
| import com.google.devtools.build.lib.profiler.Profiler; |
| import com.google.devtools.build.lib.profiler.ProfilerTask; |
| import com.google.devtools.build.lib.profiler.SilentCloseable; |
| import com.google.devtools.build.lib.rules.cpp.CppCompileAction; |
| import com.google.devtools.build.lib.rules.cpp.CppIncludeScanningContext; |
| import com.google.devtools.build.lib.rules.cpp.IncludeScanner; |
| import com.google.devtools.build.lib.rules.cpp.IncludeScanner.IncludeScanningHeaderData; |
| import com.google.devtools.build.lib.server.FailureDetails; |
| import com.google.devtools.build.lib.server.FailureDetails.FailureDetail; |
| import com.google.devtools.build.lib.server.FailureDetails.IncludeScanning.Code; |
| import com.google.devtools.build.lib.skyframe.TreeArtifactValue; |
| import com.google.devtools.build.lib.vfs.FileSystemUtils; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| import com.google.devtools.build.skyframe.SkyFunction.Environment; |
| import com.google.devtools.build.skyframe.SkyframeLookupResult; |
| import java.io.IOException; |
| import java.util.ArrayList; |
| import java.util.LinkedHashSet; |
| import java.util.List; |
| import java.util.Set; |
| import java.util.function.Supplier; |
| import javax.annotation.Nullable; |
| |
| /** Include scanning context implementation. */ |
| public final class CppIncludeScanningContextImpl implements CppIncludeScanningContext { |
| private final Supplier<IncludeScannerSupplier> includeScannerSupplier; |
| |
| public CppIncludeScanningContextImpl(Supplier<IncludeScannerSupplier> includeScannerSupplier) { |
| this.includeScannerSupplier = includeScannerSupplier; |
| } |
| |
| @Override |
| @Nullable |
| public List<Artifact> findAdditionalInputs( |
| CppCompileAction action, |
| ActionExecutionContext actionExecutionContext, |
| IncludeScanningHeaderData includeScanningHeaderData) |
| throws ExecException, InterruptedException { |
| Preconditions.checkNotNull(includeScannerSupplier, action); |
| |
| Set<Artifact> includes = Sets.newConcurrentHashSet(); |
| includes.addAll(action.getBuiltInIncludeFiles()); |
| |
| // Deduplicate include directories. This can occur especially with "built-in" and "system" |
| // include directories because of the way we retrieve them. Duplicate include directories |
| // really mess up #include_next directives. |
| Set<PathFragment> includeDirs = new LinkedHashSet<>(action.getIncludeDirs()); |
| List<PathFragment> quoteIncludeDirs = action.getQuoteIncludeDirs(); |
| List<PathFragment> frameworkIncludeDirs = action.getFrameworkIncludeDirs(); |
| List<String> cmdlineIncludes = includeScanningHeaderData.getCmdlineIncludes(); |
| |
| includeDirs.addAll(includeScanningHeaderData.getSystemIncludeDirs()); |
| |
| // Add the system include paths to the list of include paths. |
| List<PathFragment> absoluteBuiltInIncludeDirs = new ArrayList<>(); |
| for (PathFragment pathFragment : action.getBuiltInIncludeDirectories()) { |
| if (pathFragment.isAbsolute()) { |
| absoluteBuiltInIncludeDirs.add(pathFragment); |
| } |
| includeDirs.add(pathFragment); |
| } |
| |
| ImmutableList<PathFragment> includeDirList = ImmutableList.copyOf(includeDirs); |
| IncludeScanner scanner = |
| includeScannerSupplier |
| .get() |
| .scannerFor(quoteIncludeDirs, includeDirList, frameworkIncludeDirs); |
| |
| Artifact mainSource = action.getMainIncludeScannerSource(); |
| ImmutableList<Artifact> sources = |
| expandTreeArtifacts( |
| action.getIncludeScannerSources(), |
| actionExecutionContext.getEnvironmentForDiscoveringInputs()); |
| if (sources == null) { |
| return null; |
| } |
| |
| try (SilentCloseable c = |
| Profiler.instance() |
| .profile(ProfilerTask.SCANNER, action.getSourceFile().getExecPathString())) { |
| scanner.processAsync( |
| mainSource, |
| sources, |
| includeScanningHeaderData, |
| cmdlineIncludes, |
| includes, |
| action, |
| actionExecutionContext, |
| action.getGrepIncludes()); |
| if (actionExecutionContext.getEnvironmentForDiscoveringInputs().valuesMissing()) { |
| return null; |
| } |
| return collect(actionExecutionContext, includes, absoluteBuiltInIncludeDirs); |
| } catch (IOException e) { |
| throw new EnvironmentalExecException( |
| e, createFailureDetail("Include scanning IOException", Code.SCANNING_IO_EXCEPTION)); |
| } catch (NoSuchPackageException e) { |
| throw new EnvironmentalExecException( |
| e, |
| createFailureDetail( |
| "Error for BUILD file during include scanning: " + e.getMessage(), |
| Code.PACKAGE_LOAD_FAILURE)); |
| } |
| } |
| |
| /** |
| * Returns a list of artifacts with all tree artifacts replaced by their expansion or null if we |
| * are missing a Skyframe dependency. |
| * |
| * <p>We take an ad-hoc approach, which consults Skyframe to retrieve the tree expansions. This is |
| * necessary because include scanning may include tree artifacts which are not inputs of the |
| * original action. |
| * |
| * <p>Normally, we expand tree artifacts using a tree artifact expander from the {@link |
| * ActionExecutionContext}, however the expander available before include scanning only captures |
| * tree artifacts from the action inputs, which is insufficient. In fact, the action execution |
| * context for include scanning has a null expander in it. |
| */ |
| @Nullable |
| private static ImmutableList<Artifact> expandTreeArtifacts( |
| ImmutableList<Artifact> artifacts, Environment env) throws InterruptedException { |
| ImmutableList<Artifact> trees = |
| artifacts.stream().filter(Artifact::isTreeArtifact).collect(toImmutableList()); |
| if (trees.isEmpty()) { |
| return artifacts; |
| } |
| |
| SkyframeLookupResult expansions = env.getValuesAndExceptions(trees); |
| ImmutableList.Builder<Artifact> expanded = ImmutableList.builder(); |
| for (var artifact : artifacts) { |
| if (!artifact.isTreeArtifact()) { |
| expanded.add(artifact); |
| continue; |
| } |
| |
| TreeArtifactValue treeArtifactValue = (TreeArtifactValue) expansions.get(artifact); |
| if (treeArtifactValue == null) { |
| return null; |
| } |
| expanded.addAll(treeArtifactValue.getChildren()); |
| } |
| return expanded.build(); |
| } |
| |
| private static List<Artifact> collect( |
| ActionExecutionContext actionExecutionContext, |
| Set<Artifact> includes, |
| List<PathFragment> absoluteBuiltInIncludeDirs) |
| throws ExecException { |
| // Collect inputs and output |
| List<Artifact> inputs = new ArrayList<>(includes.size()); |
| for (Artifact included : includes) { |
| // Check for absolute includes -- we assign the file system root as |
| // the root path for such includes |
| if (included.getRoot().getRoot().isAbsolute()) { |
| if (FileSystemUtils.startsWithAny( |
| actionExecutionContext.getInputPath(included).asFragment(), |
| absoluteBuiltInIncludeDirs)) { |
| // Skip include files found in absolute include directories. |
| continue; |
| } |
| throw new UserExecException( |
| createFailureDetail( |
| "illegal absolute path to include file: " |
| + actionExecutionContext.getInputPath(included), |
| Code.ILLEGAL_ABSOLUTE_PATH)); |
| } |
| if (included.hasParent() && included.getParent().isTreeArtifact()) { |
| // Note that this means every file in the TreeArtifact becomes an input to the action, and |
| // we have spurious rebuilds if non-included files change. |
| Preconditions.checkArgument( |
| included instanceof TreeFileArtifact, "Not a TreeFileArtifact: %s", included); |
| inputs.add(included.getParent()); |
| } else { |
| inputs.add(included); |
| } |
| } |
| return inputs; |
| } |
| |
| private static FailureDetail createFailureDetail(String message, Code detailedCode) { |
| return FailureDetail.newBuilder() |
| .setMessage(message) |
| .setIncludeScanning(FailureDetails.IncludeScanning.newBuilder().setCode(detailedCode)) |
| .build(); |
| } |
| } |