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 */