/* * Copyright (c) 2002-2010 LWJGL Project * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * 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. * * * Neither the name of 'LWJGL' nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 THE COPYRIGHT OWNER 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. */ package org.lwjgl.opencl; import org.lwjgl.BufferUtils; import org.lwjgl.LWJGLUtil; import org.lwjgl.PointerBuffer; import org.lwjgl.opencl.FastLongMap.Entry; import java.nio.*; import java.util.HashSet; import java.util.Set; import java.util.StringTokenizer; import static org.lwjgl.opencl.APPLEGLSharing.*; import static org.lwjgl.opencl.CL10.*; import static org.lwjgl.opencl.EXTDeviceFission.*; import static org.lwjgl.opencl.KHRGLSharing.*; /** * Utility class for OpenCL API calls. * * @author spasi */ final class APIUtil { private static final int INITIAL_BUFFER_SIZE = 256; private static final int INITIAL_LENGTHS_SIZE = 4; private static final int BUFFERS_SIZE = 32; private static final ThreadLocal arrayTL = new ThreadLocal() { protected char[] initialValue() { return new char[INITIAL_BUFFER_SIZE]; } }; private static final ThreadLocal bufferByteTL = new ThreadLocal() { protected ByteBuffer initialValue() { return BufferUtils.createByteBuffer(INITIAL_BUFFER_SIZE); } }; private static final ThreadLocal bufferPointerTL = new ThreadLocal() { protected PointerBuffer initialValue() { return BufferUtils.createPointerBuffer(INITIAL_BUFFER_SIZE); } }; private static final ThreadLocal lengthsTL = new ThreadLocal() { protected PointerBuffer initialValue() { return BufferUtils.createPointerBuffer(INITIAL_LENGTHS_SIZE); } }; private static final ThreadLocal buffersTL = new ThreadLocal() { protected Buffers initialValue() { return new Buffers(); } }; private APIUtil() { } private static char[] getArray(final int size) { char[] array = arrayTL.get(); if ( array.length < size ) { int sizeNew = array.length << 1; while ( sizeNew < size ) sizeNew <<= 1; array = new char[size]; arrayTL.set(array); } return array; } static ByteBuffer getBufferByte(final int size) { ByteBuffer buffer = bufferByteTL.get(); if ( buffer.capacity() < size ) { int sizeNew = buffer.capacity() << 1; while ( sizeNew < size ) sizeNew <<= 1; buffer = BufferUtils.createByteBuffer(size); bufferByteTL.set(buffer); } else buffer.clear(); return buffer; } private static ByteBuffer getBufferByteOffset(final int size) { ByteBuffer buffer = bufferByteTL.get(); if ( buffer.capacity() < size ) { int sizeNew = buffer.capacity() << 1; while ( sizeNew < size ) sizeNew <<= 1; final ByteBuffer bufferNew = BufferUtils.createByteBuffer(size); bufferNew.put(buffer); bufferByteTL.set(buffer = bufferNew); } else { buffer.position(buffer.limit()); buffer.limit(buffer.capacity()); } return buffer; } static PointerBuffer getBufferPointer(final int size) { PointerBuffer buffer = bufferPointerTL.get(); if ( buffer.capacity() < size ) { int sizeNew = buffer.capacity() << 1; while ( sizeNew < size ) sizeNew <<= 1; buffer = BufferUtils.createPointerBuffer(size); bufferPointerTL.set(buffer); } else buffer.clear(); return buffer; } static ShortBuffer getBufferShort() { return buffersTL.get().shorts; } static IntBuffer getBufferInt() { return buffersTL.get().ints; } static IntBuffer getBufferIntDebug() { return buffersTL.get().intsDebug; } static LongBuffer getBufferLong() { return buffersTL.get().longs; } static FloatBuffer getBufferFloat() { return buffersTL.get().floats; } static DoubleBuffer getBufferDouble() { return buffersTL.get().doubles; } static PointerBuffer getBufferPointer() { return buffersTL.get().pointers; } static PointerBuffer getLengths() { return getLengths(1); } static PointerBuffer getLengths(final int size) { PointerBuffer lengths = lengthsTL.get(); if ( lengths.capacity() < size ) { int sizeNew = lengths.capacity(); while ( sizeNew < size ) sizeNew <<= 1; lengths = BufferUtils.createPointerBuffer(size); lengthsTL.set(lengths); } else lengths.clear(); return lengths; } /** * Simple ASCII encoding. * * @param buffer The target buffer * @param string The source string */ private static ByteBuffer encode(final ByteBuffer buffer, final CharSequence string) { for ( int i = 0; i < string.length(); i++ ) { final char c = string.charAt(i); if ( LWJGLUtil.DEBUG && 0x80 <= c ) // Silently ignore and map to 0x1A. buffer.put((byte)0x1A); else buffer.put((byte)c); } return buffer; } /** * Reads a byte string from the specified buffer. * * @param buffer * * @return the buffer as a String. */ static String getString(final ByteBuffer buffer) { final int length = buffer.remaining(); final char[] charArray = getArray(length); for ( int i = buffer.position(); i < buffer.limit(); i++ ) charArray[i - buffer.position()] = (char)buffer.get(i); return new String(charArray, 0, length); } /** * Returns a buffer containing the specified string as bytes. * * @param string * * @return the String as a ByteBuffer */ static ByteBuffer getBuffer(final CharSequence string) { final ByteBuffer buffer = encode(getBufferByte(string.length()), string); buffer.flip(); return buffer; } /** * Returns a buffer containing the specified string as bytes, starting at the specified offset. * * @param string * * @return the String as a ByteBuffer */ static ByteBuffer getBuffer(final CharSequence string, final int offset) { final ByteBuffer buffer = encode(getBufferByteOffset(offset + string.length()), string); buffer.flip(); return buffer; } /** * Returns a buffer containing the specified string as bytes, including null-termination. * * @param string * * @return the String as a ByteBuffer */ static ByteBuffer getBufferNT(final CharSequence string) { final ByteBuffer buffer = encode(getBufferByte(string.length() + 1), string); buffer.put((byte)0); buffer.flip(); return buffer; } static int getTotalLength(final CharSequence[] strings) { int length = 0; for ( CharSequence string : strings ) length += string.length(); return length; } /** * Returns a buffer containing the specified strings as bytes. * * @param strings * * @return the Strings as a ByteBuffer */ static ByteBuffer getBuffer(final CharSequence[] strings) { final ByteBuffer buffer = getBufferByte(getTotalLength(strings)); for ( CharSequence string : strings ) encode(buffer, string); buffer.flip(); return buffer; } /** * Returns a buffer containing the specified strings as bytes, including null-termination. * * @param strings * * @return the Strings as a ByteBuffer */ static ByteBuffer getBufferNT(final CharSequence[] strings) { final ByteBuffer buffer = getBufferByte(getTotalLength(strings) + strings.length); for ( CharSequence string : strings ) { encode(buffer, string); buffer.put((byte)0); } buffer.flip(); return buffer; } /** * Returns a buffer containing the lengths of the specified strings. * * @param strings * * @return the String lengths in a PointerBuffer */ static PointerBuffer getLengths(final CharSequence[] strings) { PointerBuffer buffer = getLengths(strings.length); for ( CharSequence string : strings ) buffer.put(string.length()); buffer.flip(); return buffer; } /** * Returns a buffer containing the lengths of the specified buffers. * * @param buffers the buffer array * * @return the buffer lengths in a PointerBuffer */ static PointerBuffer getLengths(final ByteBuffer[] buffers) { PointerBuffer lengths = getLengths(buffers.length); for ( ByteBuffer buffer : buffers ) lengths.put(buffer.remaining()); lengths.flip(); return lengths; } static int getSize(final PointerBuffer lengths) { long size = 0; for ( int i = lengths.position(); i < lengths.limit(); i++ ) size += lengths.get(i); return (int)size; } private static class Buffers { final ShortBuffer shorts; final IntBuffer ints; final IntBuffer intsDebug; final LongBuffer longs; final FloatBuffer floats; final DoubleBuffer doubles; final PointerBuffer pointers; Buffers() { shorts = BufferUtils.createShortBuffer(BUFFERS_SIZE); ints = BufferUtils.createIntBuffer(BUFFERS_SIZE); intsDebug = BufferUtils.createIntBuffer(1); longs = BufferUtils.createLongBuffer(BUFFERS_SIZE); floats = BufferUtils.createFloatBuffer(BUFFERS_SIZE); doubles = BufferUtils.createDoubleBuffer(BUFFERS_SIZE); pointers = BufferUtils.createPointerBuffer(BUFFERS_SIZE); } } /* ------------------------------------------------------------------------ --------------------------------------------------------------------------- OPENCL API UTILITIES BELOW --------------------------------------------------------------------------- ------------------------------------------------------------------------ */ static Set getExtensions(final String extensionList) { final Set extensions = new HashSet(); if ( extensionList != null ) { final StringTokenizer tokenizer = new StringTokenizer(extensionList); while ( tokenizer.hasMoreTokens() ) extensions.add(tokenizer.nextToken()); } return extensions; } static boolean isDevicesParam(final int param_name) { switch ( param_name ) { case CL_CONTEXT_DEVICES: case CL_CURRENT_DEVICE_FOR_GL_CONTEXT_KHR: case CL_DEVICES_FOR_GL_CONTEXT_KHR: case CL_CGL_DEVICE_FOR_CURRENT_VIRTUAL_SCREEN_APPLE: case CL_CGL_DEVICES_FOR_SUPPORTED_VIRTUAL_SCREENS_APPLE: return true; } return false; } static CLPlatform getCLPlatform(final PointerBuffer properties) { long platformID = 0; final int keys = properties.remaining() / 2; for ( int k = 0; k < keys; k++ ) { final long key = properties.get(k << 1); if ( key == 0 ) break; if ( key == CL_CONTEXT_PLATFORM ) { platformID = properties.get((k << 1) + 1); break; } } if ( platformID == 0 ) throw new IllegalArgumentException("Could not find CL_CONTEXT_PLATFORM in cl_context_properties."); final CLPlatform platform = CLPlatform.getCLPlatform(platformID); if ( platform == null ) throw new IllegalStateException("Could not find a valid CLPlatform. Make sure clGetPlatformIDs has been used before."); return platform; } static ByteBuffer getNativeKernelArgs(final long user_func_ref, final CLMem[] clMems, final long[] sizes) { final ByteBuffer args = getBufferByte(8 + 4 + (clMems == null ? 0 : clMems.length * (4 + PointerBuffer.getPointerSize()))); args.putLong(0, user_func_ref); if ( clMems == null ) args.putInt(8, 0); else { args.putInt(8, clMems.length); int byteIndex = 12; for ( int i = 0; i < clMems.length; i++ ) { if ( LWJGLUtil.DEBUG && !clMems[i].isValid() ) throw new IllegalArgumentException("An invalid CLMem object was specified."); args.putInt(byteIndex, (int)sizes[i]); // CLMem size byteIndex += (4 + PointerBuffer.getPointerSize()); // Skip size and make room for the pointer } } return args; } // ------------------------------------------------------------------------------------ /** * Releases all sub-devices created from the specified CLDevice. * * @param device the CLDevice to clear */ static void releaseObjects(final CLDevice device) { // Release objects only if we're about to hit 0. if ( device.getReferenceCount() > 1 ) return; releaseObjects(device.getSubCLDeviceRegistry(), DESTRUCTOR_CLSubDevice); } /** * Releases all objects contained in the specified CLContext. * * @param context the CLContext to clear */ static void releaseObjects(final CLContext context) { // Release objects only if we're about to hit 0. if ( context.getReferenceCount() > 1 ) return; releaseObjects(context.getCLEventRegistry(), DESTRUCTOR_CLEvent); releaseObjects(context.getCLProgramRegistry(), DESTRUCTOR_CLProgram); releaseObjects(context.getCLSamplerRegistry(), DESTRUCTOR_CLSampler); releaseObjects(context.getCLMemRegistry(), DESTRUCTOR_CLMem); releaseObjects(context.getCLCommandQueueRegistry(), DESTRUCTOR_CLCommandQueue); } /** * Releases all objects contained in the specified CLProgram. * * @param program the CLProgram to clear */ static void releaseObjects(final CLProgram program) { // Release objects only if we're about to hit 0. if ( program.getReferenceCount() > 1 ) return; releaseObjects(program.getCLKernelRegistry(), DESTRUCTOR_CLKernel); } /** * Releases all objects contained in the specified CLCommandQueue. * * @param queue the CLCommandQueue to clear */ static void releaseObjects(final CLCommandQueue queue) { // Release objects only if we're about to hit 0. if ( queue.getReferenceCount() > 1 ) return; releaseObjects(queue.getCLEventRegistry(), DESTRUCTOR_CLEvent); } private static void releaseObjects(final CLObjectRegistry registry, final ObjectDestructor destructor) { if ( registry.isEmpty() ) return; for ( Entry entry : registry.getAll() ) { final T object = entry.value; while ( object.isValid() ) destructor.release(object); } } private static final ObjectDestructor DESTRUCTOR_CLSubDevice = new ObjectDestructor() { public void release(final CLDevice object) { clReleaseDeviceEXT(object); } }; private static final ObjectDestructor DESTRUCTOR_CLMem = new ObjectDestructor() { public void release(final CLMem object) { clReleaseMemObject(object); } }; private static final ObjectDestructor DESTRUCTOR_CLCommandQueue = new ObjectDestructor() { public void release(final CLCommandQueue object) { clReleaseCommandQueue(object); } }; private static final ObjectDestructor DESTRUCTOR_CLSampler = new ObjectDestructor() { public void release(final CLSampler object) { clReleaseSampler(object); } }; private static final ObjectDestructor DESTRUCTOR_CLProgram = new ObjectDestructor() { public void release(final CLProgram object) { clReleaseProgram(object); } }; private static final ObjectDestructor DESTRUCTOR_CLKernel = new ObjectDestructor() { public void release(final CLKernel object) { clReleaseKernel(object); } }; private static final ObjectDestructor DESTRUCTOR_CLEvent = new ObjectDestructor() { public void release(final CLEvent object) { clReleaseEvent(object); } }; private interface ObjectDestructor { void release(T object); } }