// Copyright 2017 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.skyframe; import com.google.common.base.MoreObjects; import com.google.common.collect.Interner; import com.google.devtools.build.lib.concurrent.BlazeInterners; import com.google.devtools.build.lib.util.Preconditions; /** * Container for a pending operation on the reverse deps set. We use subclasses to save 8 bytes of * memory instead of keeping a field in this class, and we store {@link Op#CHECK} or {@link Op#ADD} * operations as the bare {@link SkyKey} in order to save the wrapper object in that case. * * <p>When a list of {@link KeyToConsolidate} operations is processed, each operation is performed * in order. Operations on a done or freshly evaluating node entry are straightforward: they apply * to the entry's reverse deps. Operations on a re-evaluating node entry have a double meaning: they * will eventually be applied to the node entry's existing reverse deps, just as for a done node * entry, but they are also used to track the entries that declared/redeclared a reverse dep on this * entry during this evaluation (and will thus need to be signaled when this entry finishes * evaluating). */ abstract class KeyToConsolidate { enum Op { /** * Assert that the reverse dep is already present in the set of reverse deps. If the entry is * re-evaluating, add this reverse dep to the set of reverse deps to signal when this entry is * done. */ CHECK, /** * Add the reverse dep to the set of reverse deps and assert it was not already present. If the * entry is re-evaluating, add this reverse dep to the set of reverse deps to signal when this * entry is done. */ ADD, /** * Remove the reverse dep from the set of reverse deps and assert it was present. If the entry * is re-evaluating, also remove the reverse dep from the set of reverse deps to signal when * this entry is done, and assert that it was present. */ REMOVE, /** * The same as {@link #REMOVE}, except that if the entry is re-evaluating, we assert that the * set of reverse deps to signal did <i>not</i> contain this reverse dep. */ REMOVE_OLD } enum OpToStoreBare { ADD(Op.ADD), CHECK(Op.CHECK); private final Op op; OpToStoreBare(Op op) { this.op = op; } } private static final Interner<KeyToConsolidate> consolidateInterner = BlazeInterners.newWeakInterner(); private final SkyKey key; /** Do not call directly -- use the {@link #create} static method instead. */ private KeyToConsolidate(SkyKey key) { this.key = key; } @Override public String toString() { return MoreObjects.toStringHelper(this).add("key", key).toString(); } /** * Gets which operation was delayed for the given object, created using {@link #create}. The same * {@code opToStoreBare} passed in to {@link #create} should be passed in here. */ static Op op(Object obj, OpToStoreBare opToStoreBare) { if (obj instanceof SkyKey) { return opToStoreBare.op; } if (obj instanceof KeyToAdd) { return Op.ADD; } if (obj instanceof KeyToCheck) { return Op.CHECK; } if (obj instanceof KeyToRemove) { return Op.REMOVE; } if (obj instanceof KeyToRemoveOld) { return Op.REMOVE_OLD; } throw new IllegalStateException( "Unknown object type: " + obj + ", " + opToStoreBare + ", " + obj.getClass()); } /** Gets the key whose operation was delayed for the given object. */ static SkyKey key(Object obj) { if (obj instanceof SkyKey) { return (SkyKey) obj; } Preconditions.checkState(obj instanceof KeyToConsolidate, obj); return ((KeyToConsolidate) obj).key; } /** * Creates a new operation, encoding the operation {@code op} with reverse dep {@code key}. To * save memory, the caller should specify the most common operation expected as {@code * opToStoreBare}. That operation will be encoded as the raw {@code key}, saving the memory of an * object wrapper. Whatever {@code opToStoreBare} is set to here, the same value must be passed in * to {@link #op} when decoding an operation emitted by this method. */ static Object create(SkyKey key, Op op, OpToStoreBare opToStoreBare) { if (op == opToStoreBare.op) { return key; } switch (op) { case CHECK: return consolidateInterner.intern(new KeyToCheck(key)); case REMOVE: return consolidateInterner.intern(new KeyToRemove(key)); case REMOVE_OLD: return consolidateInterner.intern(new KeyToRemoveOld(key)); case ADD: return consolidateInterner.intern(new KeyToAdd(key)); default: throw new IllegalStateException(op + ", " + key); } } @Override public boolean equals(Object obj) { if (obj == null) { return false; } return this.getClass() == obj.getClass() && this.key.equals(((KeyToConsolidate) obj).key); } protected int keyHashCode() { return key.hashCode(); } @Override public int hashCode() { // Overridden in subclasses. throw new UnsupportedOperationException(key.toString()); } private static final class KeyToAdd extends KeyToConsolidate { KeyToAdd(SkyKey key) { super(key); } @Override public int hashCode() { return keyHashCode(); } } private static final class KeyToCheck extends KeyToConsolidate { KeyToCheck(SkyKey key) { super(key); } @Override public int hashCode() { return 31 + 43 * keyHashCode(); } } private static final class KeyToRemove extends KeyToConsolidate { KeyToRemove(SkyKey key) { super(key); } @Override public int hashCode() { return 42 + 37 * keyHashCode(); } } private static final class KeyToRemoveOld extends KeyToConsolidate { KeyToRemoveOld(SkyKey key) { super(key); } @Override public int hashCode() { return 93 + 37 * keyHashCode(); } } }