| // Copyright 2019 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.supplier; | 
 |  | 
 | import com.google.common.base.Preconditions; | 
 | import javax.annotation.Nullable; | 
 |  | 
 | /** | 
 |  * An {@link InterruptibleSupplier} which should cache the instance retrieved during the first call | 
 |  * to {@link #get} and returns that value on subsequent calls to {@link #get}. | 
 |  * | 
 |  * <p>This is similar to, but not exactly the same as, what is returned by {@link | 
 |  * com.google.common.base.Suppliers#memoize}. | 
 |  * | 
 |  * <p>Implementations should be thread-safe. | 
 |  * | 
 |  * <p>Unlike that implementation, this is not serializable, and its initialized state (whether an | 
 |  * instance has been retrieved) is visible via {@link #isInitialized}. | 
 |  */ | 
 | public interface MemoizingInterruptibleSupplier<T> extends InterruptibleSupplier<T> { | 
 |  | 
 |   /** Returns {@code true} if the result of {@link #get} is readily available. */ | 
 |   boolean isInitialized(); | 
 |  | 
 |   static <T> MemoizingInterruptibleSupplier<T> of(InterruptibleSupplier<T> delegate) { | 
 |     if (delegate instanceof MemoizingInterruptibleSupplier) { | 
 |       return (MemoizingInterruptibleSupplier<T>) delegate; | 
 |     } | 
 |     return new DelegatingMemoizingSupplier<>(delegate); | 
 |   } | 
 |  | 
 |   /** Memoizes the result of {@code delegate} after the first call to {@link #get}. */ | 
 |   final class DelegatingMemoizingSupplier<T> implements MemoizingInterruptibleSupplier<T> { | 
 |  | 
 |     @Nullable private InterruptibleSupplier<T> delegate; | 
 |     @Nullable private volatile T value = null; | 
 |  | 
 |     private DelegatingMemoizingSupplier(InterruptibleSupplier<T> delegate) { | 
 |       this.delegate = Preconditions.checkNotNull(delegate); | 
 |     } | 
 |  | 
 |     @Override | 
 |     public T get() throws InterruptedException { | 
 |       if (value != null) { | 
 |         return value; | 
 |       } | 
 |       synchronized (this) { | 
 |         if (value == null) { | 
 |           value = delegate.get(); | 
 |           delegate = null; // Free up for GC. | 
 |         } | 
 |       } | 
 |       return value; | 
 |     } | 
 |  | 
 |     @Override | 
 |     public boolean isInitialized() { | 
 |       return value != null; | 
 |     } | 
 |   } | 
 | } |