001/*
002 * ============================================================================
003 *  Copyright © 2002-2024 by Thomas Thrien.
004 *  All Rights Reserved.
005 * ============================================================================
006 *  Licensed to the public under the agreements of the GNU Lesser General Public
007 *  License, version 3.0 (the "License"). You may obtain a copy of the License at
008 *
009 *       http://www.gnu.org/licenses/lgpl.html
010 *
011 *  Unless required by applicable law or agreed to in writing, software
012 *  distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
013 *  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
014 *  License for the specific language governing permissions and limitations
015 *  under the License.
016 */
017
018package org.tquadrat.foundation.util.internal;
019
020import static org.apiguardian.api.API.Status.INTERNAL;
021import static org.apiguardian.api.API.Status.STABLE;
022
023import java.io.Serial;
024import java.util.concurrent.Semaphore;
025
026import org.apiguardian.api.API;
027import org.tquadrat.foundation.annotation.ClassVersion;
028import org.tquadrat.foundation.util.AutoSemaphore;
029
030/**
031 *  <p>{@summary The implementation of
032 *  {@link AutoSemaphore}.}</p>
033 *
034 *  @extauthor Thomas Thrien - thomas.thrien@tquadrat.org
035 *  @version $Id: AutoSemaphoreImpl.java 1135 2024-05-28 21:32:48Z tquadrat $
036 *  @since 0.4.8
037 *
038 *  @UMLGraph.link
039 */
040@ClassVersion( sourceVersion = "$Id: AutoSemaphoreImpl.java 1135 2024-05-28 21:32:48Z tquadrat $" )
041@API( status = STABLE, since = "0.4.8" )
042public final class AutoSemaphoreImpl extends Semaphore implements AutoSemaphore
043{
044        /*---------------*\
045    ====** Inner Classes **====================================================
046        \*---------------*/
047    /**
048     *  <p>{@summary The token that holds the permits to be released when a
049     *  {@code try-with-resources} block is left.}</p>
050     *
051     *  @extauthor Thomas Thrien - thomas.thrien@tquadrat.org
052     *  @version $Id: AutoSemaphoreImpl.java 1135 2024-05-28 21:32:48Z tquadrat $
053     *  @since 0.4.8
054     *
055     *  @UMLGraph.link
056     */
057    @SuppressWarnings( "NewClassNamingConvention" )
058    @ClassVersion( sourceVersion = "$Id: AutoSemaphoreImpl.java 1135 2024-05-28 21:32:48Z tquadrat $" )
059    @API( status = INTERNAL, since = "0.4.8" )
060    private final class Token implements AutoCloseable
061    {
062            /*------------*\
063        ====** Attributes **===================================================
064            \*------------*/
065        /**
066         *  The number of permits to release on close.
067         */
068        private final int m_Permits;
069
070            /*--------------*\
071        ====** Constructors **=================================================
072            \*--------------*/
073        /**
074         *  Creates a new instance of {@code Token}.
075         *
076         *  @param  permits The number of the acquired permits.
077         */
078        public Token( final int permits )
079        {
080            m_Permits = permits;
081        }   //  Token()
082
083            /*---------*\
084        ====** Methods **======================================================
085            \*---------*/
086        /**
087         *  {@inheritDoc}
088         */
089        @Override
090        public final void close() throws Exception
091        {
092            AutoSemaphoreImpl.this.release( m_Permits );
093        }   //  close()
094    }
095    //  class Token
096
097        /*------------------------*\
098    ====** Static Initialisations **===========================================
099        \*------------------------*/
100    /**
101     *  The serial version UID for objects of this class: {@value}.
102     *
103     *  @hidden
104     */
105    @Serial
106    private static final long serialVersionUID = 539879857L;
107
108        /*--------------*\
109    ====** Constructors **=====================================================
110        \*--------------*/
111    /**
112     *  Creates an {@code AutoSemaphoreImpl} instance with the given number of
113     *  permits and non-fair fairness setting.
114     *
115     *  @param  permits The initial number of permits available. This value may
116     *      be negative, in which case releases must occur before any acquires
117     *      will be granted.
118     */
119    public AutoSemaphoreImpl( final int permits )
120    {
121        super( permits );
122    }   //  AutoSemaphoreImpl()
123
124    /**
125     *  Creates an {@code AutoSemaphoreImpl} instance with the given number of
126     *  permits and the given fairness setting.
127     *
128     *  @param  permits The initial number of permits available. This value may
129     *      be negative, in which case releases must occur before any acquires
130     *      will be granted.
131     *  @param  fair    {@code true} if this semaphore will guarantee first-in
132     *      first-out granting of permits under contention, else {@code false}.
133     */
134    public AutoSemaphoreImpl( final int permits, final boolean fair )
135    {
136        super( permits, fair );
137    }   //  AutoSemaphoreImpl()
138
139        /*---------*\
140    ====** Methods **==========================================================
141        \*---------*/
142    /**
143     *  {@inheritDoc}
144     */
145    @Override
146    @SuppressWarnings( "ReturnOfInnerClass" )
147    public final AutoCloseable acquireToken( final int permits ) throws InterruptedException, IllegalArgumentException
148    {
149        acquire( permits );
150        final var retValue = new Token( permits );
151
152        //---* Done *----------------------------------------------------------
153        return retValue;
154    }   //  acquireToken()
155
156    /**
157     *  {@inheritDoc}
158     */
159    @Override
160    @SuppressWarnings( "ReturnOfInnerClass" )
161    public final AutoCloseable acquireTokenUninterruptibly( final int permits ) throws IllegalArgumentException
162    {
163        acquireUninterruptibly( permits );
164        final var retValue = new Token( permits );
165
166        //---* Done *----------------------------------------------------------
167        return retValue;
168    }   //  acquireTokenUninterruptibly()
169
170    /**
171     *  {@inheritDoc}
172     */
173    @Override
174    public final Semaphore getSemaphore() { return this; }
175}
176//  class AutoSemaphore
177
178/*
179 *  End of File
180 */