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

org.jsl.collider.RetainableDataBlockCache Maven / Gradle / Ivy

There is a newer version: 0.2.5
Show newest version
/*
 * Copyright (C) 2015 Sergey Zubarev, [email protected]
 *
 * This file is a part of JS-Collider framework.
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as
 * published by the Free Software Foundation, either version 3 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program.  If not, see .
 */
package org.jsl.collider;

import java.nio.ByteBuffer;
import java.util.concurrent.locks.ReentrantLock;
import java.util.logging.Level;
import java.util.logging.Logger;

public class RetainableDataBlockCache
{
    private final boolean m_useDirectBuffers;
    private final int m_blockSize;
    private final int m_maxSize;
    private final ReentrantLock m_lock;
    private RetainableDataBlock m_dataBlock;
    private int m_size;
    private int m_gets;
    private int m_puts;

    private static class DataBlockImpl extends RetainableDataBlock
    {
        private final RetainableDataBlockCache m_cache;

        public DataBlockImpl( ByteBuffer byteBuffer, RetainableDataBlockCache cache )
        {
            super( byteBuffer );
            m_cache = cache;
        }

        protected void finalRelease()
        {
            reinit();
            m_cache.put( this );
        }
    }

    private RetainableDataBlock createDataBlock()
    {
        final ByteBuffer byteBuffer =
                m_useDirectBuffers ? ByteBuffer.allocateDirect( m_blockSize )
                                   : ByteBuffer.allocate( m_blockSize );
        return new DataBlockImpl( byteBuffer, this );
    }

    private void put( RetainableDataBlock dataBlock )
    {
        m_lock.lock();
        try
        {
            m_puts++;
            if (m_size < m_maxSize)
            {
                dataBlock.next = m_dataBlock;
                m_dataBlock = dataBlock;
                m_size++;
            }
        }
        finally
        {
            m_lock.unlock();
        }
    }

    public RetainableDataBlockCache( boolean useDirectBuffers, int blockSize, int initialSize, int maxSize )
    {
        m_useDirectBuffers = useDirectBuffers;
        m_blockSize = blockSize;
        m_maxSize = maxSize;
        m_lock = new ReentrantLock();
        m_dataBlock = null;
        m_size = initialSize;

        for (int idx=0; idx= 0 );

        RetainableDataBlock ret = null;
        RetainableDataBlock dataBlock = null;

        m_lock.lock();
        try
        {
            m_gets += cnt;
            if (m_dataBlock != null)
            {
                ret = m_dataBlock;
                dataBlock = m_dataBlock;
                for (;;)
                {
                    m_size--;
                    if (--cnt == 0)
                    {
                        m_dataBlock = dataBlock.next;
                        dataBlock.next = null;
                        return ret;
                    }
                    if (dataBlock.next == null)
                        break;
                    dataBlock = dataBlock.next;
                }
                m_dataBlock = null;
            }
        }
        finally
        {
            m_lock.unlock();
        }

        if (ret == null)
        {
            ret = createDataBlock();
            if (--cnt == 0)
                return ret;
            dataBlock = ret;
        }

        for (; cnt>0; cnt--)
        {
            dataBlock.next = createDataBlock();
            dataBlock = dataBlock.next;
        }

        return ret;
    }

    public final RetainableDataBlock getByDataSize( int dataSize )
    {
        int blocks = (dataSize / m_blockSize);
        if ((dataSize % m_blockSize) > 0)
            blocks++;
        return get( blocks );
    }

    public final void clear( Logger logger )
    {
        int size = 0;
        while (m_dataBlock != null)
        {
            final RetainableDataBlock next = m_dataBlock.next;
            m_dataBlock.next = null;
            m_dataBlock = next;
            size++;
        }

        if (size != m_size)
        {
            if (logger.isLoggable(Level.WARNING))
            {
                logger.warning(
                        "[" + m_blockSize + "] internal error: real size " +
                        size + " != " + m_size + "." );
            }
        }

        if (size > m_maxSize)
        {
            if (logger.isLoggable(Level.WARNING))
            {
                logger.warning(
                        "[" + m_blockSize + "] internal error: current size " +
                        size + " is greater than maximum size (" + m_maxSize + ")." );
            }
        }

        if (m_gets != m_puts)
        {
            if (logger.isLoggable(Level.WARNING))
            {
                logger.warning(
                        "[" + m_blockSize + "] resource leak detected: gets=" +
                        m_gets + ", puts=" + m_puts + "." );
            }
        }

        if (logger.isLoggable(Level.FINE))
            logger.fine( "[" + m_blockSize + "] size=" + size + ", gets=" + m_gets + "." );

        m_dataBlock = null;
        m_size = 0;
        m_gets = 0;
        m_puts = 0;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy