Package net.sf.cindy.buffer

Source Code of net.sf.cindy.buffer.LinkedBuffer

/*
* Copyright 2004-2006 the original author or authors.
*
* Licensed 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 net.sf.cindy.buffer;

import java.nio.ByteBuffer;

import net.sf.cindy.Buffer;
import net.sf.cindy.util.Charset;

/**
* Linked several <code>Buffer</code> as a single <code>Buffer</code>.
* Experiment.
*
* @author <a href="chenrui@gmail.com">Roger Chen</a>
* @version $id$
*/
public class LinkedBuffer extends AbstractBuffer {

    protected static final class Entry {

        public Buffer buffer;
        public int position;

        public Entry next;
        public Entry previous;

    }

    protected final Entry header;

    public LinkedBuffer(Buffer[] content) {
        super(0, 0);
        header = new Entry();
        header.previous = header.next = header;
        for (int i = 0; i < content.length; i++) {
            append(content[i].duplicate());
        }
        limit(capacity());
    }

    private LinkedBuffer(Entry header, int offset, int capacity) {
        super(offset, capacity);
        this.header = header;
    }

    private class DelegateReleaseBuffer extends LinkedBuffer {

        public DelegateReleaseBuffer(int offset, int capacity) {
            super(LinkedBuffer.this.header, offset, capacity);
        }

        public boolean isReleased() {
            return LinkedBuffer.this.isReleased();
        }

        public void release() {
            LinkedBuffer.this.release();
        }

        public boolean isPermanent() {
            return LinkedBuffer.this.isPermanent();
        }

        public void setPermanent(boolean b) {
            LinkedBuffer.this.setPermanent(b);
        }

    }

    protected void append(Buffer buffer) {
        if (buffer != null) {
            Entry entry = new Entry();
            Entry last = header.previous;

            last.next = entry;
            entry.previous = last;
            entry.next = header;
            header.previous = entry;

            entry.buffer = buffer;
            entry.position = (last == header ? 0 : last.buffer.remaining())
                    + last.position;

            // only effect current buffer
            capacity(capacity() + buffer.remaining());
        }
    }

    protected void remove(Buffer buffer) {
        for (Entry e = header.next; e != header; e = e.next) {
            if (e.buffer == buffer) {
                Entry removedEntry = e;
                int removedSize = e.buffer.remaining();

                e.previous.next = e.next;
                e.next.previous = e.previous;
                while ((e = e.next) != header) {
                    e.position -= removedSize;
                }

                removedEntry.previous = null;
                removedEntry.next = null;
                removedEntry.buffer = null;

                // only effect current buffer
                capacity(capacity() - removedSize);
                limit(capacity());
                position(0);
                return;
            }
        }
    }

    public boolean isDirect() {
        for (Entry e = header.next; e != header; e = e.next) {
            if (!e.buffer.isDirect())
                return false;
        }
        return header.next != header; // is not empty
    }

    public Buffer duplicate() {
        LinkedBuffer buffer = new DelegateReleaseBuffer(getIndex(0, 0),
                capacity());
        buffer.limit(limit()).position(position());
        buffer.mark(getMark());
        buffer.setReadonly(isReadonly());
        return buffer;
    }

    public Buffer slice() {
        return new DelegateReleaseBuffer(getIndex(0), remaining())
                .setReadonly(isReadonly());
    }

    protected void _release() {
        for (Entry e = header.next; e != header;) {
            e.buffer.release();
            e.buffer = null;
            e.previous = null;
            e = e.next;
            e.previous.next = null;
        }
        header.next = header.previous = header;
    }

    private Entry getEntry(int index) {
        for (Entry e = header.previous; e != header; e = e.previous) {
            if (index >= e.position)
                return e;
        }
        return null;
    }

