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

org.wildfly.security.auth.realm.IdentitySharedExclusiveLock Maven / Gradle / Ivy

The newest version!
/*
 * JBoss, Home of Professional Open Source.
 * Copyright 2016 Red Hat, Inc., and individual contributors
 * as indicated by the @author tags.
 *
 * 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 org.wildfly.security.auth.realm;

/**
 * A simple shared/exclusive lock for a realm identity.
 *
 * @author Farah Juma
 */
public class IdentitySharedExclusiveLock {

    private int sharedHoldCount;
    private boolean isExclusiveLocked;
    private int exclusiveRequests;

    /**
     * Acquire the exclusive lock. An invocation of this method will block until the lock can be acquired.
     *
     * @return a lock object representing the newly acquired lock
     */
    public synchronized IdentityLock lockExclusive() {
        boolean interrupted = false;
        try {
            exclusiveRequests++;
            while ((sharedHoldCount > 0) || isExclusiveLocked) {
                try {
                    wait();
                } catch (InterruptedException e) {
                    interrupted = true;
                }
            }
            isExclusiveLocked = true;
            exclusiveRequests--;
            return new IdentityLock(true);
        } finally {
            if (interrupted) {
                Thread.currentThread().interrupt();
            }
        }

    }

    /**
     * Acquire a shared lock. An invocation of this method will block until the lock can be acquired.
     *
     * @return a lock object representing the newly acquired lock
     */
    public synchronized IdentityLock lockShared() {
        boolean interrupted = false;
        try {
            while (isExclusiveLocked || (exclusiveRequests > 0)) {
                try {
                    wait();
                } catch (InterruptedException e) {
                    interrupted = true;
                }
            }
            sharedHoldCount++;
            return new IdentityLock(false);
        } finally {
            if (interrupted) {
                Thread.currentThread().interrupt();
            }
        }
    }

    private synchronized void release(IdentityLock identityLock) {
        if (identityLock.isExclusive()) {
            isExclusiveLocked = false;
            notifyAll();
        } else {
            if (--sharedHoldCount == 0) {
                notifyAll();
            }
        }

    }

    /**
     * Class that represents a lock on a realm identity. A lock object is created each time a lock is
     * acquired on a realm identity via {@link IdentitySharedExclusiveLock#lockExclusive()} or
     * {@link IdentitySharedExclusiveLock#lockShared()}.
     */
    public class IdentityLock implements AutoCloseable {

        private final boolean exclusive;
        private volatile boolean valid = true;

        /**
         * Construct a new instance.
         *
         * @param exclusive {@code true} if this lock is exclusive, {@code false} if this lock is shared
         */
        public IdentityLock(final boolean exclusive) {
            this.exclusive = exclusive;
        }

        /**
         * Release this lock. Invoking this method has no effect if this lock is invalid.
         */
        public synchronized void release() {
            if (valid) {
                IdentitySharedExclusiveLock.this.release(this);
                valid = false;
            }
        }

        @Override
        public void close() {
            release();
        }

        /**
         * Determine whether this lock is exclusive or shared.
         *
         * @return {@code true} if this lock is exclusive, {@code false} if this lock is shared
         */
        public boolean isExclusive() {
            return exclusive;
        }

        /**
         * Determine whether this lock is valid. A lock starts out valid and becomes invalid when it
         * is released via {@link #release()} or {@link #close()}.
         *
         * @return {@code true} if this lock is valid, {@code false} otherwise
         */
        public boolean isValid() {
            return valid;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy