diff --git a/src/com/tofvesson/async/Async.java b/src/com/tofvesson/async/Async.java index c70744b..3995910 100644 --- a/src/com/tofvesson/async/Async.java +++ b/src/com/tofvesson/async/Async.java @@ -164,12 +164,12 @@ public class Async implements Awaitable{ * @param o Return value. */ 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! */ Async() { task = null; } + /** * 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. diff --git a/src/com/tofvesson/collections/Collections.java b/src/com/tofvesson/collections/Collections.java index 0c4dcfb..43c26f3 100644 --- a/src/com/tofvesson/collections/Collections.java +++ b/src/com/tofvesson/collections/Collections.java @@ -11,4 +11,8 @@ public final class Collections { for(int i = c.size()-1; i>0; --i) a.add(t[i]); 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{ boolean apply(T t); } } diff --git a/src/com/tofvesson/reflection/SafeReflection.java b/src/com/tofvesson/reflection/SafeReflection.java index ba40ffc..5d1f4b0 100644 --- a/src/com/tofvesson/reflection/SafeReflection.java +++ b/src/com/tofvesson/reflection/SafeReflection.java @@ -1,14 +1,14 @@ package com.tofvesson.reflection; import sun.misc.Unsafe; - import java.lang.reflect.*; import java.util.*; +import java.util.Collections; /** * Safe tools to help simplify code when dealing with reflection. */ -@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue"}) +@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue", "ConstantConditions"}) public final class SafeReflection { @@ -39,11 +39,14 @@ public final class SafeReflection { try { f = Field.class.getDeclaredField("modifiers"); f.setAccessible(true); - l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")); // Most desktop versions of Java }catch(Exception e){ try { f = Field.class.getDeclaredField("artField").getType().getDeclaredField("accessFlags"); } catch(Exception ignored){ f = Field.class.getDeclaredField("accessFlags"); } 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 } try { @@ -55,7 +58,7 @@ public final class SafeReflection { b = false; } 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; newInstance = m; 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. - * @param c Class to find object in. + * Gets the object stored in the field with the specified name in the class of the defined object. + * @param from Object to find object in. + * @param c Class to get field from (if from is null). This is used for static fields. * @param name Name of field. * @return Object or null if object is null or field doesn't exist. */ - public static Object getStaticFieldObject(Class c, String name){ - Field f = getField(c, name); - try { return f!=null?f.get(null):null; } catch (Exception e) { } + public static Object getValue(Object from, Class c, String name){ + try{ return getField(from!=null?from.getClass():c, name).get(from); }catch(Throwable ignored){} return null; } /** - * Gets the object stored in the field with the specified name in the class of the defined object. - * @param o Object to find object in. + * Gets the object stored in the static field with the specified name in the class of the defined object. + * @param c Class to get field from (if from is null). This is used for static fields. * @param name Name of field. * @return Object or null if object is null or field doesn't exist. */ - public static Object getFieldObject(Object o, String name){ - Field f = getField(o.getClass(), name); - try{ return f!=null?f.get(o):null; }catch(Exception e){} - return null; + public static Object getValue(Class c, String name){ return getValue(null, c, name); } + + /** + * Set field to specified value. + *

Please note:

A JIT compiler may inline private final fields in methods which will prevent the actual value + * from changing at runtime.
This means that the value stored in the field will 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
"public static final boolean b = false;"
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 * like
"public static final boolean b = false;"
might be problematic. * @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. * @return True if setting value succeeded. */ - public static boolean setFieldValue(Object inv, Field f, Object value){ - 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); - f.set(inv, value); - return true; - }catch(Exception e){ e.printStackTrace(); } - return false; - } + public static boolean setValue(Object inv, Class in, String name, Object value){ return setValue(inv, in, name, value, false); } + + /** + * Set field to specified value as a volatile operations. + *

Please note:

A JIT compiler may inline private final fields in methods which will prevent the actual value + * from changing at runtime.
This means that the value stored in the field will 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
"public static final boolean b = false;"
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. + *

Please note:

A JIT compiler may inline private final fields in methods which will prevent the actual value + * from changing at runtime.
This means that the value stored in the field will 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
"public static final boolean b = false;"
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. + *

Please note:

A JIT compiler may inline private final fields in methods which will prevent the actual value + * from changing at runtime.
This means that the value stored in the field will 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
"public static final boolean b = false;"
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. @@ -459,7 +517,7 @@ public final class SafeReflection { */ public static Class getCallerClass(){ ArrayList 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"; return null; } @@ -493,4 +551,80 @@ public final class SafeReflection { } 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; + } }