Signed-off-by: Kai Boernert <kai-boernert@visiongamestudios.de>define_list_fix
parent
201010b6f0
commit
848a9217d0
@ -0,0 +1,14 @@ |
|||||||
|
package com.jme3.util; |
||||||
|
|
||||||
|
import java.lang.reflect.Method; |
||||||
|
import java.nio.Buffer; |
||||||
|
import java.nio.ByteBuffer; |
||||||
|
import java.nio.DoubleBuffer; |
||||||
|
|
||||||
|
public interface BufferAllocator { |
||||||
|
|
||||||
|
void destroyDirectBuffer(Buffer toBeDestroyed); |
||||||
|
|
||||||
|
ByteBuffer allocate(int size); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,129 @@ |
|||||||
|
/* |
||||||
|
* Copyright (c) 2009-2012 jMonkeyEngine |
||||||
|
* 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 'jMonkeyEngine' 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 com.jme3.util; |
||||||
|
|
||||||
|
import java.lang.reflect.InvocationTargetException; |
||||||
|
import java.lang.reflect.Method; |
||||||
|
import java.nio.Buffer; |
||||||
|
import java.nio.ByteBuffer; |
||||||
|
import java.nio.DoubleBuffer; |
||||||
|
import java.util.logging.Level; |
||||||
|
import java.util.logging.Logger; |
||||||
|
|
||||||
|
/** |
||||||
|
* This class contains the reflection based way to remove DirectByteBuffers in java < 9, |
||||||
|
* allocation is done via ByteBuffer.allocateDirect |
||||||
|
*/ |
||||||
|
public final class ReflectionBufferUtils implements BufferAllocator { |
||||||
|
private static Method cleanerMethod = null; |
||||||
|
private static Method cleanMethod = null; |
||||||
|
private static Method viewedBufferMethod = null; |
||||||
|
private static Method freeMethod = null; |
||||||
|
|
||||||
|
static { |
||||||
|
// Oracle JRE / OpenJDK
|
||||||
|
cleanerMethod = loadMethod("sun.nio.ch.DirectBuffer", "cleaner"); |
||||||
|
cleanMethod = loadMethod("sun.misc.Cleaner", "clean"); |
||||||
|
viewedBufferMethod = loadMethod("sun.nio.ch.DirectBuffer", "viewedBuffer"); |
||||||
|
if (viewedBufferMethod == null) { |
||||||
|
// They changed the name in Java 7 (???)
|
||||||
|
viewedBufferMethod = loadMethod("sun.nio.ch.DirectBuffer", "attachment"); |
||||||
|
} |
||||||
|
|
||||||
|
// Apache Harmony
|
||||||
|
ByteBuffer bb = BufferUtils.createByteBuffer(1); |
||||||
|
Class<?> clazz = bb.getClass(); |
||||||
|
try { |
||||||
|
freeMethod = clazz.getMethod("free"); |
||||||
|
} catch (NoSuchMethodException ex) { |
||||||
|
} catch (SecurityException ex) { |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private static Method loadMethod(String className, String methodName) { |
||||||
|
try { |
||||||
|
Method method = Class.forName(className).getMethod(methodName); |
||||||
|
method.setAccessible(true); |
||||||
|
return method; |
||||||
|
} catch (NoSuchMethodException ex) { |
||||||
|
return null; // the method was not found
|
||||||
|
} catch (SecurityException ex) { |
||||||
|
return null; // setAccessible not allowed by security policy
|
||||||
|
} catch (ClassNotFoundException ex) { |
||||||
|
return null; // the direct buffer implementation was not found
|
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
/** |
||||||
|
* This function explicitly calls the Cleaner method of a direct buffer. |
||||||
|
* |
||||||
|
* @param toBeDestroyed |
||||||
|
* The direct buffer that will be "cleaned". Utilizes reflection. |
||||||
|
* |
||||||
|
*/ |
||||||
|
public void destroyDirectBuffer(Buffer toBeDestroyed) { |
||||||
|
try { |
||||||
|
if (freeMethod != null) { |
||||||
|
freeMethod.invoke(toBeDestroyed); |
||||||
|
} else { |
||||||
|
Object cleaner = cleanerMethod.invoke(toBeDestroyed); |
||||||
|
if (cleaner != null) { |
||||||
|
cleanMethod.invoke(cleaner); |
||||||
|
} else { |
||||||
|
// Try the alternate approach of getting the viewed buffer
|
||||||
|
// first
|
||||||
|
Object viewedBuffer = viewedBufferMethod.invoke(toBeDestroyed); |
||||||
|
if (viewedBuffer != null) { |
||||||
|
destroyDirectBuffer((Buffer) viewedBuffer); |
||||||
|
} else { |
||||||
|
Logger.getLogger(BufferUtils.class.getName()).log(Level.SEVERE, "Buffer cannot be destroyed: {0}", toBeDestroyed); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} catch (IllegalAccessException ex) { |
||||||
|
Logger.getLogger(BufferUtils.class.getName()).log(Level.SEVERE, "{0}", ex); |
||||||
|
} catch (IllegalArgumentException ex) { |
||||||
|
Logger.getLogger(BufferUtils.class.getName()).log(Level.SEVERE, "{0}", ex); |
||||||
|
} catch (InvocationTargetException ex) { |
||||||
|
Logger.getLogger(BufferUtils.class.getName()).log(Level.SEVERE, "{0}", ex); |
||||||
|
} catch (SecurityException ex) { |
||||||
|
Logger.getLogger(BufferUtils.class.getName()).log(Level.SEVERE, "{0}", ex); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public ByteBuffer allocate(int size) { |
||||||
|
return ByteBuffer.allocateDirect(size); |
||||||
|
} |
||||||
|
|
||||||
|
} |
Loading…
Reference in new issue