// Copyright 2014 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.skyframe; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; import com.google.common.collect.Interner; import com.google.devtools.build.lib.analysis.config.BuildConfiguration; import com.google.devtools.build.lib.analysis.config.BuildConfiguration.Fragment; import com.google.devtools.build.lib.analysis.config.BuildOptions; import com.google.devtools.build.lib.concurrent.BlazeInterners; import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; import com.google.devtools.build.lib.packages.RuleClassProvider; import com.google.devtools.build.lib.skyframe.serialization.DeserializationContext; import com.google.devtools.build.lib.skyframe.serialization.InjectingObjectCodec; import com.google.devtools.build.lib.skyframe.serialization.ObjectCodec; import com.google.devtools.build.lib.skyframe.serialization.SerializationContext; import com.google.devtools.build.lib.skyframe.serialization.SerializationException; import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec; import com.google.devtools.build.lib.skyframe.serialization.strings.StringCodecs; import com.google.devtools.build.lib.util.Fingerprint; import com.google.devtools.build.lib.vfs.FileSystemProvider; import com.google.devtools.build.skyframe.SkyFunctionName; import com.google.devtools.build.skyframe.SkyKey; import com.google.devtools.build.skyframe.SkyValue; import com.google.protobuf.CodedInputStream; import com.google.protobuf.CodedOutputStream; import java.io.IOException; import java.util.Objects; import javax.annotation.Nullable; /** A Skyframe node representing a build configuration fragment. */ @Immutable @ThreadSafe @AutoCodec(dependency = FileSystemProvider.class) public class ConfigurationFragmentValue implements SkyValue { public static final InjectingObjectCodec CODEC = new ConfigurationFragmentValue_AutoCodec(); @Nullable private final BuildConfiguration.Fragment fragment; ConfigurationFragmentValue(BuildConfiguration.Fragment fragment) { this.fragment = fragment; } public BuildConfiguration.Fragment getFragment() { return fragment; } @ThreadSafe public static ConfigurationFragmentKey key( BuildOptions buildOptions, Class fragmentType, RuleClassProvider ruleClassProvider) { BuildOptions optionsKey = buildOptions.trim( BuildConfiguration.getOptionsClasses( ImmutableList.>of(fragmentType), ruleClassProvider)); return ConfigurationFragmentKey.of(optionsKey, fragmentType); } /** {@link SkyKey} for {@link ConfigurationFragmentValue}. */ public static final class ConfigurationFragmentKey implements SkyKey { private static Interner interner = BlazeInterners.newWeakInterner(); public static final ObjectCodec CODEC = new Codec(); private final BuildOptions buildOptions; private final String checksum; private final Class fragmentType; private ConfigurationFragmentKey( BuildOptions buildOptions, Class fragmentType) { this.buildOptions = Preconditions.checkNotNull(buildOptions); this.checksum = Fingerprint.md5Digest(buildOptions.computeCacheKey()); this.fragmentType = Preconditions.checkNotNull(fragmentType); } private static ConfigurationFragmentKey of( BuildOptions buildOptions, Class fragmentType) { return interner.intern(new ConfigurationFragmentKey(buildOptions, fragmentType)); } public BuildOptions getBuildOptions() { return buildOptions; } public Class getFragmentType() { return fragmentType; } @Override public boolean equals(Object o) { if (this == o) { return true; } if (!(o instanceof ConfigurationFragmentKey)) { return false; } ConfigurationFragmentKey confObject = (ConfigurationFragmentKey) o; return Objects.equals(fragmentType, confObject.fragmentType) && Objects.equals(buildOptions, confObject.buildOptions); } @Override public int hashCode() { return Objects.hash(buildOptions, fragmentType); } @Override public String toString() { return String.format("ConfigurationFragmentKey(class=%s, checksum=%s)", fragmentType.getName(), checksum); } @Override public SkyFunctionName functionName() { return SkyFunctions.CONFIGURATION_FRAGMENT; } private static class Codec implements ObjectCodec { @Override public Class getEncodedClass() { return ConfigurationFragmentKey.class; } @Override public void serialize( SerializationContext context, ConfigurationFragmentKey obj, CodedOutputStream codedOut) throws SerializationException, IOException { BuildOptions.CODEC.serialize(context, obj.buildOptions, codedOut); StringCodecs.asciiOptimized().serialize(context, obj.fragmentType.getName(), codedOut); } @SuppressWarnings("unchecked") // Cast to Class. @Override public ConfigurationFragmentKey deserialize( DeserializationContext context, CodedInputStream codedIn) throws SerializationException, IOException { try { return of( BuildOptions.CODEC.deserialize(context, codedIn), (Class) Class.forName(StringCodecs.asciiOptimized().deserialize(context, codedIn))); } catch (ClassNotFoundException e) { throw new SerializationException("Couldn't deserialize ConfigurationFragmentKey", e); } } } } }