blob: 3965f7928e6aa143e53c4f91e10cea16a6ae6aa0 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more contributor license
* agreements. See the NOTICE file distributed with this work for additional information regarding
* copyright ownership. The ASF licenses this file to You 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.apache.geode.cache.client.internal.pooling;
import java.util.concurrent.atomic.AtomicInteger;
/**
* Responsible for counting connections.
* The count maintained by this class will eventually be consistent with the actual number of
* connections. Since the count is changed before and after the actual connections are created and
* destroyed, and not changed while holding a lock, the count should be treated as an estimate of
* the current number of connections.
*/
public class ConnectionAccounting {
private final int minimum;
private final int maximum;
private final AtomicInteger count = new AtomicInteger();
public ConnectionAccounting(int min, int max) {
this.minimum = min;
this.maximum = max;
}
public int getMinimum() {
return minimum;
}
public int getMaximum() {
return maximum;
}
public int getCount() {
return count.get();
}
/**
* Should be called when prefilling connections to reach minimum connections. Caller should only
* create a connection if this method returns {@code true}. If connection creation fails then
* {@link #cancelTryPrefill} must be called to revert the count increase.
*
* @return {@code true} if count was under minimum and we increased it, otherwise {@code false}.
*/
public boolean tryPrefill() {
return tryReserve(minimum);
}
/**
* Should only be called if connection creation failed after calling {@link #tryPrefill()} ()}.
*/
public void cancelTryPrefill() {
count.getAndDecrement();
}
/**
* Should be called when a new connection would be nice to have when count is under maximum.
* Caller should only create a connection if this method returns {@code true}. If connection
* creation fails then {@link #cancelTryCreate} must be called to revert the count increase.
*
* @return {@code true} if count was under maximum and we increased it, otherwise {@code false}.
*/
public boolean tryCreate() {
return tryReserve(maximum);
}
/**
* Should only be called if connection creation failed after calling {@link #tryCreate()}.
*/
public void cancelTryCreate() {
count.decrementAndGet();
}
/**
* Count a created connection regardless of maximum. Should not be called after
* {@link #tryCreate()}.
*/
public void create() {
count.getAndIncrement();
}
/**
* Should be called when a connection is being returned and the caller should destroy the
* connection if {@code true} is returned. If connection destroy fails then
* {@link #cancelTryDestroy()} must be called.
*
* @return {@code true} if count was over maximum and we decreased it, otherwise {@code false}.
*/
public boolean tryDestroy() {
int currentCount;
while ((currentCount = count.get()) > maximum) {
if (count.compareAndSet(currentCount, currentCount - 1)) {
return true;
}
}
return false;
}
/**
* Should only be called if connection destroy failed after calling {@link #tryDestroy()}.
*/
public void cancelTryDestroy() {
count.getAndIncrement();
}
/**
* Should be called after any connection destroys are done. Should not be called
* after {@link #tryDestroy()}.
*
* @param destroyCount number of connections being destroyed.
*
* @return {@code true} if after decreasing count it is under the minimum, otherwise
* {@code false}.
*/
public boolean destroyAndIsUnderMinimum(int destroyCount) {
int newCount = count.addAndGet(-destroyCount);
return newCount < minimum;
}
public boolean isUnderMinimum() {
return count.get() < minimum;
}
public boolean isOverMinimum() {
return count.get() > minimum;
}
private boolean tryReserve(int upperBound) {
int currentCount;
while ((currentCount = count.get()) < upperBound) {
if (count.compareAndSet(currentCount, currentCount + 1)) {
return true;
}
}
return false;
}
}