Damien Martin-Guillerez | f88f4d8 | 2015-09-25 13:56:55 +0000 | [diff] [blame] | 1 | // Copyright 2014 The Bazel Authors. All rights reserved. |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 2 | // |
| 3 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | // you may not use this file except in compliance with the License. |
| 5 | // You may obtain a copy of the License at |
| 6 | // |
| 7 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | // |
| 9 | // Unless required by applicable law or agreed to in writing, software |
| 10 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | // See the License for the specific language governing permissions and |
| 13 | // limitations under the License. |
| 14 | package com.google.devtools.build.lib.actions; |
| 15 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 16 | import com.google.common.base.Predicate; |
Janak Ramakrishnan | 90f3d34 | 2015-03-27 19:45:18 +0000 | [diff] [blame] | 17 | import com.google.common.collect.ImmutableList; |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 18 | import com.google.common.collect.ImmutableMap; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 19 | import com.google.common.collect.Iterables; |
Rumou Duan | 33bab46 | 2016-04-25 17:55:12 +0000 | [diff] [blame] | 20 | import com.google.devtools.build.lib.actions.ActionAnalysisMetadata.MiddlemanType; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 21 | import com.google.devtools.build.lib.actions.cache.ActionCache; |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 22 | import com.google.devtools.build.lib.actions.cache.ActionCache.Entry; |
Shreya Bhattarai | 141b6c2 | 2016-08-22 22:00:24 +0000 | [diff] [blame] | 23 | import com.google.devtools.build.lib.actions.cache.DigestUtils; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 24 | import com.google.devtools.build.lib.actions.cache.Metadata; |
| 25 | import com.google.devtools.build.lib.actions.cache.MetadataHandler; |
| 26 | import com.google.devtools.build.lib.events.Event; |
| 27 | import com.google.devtools.build.lib.events.EventHandler; |
| 28 | import com.google.devtools.build.lib.events.EventKind; |
Mark Schaller | 6df8179 | 2015-12-10 18:47:47 +0000 | [diff] [blame] | 29 | import com.google.devtools.build.lib.util.Preconditions; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 30 | import com.google.devtools.build.lib.vfs.PathFragment; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 31 | import java.io.IOException; |
| 32 | import java.util.ArrayList; |
| 33 | import java.util.HashMap; |
| 34 | import java.util.List; |
| 35 | import java.util.Map; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 36 | import javax.annotation.Nullable; |
| 37 | |
| 38 | /** |
| 39 | * Checks whether an {@link Action} needs to be executed, or whether it has not changed since it was |
| 40 | * last stored in the action cache. Must be informed of the new Action data after execution as well. |
| 41 | * |
| 42 | * <p>The fingerprint, input files names, and metadata (either mtimes or MD5sums) of each action are |
| 43 | * cached in the action cache to avoid unnecessary rebuilds. Middleman artifacts are handled |
| 44 | * specially, avoiding the need to create actual files corresponding to the middleman artifacts. |
| 45 | * Instead of that, results of MiddlemanAction dependency checks are cached internally and then |
| 46 | * reused whenever an input middleman artifact is encountered. |
| 47 | * |
| 48 | * <p>While instances of this class hold references to action and metadata cache instances, they are |
| 49 | * otherwise lightweight, and should be constructed anew and discarded for each build request. |
| 50 | */ |
| 51 | public class ActionCacheChecker { |
| 52 | private final ActionCache actionCache; |
| 53 | private final Predicate<? super Action> executionFilter; |
| 54 | private final ArtifactResolver artifactResolver; |
| 55 | // True iff --verbose_explanations flag is set. |
| 56 | private final boolean verboseExplanations; |
| 57 | |
| 58 | public ActionCacheChecker(ActionCache actionCache, ArtifactResolver artifactResolver, |
| 59 | Predicate<? super Action> executionFilter, boolean verboseExplanations) { |
| 60 | this.actionCache = actionCache; |
| 61 | this.executionFilter = executionFilter; |
| 62 | this.artifactResolver = artifactResolver; |
| 63 | this.verboseExplanations = verboseExplanations; |
| 64 | } |
| 65 | |
| 66 | public boolean isActionExecutionProhibited(Action action) { |
| 67 | return !executionFilter.apply(action); |
| 68 | } |
| 69 | |
| 70 | /** |
| 71 | * Checks whether one of existing output paths is already used as a key. |
| 72 | * If yes, returns it - otherwise uses first output file as a key |
| 73 | */ |
| 74 | private ActionCache.Entry getCacheEntry(Action action) { |
| 75 | for (Artifact output : action.getOutputs()) { |
| 76 | ActionCache.Entry entry = actionCache.get(output.getExecPathString()); |
| 77 | if (entry != null) { |
| 78 | return entry; |
| 79 | } |
| 80 | } |
| 81 | return null; |
| 82 | } |
| 83 | |
Janak Ramakrishnan | e3f04b8 | 2015-03-27 16:45:34 +0000 | [diff] [blame] | 84 | private void removeCacheEntry(Action action) { |
| 85 | for (Artifact output : action.getOutputs()) { |
| 86 | actionCache.remove(output.getExecPathString()); |
| 87 | } |
| 88 | } |
| 89 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 90 | /** |
| 91 | * Validate metadata state for action input or output artifacts. |
| 92 | * |
| 93 | * @param entry cached action information. |
| 94 | * @param action action to be validated. |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 95 | * @param actionInputs the inputs of the action. Normally just the result of action.getInputs(), |
| 96 | * but if this action doesn't yet know its inputs, we check the inputs from the cache. |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 97 | * @param metadataHandler provider of metadata for the artifacts this action interacts with. |
| 98 | * @param checkOutput true to validate output artifacts, Otherwise, just |
| 99 | * validate inputs. |
| 100 | * |
| 101 | * @return true if at least one artifact has changed, false - otherwise. |
| 102 | */ |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 103 | private boolean validateArtifacts(Entry entry, Action action, |
| 104 | Iterable<Artifact> actionInputs, MetadataHandler metadataHandler, boolean checkOutput) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 105 | Iterable<Artifact> artifacts = checkOutput |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 106 | ? Iterables.concat(action.getOutputs(), actionInputs) |
| 107 | : actionInputs; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 108 | Map<String, Metadata> mdMap = new HashMap<>(); |
| 109 | for (Artifact artifact : artifacts) { |
| 110 | mdMap.put(artifact.getExecPathString(), metadataHandler.getMetadataMaybe(artifact)); |
| 111 | } |
Shreya Bhattarai | 141b6c2 | 2016-08-22 22:00:24 +0000 | [diff] [blame] | 112 | return !DigestUtils.fromMetadata(mdMap).equals(entry.getFileDigest()); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 113 | } |
| 114 | |
| 115 | private void reportCommand(EventHandler handler, Action action) { |
| 116 | if (handler != null) { |
| 117 | if (verboseExplanations) { |
| 118 | String keyDescription = action.describeKey(); |
Laszlo Csomor | f04efcc | 2015-02-12 17:08:06 +0000 | [diff] [blame] | 119 | reportRebuild(handler, action, keyDescription == null |
| 120 | ? "action command has changed" |
| 121 | : "action command has changed.\nNew action: " + keyDescription); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 122 | } else { |
| 123 | reportRebuild(handler, action, |
| 124 | "action command has changed (try --verbose_explanations for more info)"); |
| 125 | } |
| 126 | } |
| 127 | } |
| 128 | |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 129 | private void reportClientEnv(EventHandler handler, Action action, Map<String, String> used) { |
| 130 | if (handler != null) { |
| 131 | if (verboseExplanations) { |
| 132 | StringBuilder message = new StringBuilder(); |
| 133 | message.append("Effective client environment has changed. Now using\n"); |
| 134 | for (Map.Entry<String, String> entry : used.entrySet()) { |
| 135 | message.append(" ").append(entry.getKey()).append("=").append(entry.getValue()) |
| 136 | .append("\n"); |
| 137 | } |
| 138 | reportRebuild(handler, action, message.toString()); |
| 139 | } else { |
| 140 | reportRebuild( |
| 141 | handler, |
| 142 | action, |
| 143 | "Effective client environment has changed (try --verbose_explanations for more info)"); |
| 144 | } |
| 145 | } |
| 146 | } |
| 147 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 148 | protected boolean unconditionalExecution(Action action) { |
| 149 | return !isActionExecutionProhibited(action) && action.executeUnconditionally(); |
| 150 | } |
| 151 | |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 152 | private static Map<String, String> computeUsedClientEnv( |
| 153 | Action action, Map<String, String> clientEnv) { |
| 154 | Map<String, String> used = new HashMap<>(); |
| 155 | for (String var : action.getClientEnvironmentVariables()) { |
| 156 | String value = clientEnv.get(var); |
| 157 | if (value != null) { |
| 158 | used.put(var, value); |
| 159 | } |
| 160 | } |
| 161 | return used; |
| 162 | } |
| 163 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 164 | /** |
| 165 | * Checks whether {@code action} needs to be executed and returns a non-null Token if so. |
| 166 | * |
| 167 | * <p>The method checks if any of the action's inputs or outputs have changed. Returns a non-null |
| 168 | * {@link Token} if the action needs to be executed, and null otherwise. |
| 169 | * |
| 170 | * <p>If this method returns non-null, indicating that the action will be executed, the |
Janak Ramakrishnan | 73055be | 2015-04-13 18:32:49 +0000 | [diff] [blame] | 171 | * metadataHandler's {@link MetadataHandler#discardOutputMetadata} method must be called, so that |
| 172 | * it does not serve stale metadata for the action's outputs after the action is executed. |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 173 | */ |
| 174 | // Note: the handler should only be used for DEPCHECKER events; there's no |
| 175 | // guarantee it will be available for other events. |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 176 | public Token getTokenIfNeedToExecute( |
| 177 | Action action, |
| 178 | Iterable<Artifact> resolvedCacheArtifacts, |
| 179 | Map<String, String> clientEnv, |
| 180 | EventHandler handler, |
| 181 | MetadataHandler metadataHandler) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 182 | // TODO(bazel-team): (2010) For RunfilesAction/SymlinkAction and similar actions that |
| 183 | // produce only symlinks we should not check whether inputs are valid at all - all that matters |
| 184 | // that inputs and outputs are still exist (and new inputs have not appeared). All other checks |
| 185 | // are unnecessary. In other words, the only metadata we should check for them is file existence |
| 186 | // itself. |
| 187 | |
| 188 | MiddlemanType middlemanType = action.getActionType(); |
| 189 | if (middlemanType.isMiddleman()) { |
| 190 | // Some types of middlemen are not checked because they should not |
| 191 | // propagate invalidation of their inputs. |
| 192 | if (middlemanType != MiddlemanType.ERROR_PROPAGATING_MIDDLEMAN) { |
| 193 | checkMiddlemanAction(action, handler, metadataHandler); |
| 194 | } |
| 195 | return null; |
| 196 | } |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 197 | Iterable<Artifact> actionInputs = action.getInputs(); |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 198 | // Resolve action inputs from cache, if necessary. |
Lukacs Berki | 5ea2b14 | 2017-02-28 10:46:53 +0000 | [diff] [blame] | 199 | boolean inputsDiscovered = action.inputsDiscovered(); |
| 200 | if (!inputsDiscovered && resolvedCacheArtifacts != null) { |
Janak Ramakrishnan | 90f3d34 | 2015-03-27 19:45:18 +0000 | [diff] [blame] | 201 | // The action doesn't know its inputs, but the caller has a good idea of what they are. |
| 202 | Preconditions.checkState(action.discoversInputs(), |
| 203 | "Actions that don't know their inputs must discover them: %s", action); |
| 204 | actionInputs = resolvedCacheArtifacts; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 205 | } |
Janak Ramakrishnan | 90f3d34 | 2015-03-27 19:45:18 +0000 | [diff] [blame] | 206 | ActionCache.Entry entry = getCacheEntry(action); |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 207 | if (mustExecute(action, entry, handler, metadataHandler, actionInputs, clientEnv)) { |
Janak Ramakrishnan | e3f04b8 | 2015-03-27 16:45:34 +0000 | [diff] [blame] | 208 | if (entry != null) { |
| 209 | removeCacheEntry(action); |
| 210 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 211 | return new Token(getKeyString(action)); |
| 212 | } |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 213 | |
Lukacs Berki | 5ea2b14 | 2017-02-28 10:46:53 +0000 | [diff] [blame] | 214 | if (!inputsDiscovered) { |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 215 | action.updateInputs(actionInputs); |
| 216 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 217 | return null; |
| 218 | } |
| 219 | |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 220 | protected boolean mustExecute( |
| 221 | Action action, |
| 222 | @Nullable ActionCache.Entry entry, |
| 223 | EventHandler handler, |
| 224 | MetadataHandler metadataHandler, |
| 225 | Iterable<Artifact> actionInputs, |
| 226 | Map<String, String> clientEnv) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 227 | // Unconditional execution can be applied only for actions that are allowed to be executed. |
| 228 | if (unconditionalExecution(action)) { |
| 229 | Preconditions.checkState(action.isVolatile()); |
| 230 | reportUnconditionalExecution(handler, action); |
| 231 | return true; // must execute - unconditional execution is requested. |
| 232 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 233 | if (entry == null) { |
| 234 | reportNewAction(handler, action); |
| 235 | return true; // must execute -- no cache entry (e.g. first build) |
| 236 | } |
| 237 | |
| 238 | if (entry.isCorrupted()) { |
| 239 | reportCorruptedCacheEntry(handler, action); |
| 240 | return true; // cache entry is corrupted - must execute |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 241 | } else if (validateArtifacts(entry, action, actionInputs, metadataHandler, true)) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 242 | reportChanged(handler, action); |
| 243 | return true; // files have changed |
Eric Fellheimer | 1a34a24 | 2016-01-21 19:17:19 +0000 | [diff] [blame] | 244 | } else if (!entry.getActionKey().equals(action.getKey())) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 245 | reportCommand(handler, action); |
| 246 | return true; // must execute -- action key is different |
| 247 | } |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 248 | Map<String, String> usedClientEnv = computeUsedClientEnv(action, clientEnv); |
| 249 | if (!entry.getUsedClientEnvDigest().equals(DigestUtils.fromEnv(usedClientEnv))) { |
| 250 | reportClientEnv(handler, action, usedClientEnv); |
| 251 | return true; // different values taken from the environment -- must execute |
| 252 | } |
| 253 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 254 | |
| 255 | entry.getFileDigest(); |
| 256 | return false; // cache hit |
| 257 | } |
| 258 | |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 259 | public void afterExecution( |
| 260 | Action action, Token token, MetadataHandler metadataHandler, Map<String, String> clientEnv) |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 261 | throws IOException { |
| 262 | Preconditions.checkArgument(token != null); |
| 263 | String key = token.cacheKey; |
Janak Ramakrishnan | e3f04b8 | 2015-03-27 16:45:34 +0000 | [diff] [blame] | 264 | if (actionCache.get(key) != null) { |
| 265 | // This cache entry has already been updated by a shared action. We don't need to do it again. |
| 266 | return; |
| 267 | } |
Klaus Aehlig | 6f33a1c | 2016-09-13 16:46:10 +0000 | [diff] [blame] | 268 | Map<String, String> usedClientEnv = computeUsedClientEnv(action, clientEnv); |
| 269 | ActionCache.Entry entry = |
Googler | a2d61fa | 2017-02-16 16:14:32 +0000 | [diff] [blame] | 270 | actionCache.newEntry(action.getKey(), usedClientEnv, action.discoversInputs()); |
| 271 | if (entry == null) { |
| 272 | // Action cache is disabled, don't generate digests. |
| 273 | return; |
| 274 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 275 | for (Artifact output : action.getOutputs()) { |
| 276 | // Remove old records from the cache if they used different key. |
| 277 | String execPath = output.getExecPathString(); |
| 278 | if (!key.equals(execPath)) { |
Philipp Wollermann | e03758b | 2015-06-15 08:10:47 +0000 | [diff] [blame] | 279 | actionCache.remove(execPath); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 280 | } |
Michajlo Matijkiw | 13459b4 | 2015-03-12 19:43:20 +0000 | [diff] [blame] | 281 | if (!metadataHandler.artifactOmitted(output)) { |
| 282 | // Output files *must* exist and be accessible after successful action execution. |
| 283 | Metadata metadata = metadataHandler.getMetadata(output); |
| 284 | Preconditions.checkState(metadata != null); |
| 285 | entry.addFile(output.getExecPath(), metadata); |
| 286 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 287 | } |
| 288 | for (Artifact input : action.getInputs()) { |
| 289 | entry.addFile(input.getExecPath(), metadataHandler.getMetadataMaybe(input)); |
| 290 | } |
| 291 | entry.getFileDigest(); |
| 292 | actionCache.put(key, entry); |
| 293 | } |
| 294 | |
Janak Ramakrishnan | 90f3d34 | 2015-03-27 19:45:18 +0000 | [diff] [blame] | 295 | @Nullable |
Michajlo Matijkiw | 331633c | 2015-06-09 22:09:03 +0000 | [diff] [blame] | 296 | public Iterable<Artifact> getCachedInputs(Action action, PackageRootResolver resolver) |
Janak Ramakrishnan | 3c0adb2 | 2016-08-15 21:54:55 +0000 | [diff] [blame] | 297 | throws PackageRootResolutionException, InterruptedException { |
Janak Ramakrishnan | 90f3d34 | 2015-03-27 19:45:18 +0000 | [diff] [blame] | 298 | ActionCache.Entry entry = getCacheEntry(action); |
| 299 | if (entry == null || entry.isCorrupted()) { |
| 300 | return ImmutableList.of(); |
| 301 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 302 | |
| 303 | List<PathFragment> outputs = new ArrayList<>(); |
| 304 | for (Artifact output : action.getOutputs()) { |
| 305 | outputs.add(output.getExecPath()); |
| 306 | } |
Lukacs Berki | 1f2caa5 | 2017-02-02 10:47:07 +0000 | [diff] [blame] | 307 | List<PathFragment> inputExecPaths = new ArrayList<>(); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 308 | for (String path : entry.getPaths()) { |
| 309 | PathFragment execPath = new PathFragment(path); |
| 310 | // Code assumes that action has only 1-2 outputs and ArrayList.contains() will be |
| 311 | // most efficient. |
| 312 | if (!outputs.contains(execPath)) { |
Lukacs Berki | 1f2caa5 | 2017-02-02 10:47:07 +0000 | [diff] [blame] | 313 | inputExecPaths.add(execPath); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 314 | } |
| 315 | } |
Lukacs Berki | 1f2caa5 | 2017-02-02 10:47:07 +0000 | [diff] [blame] | 316 | |
| 317 | // Note that this method may trigger a violation of the desirable invariant that getInputs() |
| 318 | // is a superset of getMandatoryInputs(). See bug about an "action not in canonical form" |
| 319 | // error message and the integration test test_crosstool_change_and_failure(). |
| 320 | Map<PathFragment, Artifact> allowedDerivedInputsMap = new HashMap<>(); |
| 321 | for (Artifact derivedInput : action.getAllowedDerivedInputs()) { |
| 322 | if (!derivedInput.isSourceArtifact()) { |
| 323 | allowedDerivedInputsMap.put(derivedInput.getExecPath(), derivedInput); |
| 324 | } |
| 325 | } |
| 326 | |
| 327 | List<Artifact> inputArtifacts = new ArrayList<>(); |
| 328 | List<PathFragment> unresolvedPaths = new ArrayList<>(); |
| 329 | for (PathFragment execPath : inputExecPaths) { |
| 330 | Artifact artifact = allowedDerivedInputsMap.get(execPath); |
| 331 | if (artifact != null) { |
| 332 | inputArtifacts.add(artifact); |
| 333 | } else { |
| 334 | // Remember this execPath, we will try to resolve it as a source artifact. |
| 335 | unresolvedPaths.add(execPath); |
| 336 | } |
| 337 | } |
| 338 | |
| 339 | Map<PathFragment, Artifact> resolvedArtifacts = |
| 340 | artifactResolver.resolveSourceArtifacts(unresolvedPaths, resolver); |
| 341 | if (resolvedArtifacts == null) { |
| 342 | // We are missing some dependencies. We need to rerun this update later. |
| 343 | return null; |
| 344 | } |
| 345 | |
| 346 | for (PathFragment execPath : unresolvedPaths) { |
| 347 | Artifact artifact = resolvedArtifacts.get(execPath); |
| 348 | // If PathFragment cannot be resolved into the artifact, ignore it. This could happen if the |
| 349 | // rule has changed and the action no longer depends on, e.g., an additional source file in a |
| 350 | // separate package and that package is no longer referenced anywhere else. It is safe to |
| 351 | // ignore such paths because dependency checker would identify changes in inputs (ignored path |
| 352 | // was used before) and will force action execution. |
| 353 | if (artifact != null) { |
| 354 | inputArtifacts.add(artifact); |
| 355 | } |
| 356 | } |
| 357 | return inputArtifacts; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 358 | } |
| 359 | |
| 360 | /** |
| 361 | * Special handling for the MiddlemanAction. Since MiddlemanAction output |
| 362 | * artifacts are purely fictional and used only to stay within dependency |
| 363 | * graph model limitations (action has to depend on artifacts, not on other |
| 364 | * actions), we do not need to validate metadata for the outputs - only for |
| 365 | * inputs. We also do not need to validate MiddlemanAction key, since action |
| 366 | * cache entry key already incorporates that information for the middlemen |
| 367 | * and we will experience a cache miss when it is different. Whenever it |
| 368 | * encounters middleman artifacts as input artifacts for other actions, it |
| 369 | * consults with the aggregated middleman digest computed here. |
| 370 | */ |
| 371 | protected void checkMiddlemanAction(Action action, EventHandler handler, |
| 372 | MetadataHandler metadataHandler) { |
| 373 | Artifact middleman = action.getPrimaryOutput(); |
| 374 | String cacheKey = middleman.getExecPathString(); |
| 375 | ActionCache.Entry entry = actionCache.get(cacheKey); |
| 376 | boolean changed = false; |
| 377 | if (entry != null) { |
| 378 | if (entry.isCorrupted()) { |
| 379 | reportCorruptedCacheEntry(handler, action); |
| 380 | changed = true; |
Janak Ramakrishnan | a7c84b5 | 2015-03-18 21:59:16 +0000 | [diff] [blame] | 381 | } else if (validateArtifacts(entry, action, action.getInputs(), metadataHandler, false)) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 382 | reportChanged(handler, action); |
| 383 | changed = true; |
| 384 | } |
| 385 | } else { |
| 386 | reportChangedDeps(handler, action); |
| 387 | changed = true; |
| 388 | } |
| 389 | if (changed) { |
| 390 | // Compute the aggregated middleman digest. |
| 391 | // Since we never validate action key for middlemen, we should not store |
| 392 | // it in the cache entry and just use empty string instead. |
Googler | a2d61fa | 2017-02-16 16:14:32 +0000 | [diff] [blame] | 393 | entry = actionCache.newEntry("", ImmutableMap.<String, String>of(), false); |
| 394 | if (entry != null) { |
| 395 | for (Artifact input : action.getInputs()) { |
| 396 | entry.addFile(input.getExecPath(), metadataHandler.getMetadataMaybe(input)); |
| 397 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 398 | } |
| 399 | } |
| 400 | |
Googler | a2d61fa | 2017-02-16 16:14:32 +0000 | [diff] [blame] | 401 | // Action cache is disabled, skip the digest. |
| 402 | if (entry == null) { |
| 403 | return; |
| 404 | } |
| 405 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 406 | metadataHandler.setDigestForVirtualArtifact(middleman, entry.getFileDigest()); |
| 407 | if (changed) { |
| 408 | actionCache.put(cacheKey, entry); |
| 409 | } |
| 410 | } |
| 411 | |
| 412 | /** |
| 413 | * Returns an action key. It is always set to the first output exec path string. |
| 414 | */ |
| 415 | private static String getKeyString(Action action) { |
| 416 | Preconditions.checkState(!action.getOutputs().isEmpty()); |
| 417 | return action.getOutputs().iterator().next().getExecPathString(); |
| 418 | } |
| 419 | |
| 420 | |
| 421 | /** |
| 422 | * In most cases, this method should not be called directly - reportXXX() methods |
| 423 | * should be used instead. This is done to avoid cost associated with building |
| 424 | * the message. |
| 425 | */ |
| 426 | private static void reportRebuild(@Nullable EventHandler handler, Action action, String message) { |
| 427 | // For MiddlemanAction, do not report rebuild. |
| 428 | if (handler != null && !action.getActionType().isMiddleman()) { |
Ulf Adams | 760e709 | 2016-04-21 08:09:51 +0000 | [diff] [blame] | 429 | handler.handle(Event.of( |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 430 | EventKind.DEPCHECKER, null, "Executing " + action.prettyPrint() + ": " + message + ".")); |
| 431 | } |
| 432 | } |
| 433 | |
| 434 | // Called by IncrementalDependencyChecker. |
| 435 | protected static void reportUnconditionalExecution( |
| 436 | @Nullable EventHandler handler, Action action) { |
| 437 | reportRebuild(handler, action, "unconditional execution is requested"); |
| 438 | } |
| 439 | |
| 440 | private static void reportChanged(@Nullable EventHandler handler, Action action) { |
| 441 | reportRebuild(handler, action, "One of the files has changed"); |
| 442 | } |
| 443 | |
| 444 | private static void reportChangedDeps(@Nullable EventHandler handler, Action action) { |
| 445 | reportRebuild(handler, action, "the set of files on which this action depends has changed"); |
| 446 | } |
| 447 | |
| 448 | private static void reportNewAction(@Nullable EventHandler handler, Action action) { |
| 449 | reportRebuild(handler, action, "no entry in the cache (action is new)"); |
| 450 | } |
| 451 | |
| 452 | private static void reportCorruptedCacheEntry(@Nullable EventHandler handler, Action action) { |
| 453 | reportRebuild(handler, action, "cache entry is corrupted"); |
| 454 | } |
| 455 | |
| 456 | /** Wrapper for all context needed by the ActionCacheChecker to handle a single action. */ |
| 457 | public static final class Token { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 458 | private final String cacheKey; |
| 459 | |
| 460 | private Token(String cacheKey) { |
| 461 | this.cacheKey = Preconditions.checkNotNull(cacheKey); |
| 462 | } |
| 463 | } |
| 464 | } |