    protected byte _get(int index) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;
        return buffer.get(buffer.position() + index - entry.position);
    }

    protected void _put(int index, byte b) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;
        buffer.put(buffer.position() + index - entry.position, b);
    }

    protected Buffer batch(boolean get, int index, byte[] array, int offset,
            int length) {
        Entry entry = getEntry(index);
        int off = index - entry.position;
        do {
            Buffer buffer = entry.buffer;
            int len = Math.min(buffer.remaining() - off, length);

            if (get)
                buffer.get(buffer.position() + off, array, offset, len);
            else
                buffer.put(buffer.position() + off, array, offset, len);

            offset += len;
            length -= len;
            if (length <= 0)
                break;
            off = 0;
        } while ((entry = entry.next) != header);
        return this;
    }

    protected Buffer batch(boolean get, int index, ByteBuffer buffer, int length) {
        Entry entry = getEntry(index);
        int off = index - entry.position;
        do {
            Buffer content = entry.buffer;
            int len = Math.min(content.remaining() - off, length);

            if (get)
                content.get(content.position() + off, buffer, len);
            else {
                content.put(content.position() + off, buffer, len);
            }

            length -= len;
            if (length <= 0)
                break;
            off = 0;
        } while ((entry = entry.next) != header);
        return this;
    }

    protected Buffer batch(boolean get, int index, Buffer buffer, int length) {
        Entry entry = getEntry(index);
        int off = index - entry.position;
        do {
            Buffer content = entry.buffer;
            int len = Math.min(content.remaining() - off, length);

            if (get)
                content.get(content.position() + off, buffer, len);
            else
                content.put(content.position() + off, buffer, len);

            length -= len;
            if (length <= 0)
                break;
            off = 0;
        } while ((entry = entry.next) != header);
        return this;
    }

    public ByteBuffer asByteBuffer() {
        ByteBuffer buffer = ByteBuffer.allocate(capacity());
        batch(true, getIndex(0, 0), buffer, buffer.capacity());
        buffer.position(position());
        buffer.limit(limit());
        return buffer;
    }

    public Buffer compact() {
        checkReadonly();

        Buffer buffer = BufferFactory.allocate(remaining());
        batch(true, getIndex(0), buffer, buffer.capacity());
        buffer.position(0);

        int index = getIndex(0, 0);
        Entry entry = getEntry(index);
        int offset = index - entry.position;

        do {
            Buffer content = entry.buffer;
            buffer.limit(Math.min(buffer.capacity(), buffer.position()
                    + content.remaining() - offset));
            content.put(content.position() + offset, buffer);
            if (buffer.position() == buffer.capacity())
                break;
            offset = 0;
        } while ((entry = entry.next) != header);

        buffer.release();

        position(remaining());
        limit(capacity());
        return this;
    }

    public Buffer get(byte[] dst, int offset, int length) {
        checkBounds(offset, length, dst.length);
        return batch(true, getIndex(length), dst, offset, length);
    }

    public Buffer get(int index, byte[] dst, int offset, int length) {
        checkBounds(offset, length, dst.length);
        return batch(true, getIndex(index, length), dst, offset, length);
    }

    public Buffer get(ByteBuffer dst, int length) {
        checkBounds(0, length, dst.remaining());
        return batch(true, getIndex(length), dst, length);
    }

    public Buffer get(int index, ByteBuffer dst, int length) {
        checkBounds(0, length, dst.remaining());
        return batch(true, getIndex(index, length), dst, length);
    }

    public Buffer get(Buffer dst, int length) {
        checkBounds(0, length, dst.remaining());
        return batch(true, getIndex(length), dst, length);
    }

    public Buffer get(int index, Buffer dst, int length) {
        checkBounds(0, length, dst.remaining());
        return batch(true, getIndex(index, length), dst, length);
    }

    public Buffer put(byte[] src, int offset, int length) {
        checkBounds(offset, length, src.length);
        return batch(false, putIndex(length), src, offset, length);
    }

    public Buffer put(int index, byte[] src, int offset, int length) {
        checkBounds(offset, length, src.length);
        return batch(false, putIndex(index, length), src, offset, length);
    }

    public Buffer put(ByteBuffer src, int length) {
        checkBounds(0, length, src.remaining());
        return batch(false, putIndex(length), src, length);
    }

    public Buffer put(int index, ByteBuffer src, int length) {
        checkBounds(0, length, src.remaining());
        return batch(false, putIndex(index, length), src, length);
    }

    public Buffer put(Buffer src, int length) {
        checkBounds(0, length, src.remaining());
        return batch(false, putIndex(length), src, length);
    }

    public Buffer put(int index, Buffer src, int length) {
        checkBounds(0, length, src.remaining());
        return batch(false, putIndex(index, length), src, length);
    }

    private short decodeShort(int index) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 2)
            return buffer.getShort(idx);
        return Bits.decodeShort(this, index);
    }

    private Buffer encodeShort(int index, short s) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 2)
            buffer.putShort(idx, s);
        else
            Bits.encodeShort(this, index, s);
        return this;
    }

    public short getShort() {
        return decodeShort(getIndex(2));
    }

    public short getShort(int index) {
        return decodeShort(getIndex(index, 2));
    }

    public Buffer putShort(short s) {
        return encodeShort(putIndex(2), s);
    }

    public Buffer putShort(int index, short s) {
        return encodeShort(putIndex(index, 2), s);
    }

    private int decodeInt(int index) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 4)
            return buffer.getInt(idx);
        return Bits.decodeInt(this, index);
    }

    private Buffer encodeInt(int index, int i) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 4)
            buffer.putInt(idx, i);
        else
            Bits.encodeInt(this, index, i);
        return this;
    }

    public int getInt() {
        return decodeInt(getIndex(4));
    }

    public int getInt(int index) {
        return decodeInt(getIndex(index, 4));
    }

    public Buffer putInt(int i) {
        return encodeInt(putIndex(4), i);
    }

    public Buffer putInt(int index, int i) {
        return encodeInt(putIndex(index, 4), i);
    }

    private long decodeLong(int index) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 8)
            return buffer.getLong(idx);
        return Bits.decodeLong(this, index);
    }

    private Buffer encodeLong(int index, long l) {
        Entry entry = getEntry(index);
        Buffer buffer = entry.buffer;

        int idx = buffer.position() + index - entry.position;
        if (buffer.limit() - idx >= 8)
            buffer.putLong(idx, l);
        else
            Bits.encodeLong(this, index, l);
        return this;
    }

    public long getLong() {
        return decodeLong(getIndex(8));
    }

    public long getLong(int index) {
        return decodeLong(getIndex(index, 8));
    }

    public Buffer putLong(long l) {
        return encodeLong(putIndex(8), l);
    }

    public Buffer putLong(int index, long l) {
        return encodeLong(putIndex(index, 8), l);
    }

    private String _getString(int i, Charset charset, int bufferLen) {
        Buffer buffer = BufferFactory.allocate(bufferLen);
        batch(true, i, buffer, bufferLen);
        buffer.position(0);
        try {
            return charset.decode(buffer.asByteBuffer());
        } finally {
            buffer.release();
        }
    }

    public String getString(Charset charset, int bufferLen) {
        return _getString(getIndex(bufferLen), charset, bufferLen);
    }

    public String getString(int index, Charset charset, int bufferLen) {
        return _getString(getIndex(index, bufferLen), charset, bufferLen);
    }
}
TOP

Related Classes of net.sf.cindy.buffer.LinkedBuffer

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.