blob: 2215c7bb3dc17ab7d05731e80fb7801ac7176aac [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.qpid.server.flow;
import org.apache.log4j.Logger;
public class WindowCreditManager extends AbstractFlowCreditManager implements FlowCreditManager_0_10
{
private static final Logger LOGGER = Logger.getLogger(WindowCreditManager.class);
private volatile long _bytesCreditLimit;
private volatile long _messageCreditLimit;
private volatile long _bytesUsed;
private volatile long _messageUsed;
public WindowCreditManager()
{
this(0L, 0L);
}
public WindowCreditManager(long bytesCreditLimit, long messageCreditLimit)
{
_bytesCreditLimit = bytesCreditLimit;
_messageCreditLimit = messageCreditLimit;
setSuspended(!hasCredit());
}
public long getBytesCreditLimit()
{
return _bytesCreditLimit;
}
public long getMessageCreditLimit()
{
return _messageCreditLimit;
}
public synchronized void setCreditLimits(final long bytesCreditLimit, final long messageCreditLimit)
{
_bytesCreditLimit = bytesCreditLimit;
_messageCreditLimit = messageCreditLimit;
setSuspended(!hasCredit());
}
public long getMessageCredit()
{
return _messageCreditLimit == -1L
? Long.MAX_VALUE
: _messageUsed < _messageCreditLimit ? _messageCreditLimit - _messageUsed : 0L;
}
public long getBytesCredit()
{
return _bytesCreditLimit == -1L
? Long.MAX_VALUE
: _bytesUsed < _bytesCreditLimit ? _bytesCreditLimit - _bytesUsed : 0L;
}
public synchronized void restoreCredit(final long messageCredit, final long bytesCredit)
{
_messageUsed -= messageCredit;
if(_messageUsed < 0L)
{
LOGGER.error("Message credit used value was negative: "+ _messageUsed);
_messageUsed = 0;
}
boolean notifyIncrease = true;
if(_messageCreditLimit > 0L)
{
notifyIncrease = (_messageUsed != _messageCreditLimit);
}
_bytesUsed -= bytesCredit;
if(_bytesUsed < 0L)
{
LOGGER.error("Bytes credit used value was negative: "+ _messageUsed);
_bytesUsed = 0;
}
if(_bytesCreditLimit > 0L)
{
notifyIncrease = notifyIncrease && bytesCredit>0;
if(notifyIncrease)
{
notifyIncreaseBytesCredit();
}
}
setSuspended(!hasCredit());
}
public synchronized boolean hasCredit()
{
return (_bytesCreditLimit < 0L || _bytesCreditLimit > _bytesUsed)
&& (_messageCreditLimit < 0L || _messageCreditLimit > _messageUsed);
}
public synchronized boolean useCreditForMessage(final long msgSize)
{
if(_messageCreditLimit >= 0L)
{
if(_messageUsed < _messageCreditLimit)
{
if(_bytesCreditLimit < 0L)
{
_messageUsed++;
return true;
}
else if(_bytesUsed + msgSize <= _bytesCreditLimit)
{
_messageUsed++;
_bytesUsed += msgSize;
return true;
}
else
{
return false;
}
}
else
{
setSuspended(true);
return false;
}
}
else if(_bytesCreditLimit >= 0L)
{
if(_bytesUsed + msgSize <= _bytesCreditLimit)
{
_bytesUsed += msgSize;
return true;
}
else
{
return false;
}
}
else
{
return true;
}
}
public synchronized void addCredit(long count, long bytes)
{
if(bytes > 0)
{
_bytesCreditLimit += bytes;
}
else if(bytes == -1)
{
_bytesCreditLimit = -1;
}
if(count > 0)
{
_messageCreditLimit += count;
}
else if(count == -1)
{
_messageCreditLimit = -1;
}
}
public void clearCredit()
{
_bytesCreditLimit = 0l;
_messageCreditLimit = 0l;
setSuspended(true);
}
}