Minor update

- Added more Android compatibility code
This commit is contained in:
Gabriel Tofvesson 2017-03-12 01:21:08 +01:00
parent 8ef3df8245
commit da48afd91c
3 changed files with 165 additions and 27 deletions

View File

@ -164,12 +164,12 @@ public class Async<T> implements Awaitable{
* @param o Return value. * @param o Return value.
*/ */
public Async(T o){ complete = true; ret = o; } public Async(T o){ complete = true; ret = o; }
/** /**
* WARNING: Package-scoped because it should only be used when overriding standard construction. Should not bw used haphazardly! * WARNING: Package-scoped because it should only be used when overriding standard construction. Should not bw used haphazardly!
*/ */
Async() { task = null; } Async() { task = null; }
/** /**
* Await completion of async task. Blocks thread if task isn't complete. * Await completion of async task. Blocks thread if task isn't complete.
* @return Return value from async method call. Return is null if {@link #cancel()} is called before this method and async task wan't finished. * @return Return value from async method call. Return is null if {@link #cancel()} is called before this method and async task wan't finished.

View File

@ -11,4 +11,8 @@ public final class Collections {
for(int i = c.size()-1; i>0; --i) a.add(t[i]); for(int i = c.size()-1; i>0; --i) a.add(t[i]);
return a; return a;
} }
public static boolean arrayContains(Object[] o, PredicateCompat p){ for(Object o1 : o) if(p.apply(o1)) return true; return false; }
public interface PredicateCompat<T>{ boolean apply(T t); }
} }

View File

@ -1,14 +1,14 @@
package com.tofvesson.reflection; package com.tofvesson.reflection;
import sun.misc.Unsafe; import sun.misc.Unsafe;
import java.lang.reflect.*; import java.lang.reflect.*;
import java.util.*; import java.util.*;
import java.util.Collections;
/** /**
* Safe tools to help simplify code when dealing with reflection. * Safe tools to help simplify code when dealing with reflection.
*/ */
@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue"}) @SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue", "ConstantConditions"})
public final class SafeReflection { public final class SafeReflection {
@ -39,11 +39,14 @@ public final class SafeReflection {
try { try {
f = Field.class.getDeclaredField("modifiers"); f = Field.class.getDeclaredField("modifiers");
f.setAccessible(true); f.setAccessible(true);
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")); // Most desktop versions of Java
}catch(Exception e){ }catch(Exception e){
try { f = Field.class.getDeclaredField("artField").getType().getDeclaredField("accessFlags"); } try { f = Field.class.getDeclaredField("artField").getType().getDeclaredField("accessFlags"); }
catch(Exception ignored){ f = Field.class.getDeclaredField("accessFlags"); } catch(Exception ignored){ f = Field.class.getDeclaredField("accessFlags"); }
f.setAccessible(true); f.setAccessible(true);
}
try{
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")); // Most desktop versions of Java
}catch(Exception e){
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("flag")); // God-damned Android l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("flag")); // God-damned Android
} }
try { try {
@ -55,7 +58,7 @@ public final class SafeReflection {
b = false; b = false;
} }
u.putInt(f, l, 1); u.putInt(f, l, 1);
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception is never thrown }catch(Exception ignored){ ignored.printStackTrace(); } // Exception should never be thrown
unsafe = u; unsafe = u;
newInstance = m; newInstance = m;
aConAccess = m1; aConAccess = m1;
@ -190,27 +193,46 @@ public final class SafeReflection {
} }
/** /**
* Gets the static object stored in the field with the specified name in the defined class. * Gets the object stored in the field with the specified name in the class of the defined object.
* @param c Class to find object in. * @param from Object to find object in.
* @param c Class to get field from (if <b>from</b> is null). This is used for static fields.
* @param name Name of field. * @param name Name of field.
* @return Object or null if object is null or field doesn't exist. * @return Object or null if object is null or field doesn't exist.
*/ */
public static Object getStaticFieldObject(Class<?> c, String name){ public static Object getValue(Object from, Class<?> c, String name){
Field f = getField(c, name); try{ return getField(from!=null?from.getClass():c, name).get(from); }catch(Throwable ignored){}
try { return f!=null?f.get(null):null; } catch (Exception e) { }
return null; return null;
} }
/** /**
* Gets the object stored in the field with the specified name in the class of the defined object. * Gets the object stored in the static field with the specified name in the class of the defined object.
* @param o Object to find object in. * @param c Class to get field from (if <b>from</b> is null). This is used for static fields.
* @param name Name of field. * @param name Name of field.
* @return Object or null if object is null or field doesn't exist. * @return Object or null if object is null or field doesn't exist.
*/ */
public static Object getFieldObject(Object o, String name){ public static Object getValue(Class<?> c, String name){ return getValue(null, c, name); }
Field f = getField(o.getClass(), name);
try{ return f!=null?f.get(o):null; }catch(Exception e){} /**
return null; * Set field to specified value.
* <br><h1 style="text-align: center; color: red;">Please note:</h1>A JIT compiler may inline private final fields in methods which will prevent the actual value
* from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
* (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
* This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
* like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
* @param inv Object whose field to set the value of. Can be null.
* @param in Class to find field in. (If inv is null)
* @param name Name of field to modify.
* @param value Value to set the field to.
* @param vol Whether or not the operation should be volatile.
* @return True if setting value succeeded.
*/
public static boolean setValue(Object inv, Class<?> in, String name, Object value, boolean vol){
try{
Field f = getField(inv==null?in:inv.getClass(), name);
getPutMethod(f.getType(), vol).invoke(unsafe, createUnsafePutParams(inv, f, value));
return true;
}catch(Exception e){ e.printStackTrace(); }
return false;
} }
/** /**
@ -221,19 +243,55 @@ public final class SafeReflection {
* This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant * This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
* like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic. * like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
* @param inv Object whose field to set the value of. Can be null. * @param inv Object whose field to set the value of. Can be null.
* @param f Field to modify. * @param in Class to find field in. (If inv is null)
* @param name Name of field to modify.
* @param value Value to set the field to. * @param value Value to set the field to.
* @return True if setting value succeeded. * @return True if setting value succeeded.
*/ */
public static boolean setFieldValue(Object inv, Field f, Object value){ public static boolean setValue(Object inv, Class<?> in, String name, Object value){ return setValue(inv, in, name, value, false); }
try{
unsafe.putInt(f, override, 1); // Ensure field override flag is set /**
if(Modifier.isFinal(f.getModifiers()) && Modifier.isStatic(f.getModifiers())) modifiers.setInt(f, f.getModifiers() &~ Modifier.FINAL); * Set field to specified value as a volatile operations.
f.set(inv, value); * <br><h1 style="text-align: center; color: red;">Please note:</h1>A JIT compiler may inline private final fields in methods which will prevent the actual value
return true; * from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
}catch(Exception e){ e.printStackTrace(); } * (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
return false; * This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
} * like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
* @param inv Object whose field to set the value of. Can be null.
* @param in Class to find field in. (If inv is null)
* @param name Name of field to modify.
* @param value Value to set the field to.
* @return True if setting value succeeded.
*/
public static boolean setValueVolatile(Object inv, Class<?> in, String name, Object value){ return setValue(inv, in, name, value, true); }
/**
* Set static field to specified value.
* <br><h1 style="text-align: center; color: red;">Please note:</h1>A JIT compiler may inline private final fields in methods which will prevent the actual value
* from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
* (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
* This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
* like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
* @param in Class to find field in.
* @param name Name of field to modify.
* @param value Value to set the field to.
* @return True if setting value succeeded.
*/
public static boolean setValue(Class<?> in, String name, Object value){ return setValue(null, in, name, value, false); }
/**
* Set static field to specified value as a volatile operation.
* <br><h1 style="text-align: center; color: red;">Please note:</h1>A JIT compiler may inline private final fields in methods which will prevent the actual value
* from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
* (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
* This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
* like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
* @param in Class to find field in.
* @param name Name of field to modify.
* @param value Value to set the field to.
* @return True if setting value succeeded.
*/
public static boolean setValueVolatile(Class<?> in, String name, Object value){ return setValue(null, in, name, value, true); }
/** /**
* Gets a reference to the enclosing class from a defined inner (nested) object. * Gets a reference to the enclosing class from a defined inner (nested) object.
@ -459,7 +517,7 @@ public final class SafeReflection {
*/ */
public static Class<?> getCallerClass(){ public static Class<?> getCallerClass(){
ArrayList<StackTraceElement> s = getStacktraceWithoutReflection(); ArrayList<StackTraceElement> s = getStacktraceWithoutReflection();
try { return Class.forName(s.get(s.size()==2?1:2).getClassName()); } catch (ClassNotFoundException e) { } try { return Class.forName(s.get(s.size()==3?2:s.size()==2?1:0).getClassName()); } catch (ClassNotFoundException e) { }
assert false:"Unreachable code reached"; assert false:"Unreachable code reached";
return null; return null;
} }
@ -493,4 +551,80 @@ public final class SafeReflection {
} }
return s; return s;
} }
public static Method getSetMethod(Class<?> c){
try {
Method m = Field.class.getDeclaredMethod(getSetMethodName(c), Object.class, isAutoBoxedClass(c)?unbox(c):c.isPrimitive()?c:Object.class);
m.setAccessible(true);
return m;
} catch (Exception e) { e.printStackTrace(); }
return null; // Not object version of primitive
}
public static String getSetMethodName(Class<?> c){
try {
String s;
boolean b;
Field f = null;
try{
f = c.getDeclaredField("value");
}catch(Exception e){}
if(f!=null) f.setAccessible(true);
return"set"+
(f!=null && Number.class.isAssignableFrom(f.getType()) && f.getType().isPrimitive()?
(s=f.getType().getSimpleName()).replaceFirst(s.charAt(0)+"", Character.toUpperCase(s.charAt(0))+""):
c.isPrimitive()?(s=c.getSimpleName()).replaceFirst(s.charAt(0)+"", Character.toUpperCase(s.charAt(0))+"")
:"");
} catch (Exception e) { e.printStackTrace(); }
return null; // Not object version of primitive
}
public static boolean isAutoBoxedClass(final Class<?> c){
return Number.class.isAssignableFrom(c) && com.tofvesson.collections.Collections.arrayContains(c.getDeclaredFields(),
new com.tofvesson.collections.Collections.PredicateCompat() {
public boolean apply(Object o) {
return ((Field) o).getName().equals("value") && box(((Field) o).getType())==c;
}
});
}
public static Method getPutMethod(Class<?> c, boolean vol){
try{
Method m = Unsafe.class.getDeclaredMethod(
getSetMethodName(c).replaceFirst("set", "put") + (isAutoBoxedClass(c)||c.isPrimitive()?"":"Object")+(vol?"Volatile":""),
Object.class,
long.class,
c.isPrimitive()?c:Object.class
);
m.setAccessible(true);
return m;
}catch(Exception ignored){ ignored.printStackTrace(); }
return null; // Something went wrong...
}
public static Object[] createUnsafePutParams(Object invokee, Field field, Object value){
return new Object[]{ invokee==null?field.getDeclaringClass():invokee, invokee==null?unsafe.staticFieldOffset(field):unsafe.objectFieldOffset(field), value };
}
public static Class<?> box(Class<?> c){
if(!c.isPrimitive()) return c;
if(c==int.class) return Integer.class;
if(c==char.class) return Character.class;
if(c==byte.class) return Byte.class;
if(c==float.class) return Float.class;
if(c==long.class) return Long.class;
if(c==boolean.class) return Boolean.class;
return Double.class;
}
public static Class<?> unbox(Class<?> c){
if(c.isPrimitive()) return c;
if(c==Integer.class) return int.class;
if(c==Character.class) return char.class;
if(c==Byte.class) return byte.class;
if(c==Float.class) return float.class;
if(c==Long.class) return long.class;
if(c==Boolean.class) return boolean.class;
return double.class;
}
} }