blob: 66fce142d86ba76f8318f2045d6d8b63702c8755 [file] [log] [blame]
// 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;
}
}
}