All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.servicetalk.concurrent.internal.SequentialCancellable Maven / Gradle / Ivy

/*
 * Copyright © 2018 Apple Inc. and the ServiceTalk project authors
 *
 * 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 io.servicetalk.concurrent.internal;

import io.servicetalk.concurrent.Cancellable;

import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import javax.annotation.Nullable;

import static java.util.Objects.requireNonNull;
import static java.util.concurrent.atomic.AtomicReferenceFieldUpdater.newUpdater;

/**
 * A {@link Cancellable} that can hold at most one {@link Cancellable} that will be cancelled when this is cancelled.
 */
public class SequentialCancellable implements Cancellable {

    private static final Cancellable CANCELLED = () -> { };

    private static final AtomicReferenceFieldUpdater currentUpdater =
            newUpdater(SequentialCancellable.class, Cancellable.class, "current");
    @SuppressWarnings("unused")
    @Nullable
    private volatile Cancellable current;

    /**
     * Create a new instance with no current {@link Cancellable}.
     */
    public SequentialCancellable() {
    }

    /**
     * Create a new instance with the current {@link Cancellable} set to {@code cancellable}.
     * @param cancellable the initial {@link Cancellable}.
     */
    public SequentialCancellable(Cancellable cancellable) {
        this.current = requireNonNull(cancellable);
    }

    /**
     * Sets the current {@link Cancellable}.
     *
     * @param next to set.
     */
    public final void nextCancellable(Cancellable next) {
        Cancellable oldVal = currentUpdater.getAndSet(this, requireNonNull(next));
        if (oldVal == CANCELLED) {
            next.cancel();
        }
    }

    @Override
    public final void cancel() {
        Cancellable oldVal = currentUpdater.getAndSet(this, CANCELLED);
        if (oldVal != null) {
            oldVal.cancel();
        }
    }

    /**
     * Returns {@code true} if this {@link Cancellable} is cancelled.
     *
     * @return {@code true} if this {@link Cancellable} is cancelled.
     */
    public boolean isCancelled() {
        return current == CANCELLED;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy