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

net.tascalate.concurrent.locks.DefaultAsyncLock Maven / Gradle / Ivy

Go to download

Implementation of blocking (IO-Bound) cancellable java.util.concurrent.CompletionStage and related extensions to java.util.concurrent.ExecutorService-s

There is a newer version: 0.9.8
Show newest version
/**
 * Copyright 2015-2020 Valery Silaev (http://vsilaev.com)
 *
 * 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 net.tascalate.concurrent.locks;

import java.util.Optional;
import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;

import net.tascalate.concurrent.CompletableFutureWrapper;
import net.tascalate.concurrent.Promise;

public class DefaultAsyncLock implements AsyncLock {
    final Queue waiters = new ConcurrentLinkedQueue<>();
    
    @Override
    public Optional tryAcquire() {
        Promise lock = acquire();
        if (lock.isDone()) {
            return Optional.of(lock.join());
        } else {
            waiters.remove(lock);
            return Optional.empty();
        }
    }
    
    @Override
    public Promise acquire() {
        LockPromise promise = new LockPromise();
        waiters.add(promise); // Add to tail;
        
        nextWaiter();
        
        return promise;
    }

    private void nextWaiter() {
        LockPromise head = waiters.peek(); // Peek from head
        if (head != null) {
            head.acquire();
        }
    }

    private class LockPromise extends CompletableFutureWrapper {
        private final AtomicBoolean released = new AtomicBoolean();
        private final Token token = this::release;

        @Override
        public boolean cancel(boolean mayInterruptIfRunning) {
            if (super.cancel(mayInterruptIfRunning)) {
                waiters.remove(this);
                return true;
            } else {
                return false;
            }
        }

        boolean acquire() {
            return success(token);
        }
        
        void release() {
            if (released.compareAndSet(false, true)) {
                LockPromise released = waiters.poll(); // Remove current head
                if (this != released) {
                    throw new IllegalStateException("The released lock is not equal to the top of the queue");
                }
                nextWaiter();
            }            
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy