/** * Copyright 2010-2023 JogAmp Community. All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are * permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list * of conditions and the following disclaimer in the documentation and/or other materials * provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY JogAmp Community ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JogAmp Community OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * The views and conclusions contained in the software and documentation are those of the * authors and should not be interpreted as representing official policies, either expressed * or implied, of JogAmp Community. */ /* * Created on Saturday, March 27 2010 11:55 */ package com.jogamp.common.nio; import java.nio.Buffer; import java.nio.ByteBuffer; import com.jogamp.common.os.Platform; /** * @author Sven Gothel * @author Michael Bien */ @SuppressWarnings("rawtypes") public abstract class AbstractBuffer implements NativeBuffer { /** Platform dependent pointer size in bytes, i.e. 32bit or 64bit wide, depending of the CPU pointer width. */ public static final int POINTER_SIZE; protected final Buffer buffer; protected final int elementSize; protected final int capacity; protected int limit; protected int position; static { Platform.initSingleton(); // loads native gluegen_rt library POINTER_SIZE = Platform.is32Bit() ? Buffers.SIZEOF_INT : Buffers.SIZEOF_LONG ; } /** * capacity and elementSize should be match the equation w/ target buffer type *
     *    capacity = elementSizeInBytes(buffer) * buffer.capacity() ) / elementSize
     * 
* @param buffer shall be in target format. * @param elementSize the target element size in bytes. * @param capacity the target capacity in elements of size elementSize. */ protected AbstractBuffer(final Buffer buffer, final int elementSize, final int capacity) { this.buffer = buffer; this.elementSize = elementSize; this.capacity = capacity; this.limit = capacity; this.position = 0; } @Override public final int elementSize() { return elementSize; } @Override public final int limit() { return limit; } @SuppressWarnings("unchecked") @Override public final B limit(final int newLim) { if (0 > newLim || newLim >= capacity) { throw new IllegalArgumentException("New limit "+newLim+" out of bounds [0 .. capacity " +capacity()+"]."); } limit = newLim; return (B)this; } @Override public final int capacity() { return capacity; } @Override public final int position() { return position; } @SuppressWarnings("unchecked") @Override public final B position(final int newPos) { if (0 > newPos || newPos > limit) { throw new IllegalArgumentException("New position "+newPos+" out of bounds [0 .. limit " +limit()+"]."); } position = newPos; return (B)this; } @Override public final int remaining() { return limit - position; } @Override public final boolean hasRemaining() { return limit > position; } @SuppressWarnings("unchecked") @Override public final B clear() { limit = capacity; position = 0; return (B) this; } @SuppressWarnings("unchecked") @Override public final B flip() { limit = position; position = 0; return (B) this; } @SuppressWarnings("unchecked") @Override public final B rewind() { position = 0; return (B) this; } @Override public final Buffer getBuffer() { return buffer; } @Override public final boolean isDirect() { return buffer.isDirect(); } @Override public long getDirectBufferAddress() { if( isDirect() ) { return Buffers.getDirectBufferAddressImpl(buffer); } else { return 0; } } @Override public void storeDirectAddress(final ByteBuffer directDest) { final long addr = getDirectBufferAddress(); switch(POINTER_SIZE) { case 4: directDest.putInt(0, (int) ( addr & 0x00000000FFFFFFFFL ) ); break; case 8: directDest.putLong(0, addr); break; } directDest.position(directDest.position()+POINTER_SIZE); } @Override public void storeDirectAddress(final ByteBuffer directDest, final int destBytePos) { final long addr = getDirectBufferAddress(); switch(POINTER_SIZE) { case 4: directDest.putInt(destBytePos, (int) ( addr & 0x00000000FFFFFFFFL ) ); break; case 8: directDest.putLong(destBytePos, addr); break; } } @Override public final boolean hasArray() { return buffer.hasArray(); } @Override public final int arrayOffset() { if( hasArray() ) { return buffer.arrayOffset(); } else { return 0; } } @Override public Object array() throws UnsupportedOperationException { return buffer.array(); } protected String toSubString() { return "[direct["+isDirect()+", addr 0x"+Long.toHexString(getDirectBufferAddress())+"], hasArray "+hasArray()+", capacity "+capacity+", position "+position+", elementSize "+elementSize+", buffer[capacity "+buffer.capacity()+", lim "+buffer.limit()+", pos "+buffer.position()+"]]"; } @Override public String toString() { return "AbstractBuffer"+toSubString(); } }