From 1321d49f8c578d5663e57c3d94db5876d44cd80f Mon Sep 17 00:00:00 2001 From: FuckYou Date: Sat, 12 Nov 2016 01:28:54 +0400 Subject: [PATCH] Major update - Fixed prominent bugs in ShiftingList - Fixed shifting of data in the internal array - Fixed retainAll() - Fixed add(int, collection) - Fixed much more - Changed name of enqueue() method in WorkerThread to push() - Moved Enum reflection code to SafeReflection class - Moved class instantiation (bypass construction) code to SafeReflection and optimized code - Moved EnumDefinition class into SafeReflection as a static class --- src/com/tofvesson/async/WorkerThread.java | 4 +- .../tofvesson/collections/ShiftingList.java | 48 ++-- .../tofvesson/reflection/EnumDefinition.java | 114 --------- src/com/tofvesson/reflection/Helper.java | 111 --------- .../tofvesson/reflection/SafeReflection.java | 225 ++++++++++++++++++ 5 files changed, 252 insertions(+), 250 deletions(-) delete mode 100644 src/com/tofvesson/reflection/EnumDefinition.java delete mode 100644 src/com/tofvesson/reflection/Helper.java diff --git a/src/com/tofvesson/async/WorkerThread.java b/src/com/tofvesson/async/WorkerThread.java index 00f3d5f..95e8f59 100644 --- a/src/com/tofvesson/async/WorkerThread.java +++ b/src/com/tofvesson/async/WorkerThread.java @@ -56,7 +56,7 @@ public class WorkerThread extends Thread { * @param params Parameters for method. * @return A UID corresponding to the queued instruction. */ - public long enqueue(Object invokeOn, Method m, Object... params){ + public long push(Object invokeOn, Method m, Object... params){ m.setAccessible(true); long id; do{ id = ThreadLocalRandom.current().nextLong(); }while(ids.contains(id)); @@ -73,6 +73,7 @@ public class WorkerThread extends Thread { public Object pop(long id){ if(!ids.contains(id)) return null; if(Thread.currentThread() == this) throw new RuntimeException("Attempting to await result in worker thread! This causes the thread to lock."); + //noinspection StatementWithEmptyBody while(!output.containsKey(id)) ; // Block caller thread until result is received Object o = output.get(id); output.remove(id); @@ -93,6 +94,7 @@ public class WorkerThread extends Thread { */ public void stopForced(){ alive = false; + //noinspection deprecation stop(); } } diff --git a/src/com/tofvesson/collections/ShiftingList.java b/src/com/tofvesson/collections/ShiftingList.java index 67b525c..8712ff8 100644 --- a/src/com/tofvesson/collections/ShiftingList.java +++ b/src/com/tofvesson/collections/ShiftingList.java @@ -3,8 +3,9 @@ package com.tofvesson.collections; import java.util.ArrayList; import java.util.Collection; import java.util.List; +import java.util.stream.Collectors; -@SuppressWarnings({"unchecked", "ReturnOfInnerClass"}) +@SuppressWarnings({"unchecked", "ReturnOfInnerClass", "unused"}) public class ShiftingList implements List { /** * Holder for entries. Dur to reasons, the array holds objects. @@ -54,7 +55,7 @@ public class ShiftingList implements List { @Override public boolean contains(Object o) { if(o==empty) return false; - for(Object o1 : entries) if(o.equals(o1)) return true; + for(int i = 0; i implements List { @Override public boolean add(E e) { + if(contains(e)) return false; preparePopulate(1); entries[0] = e!=null?e:empty; pop=pop!=maxSize?pop+1:pop; @@ -103,28 +105,25 @@ public class ShiftingList implements List { @Override public boolean addAll(Collection c) { - ArrayList l = new ArrayList<>(c); - preparePopulate(c.size()); - for(int i = 0; i l = c.stream().filter(e -> !contains(e)).collect(Collectors.toCollection(ArrayList::new)); + if(l.size()>maxSize) for(int i = maxSize; i c) { + if(index>=maxSize) return false; if(index>=entries.length || index<0 || c.size()==0) return false; - ArrayList l = new ArrayList<>(c); - for(int i = 0; imaxSize) for(int i = pop+l.size()-maxSize; i l = c.stream().filter(e -> !contains(e)).collect(Collectors.toCollection(ArrayList::new)); + if(index+l.size()>maxSize) for(int i = maxSize-index; i= maxSize ? pop : pop+l.size(); + if(l.size()+index implements List { } Object[] o = new Object[(int) Math.max(1, Math.min((pop+accountFor)/load, maxSize))]; // Load adaptation algorithm capping at maxSize or 0 System.arraycopy(entries, 0, o, 0, Math.min(o.length, entries.length)); // Move as many entries as possible + entries = o; } /** @@ -195,17 +195,17 @@ public class ShiftingList implements List { */ protected void shift(){ for(int i = 0; ientries.length) adaptLoad(accountFor); // If new elements exceed limit, adapt load + if(accountFor+pop>entries.length) adaptLoad(accountFor); // If new elements exceed limit, adapt load if(accountFor>entries.length) return; // No need to delete elements if new values exceed limit System.arraycopy(entries, 0, entries, accountFor, entries.length-accountFor); // Shift array elements to account for new elements } @@ -226,7 +226,7 @@ public class ShiftingList implements List { @Override public void add(int index, E element) { - if(index>=entries.length || index<0) return; + if(index>=entries.length || index<0 || contains(element)) return; Object o = element==null?empty:element; pop = pop==maxSize?pop:pop+1; if(pop==entries.length) adaptLoad(); @@ -284,9 +284,9 @@ public class ShiftingList implements List { protected int counter = 0; private final int pop; private final Object[] entries; - public Iterator(int pop, Object[] entries){ this.pop = pop; this.entries = entries; } + public Iterator(int pop, Object[] entries){ this.pop = pop; this.entries = entries; } @Override public boolean hasNext() { return counter params = new HashMap<>(); // Assign a specific type to each parameter - - /** - * Put an object in the parameter list. - * @param value The parameter to supply. (Type is derived automatically) - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putObject(Object value){ - params.put(value, value.getClass()); - return this; - } - - /** - * Put a primitive value in the parameter list. - * @param autoBoxed An autoboxed version of the parameter. (For example putPrimitive(5) will automatically become Integer) - * @return A reference to the EnumDefinition object this method was called on (for chaining). - * @throws NoSuchFieldException - */ - public EnumDefinition putPrimitive(Object autoBoxed) throws NotAutoBoxedException{ - // All autoboxed versions of primitives have a reference to their boxed primitive - try { - params.put(autoBoxed, autoBoxed.getClass().getDeclaredField("value").getType()); - }catch(Exception e){ throw new NotAutoBoxedException(); } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for boolean. - * @param b Boolean parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putBoolean(boolean b){ - try { return putPrimitive(b); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for byte. - * @param b Byte parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putByte(byte b){ - try { return putPrimitive(b); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for char. - * @param c Character parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putChar(char c){ - try { return putPrimitive(c); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for short. - * @param s Short parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putShort(short s){ - try { return putPrimitive(s); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for int. - * @param i Integer parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putInt(int i){ - try { return putPrimitive(i); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for long. - * @param l Long parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putLong(long l){ - try { return putPrimitive(l); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for float. - * @param f Floating point parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putFloat(float f){ - try { return putPrimitive(f); } catch (NotAutoBoxedException e) { } - return this; - } - - /** - * Safe wrapper of {@link #putPrimitive(Object)} for double. - * @param d Double parameter. - * @return A reference to the EnumDefinition object this method was called on (for chaining). - */ - public EnumDefinition putDouble(double d){ - try { return putPrimitive(d); } catch (NotAutoBoxedException e) { } - return this; - } -} diff --git a/src/com/tofvesson/reflection/Helper.java b/src/com/tofvesson/reflection/Helper.java deleted file mode 100644 index 39003c4..0000000 --- a/src/com/tofvesson/reflection/Helper.java +++ /dev/null @@ -1,111 +0,0 @@ -package com.tofvesson.reflection; - -import sun.misc.Unsafe; -import sun.reflect.ConstructorAccessor; -import java.lang.reflect.Array; -import java.lang.reflect.Constructor; -import java.lang.reflect.Field; -import java.lang.reflect.Method; - -public class Helper { - /** - * Allows you to create a completely custom Enum value. If the supplied value already exists, it will be returned. - * will not be created; the value of the existing one will be updated. - * @param clazz The class to attribute the new enum to. - * @param addToValuesArray Whether or not to update the internal, "immutable" values array with the new value (ignored if value already exists). - * @return A new/existing enum. - */ - @SuppressWarnings("unchecked") - public static > T customEnum(Class clazz, boolean addToValuesArray, String name, EnumDefinition def){ - T u; - try { - // Get a reference to the static method values() and get values - Method v = clazz.getDeclaredMethod("values"); - v.setAccessible(true); - T[] values = (T[]) v.invoke(null); - - // Return object if it already exists - for(T u2 : values) if(u2.name().equals(name)) return u2; - - // Generate enum parameter definition - Class[] paramList = new Class[def.params.size()+2]; - paramList[0] = String.class; // Name - paramList[1] = int.class; // Ordinal - int iterator = paramList.length; - for(Class c : def.params.values()) paramList[--iterator] = c; // Shit's fucking reversed - - // Get enum constructor (inherited from Enum.class) - Constructor c = clazz.getDeclaredConstructor(paramList); - c.setAccessible(true); - Method m = Constructor.class.getDeclaredMethod("acquireConstructorAccessor"); - m.setAccessible(true); - - // Get constructor accessor since Constructor.newInstance throws an exception because Enums are "immutable" - ConstructorAccessor access = (ConstructorAccessor) m.invoke(c); - - Object[] parameters = new Object[def.params.size()+2]; - parameters[0] = name; - parameters[1] = values.length; - iterator = parameters.length; - for(Object o : def.params.keySet()) parameters[--iterator] = o; - - // Create new instance of enum with valid name and ordinal - u = (T) access.newInstance(parameters); - - // Get the final name field from Enum.class and make it temporarily modifiable - Field f = Enum.class.getDeclaredField("name"); - f.setAccessible(true); - - // Rename the newly created enum to the requested name - f.set(u, name); - - if(!addToValuesArray) return u; // Stops here if - - // Get the current values field from Enum (a bitch to modify) - f = clazz.getDeclaredField("$VALUES"); - f.setAccessible(true); - T[] $VALUES = (T[]) Array.newInstance(clazz, values.length+1); - System.arraycopy(values, 0, $VALUES, 0, values.length); // Copy over values from old array - $VALUES[values.length] = u; // Add out custom enum to our local array - - // Start doing magic by getting an instance of sun.reflect.UnsafeQualifiedStaticObjectFieldAccessorImpl.class - // Class is package-local so we can't reference it by anything other than Object - m = Field.class.getDeclaredMethod("getFieldAccessor", Object.class); - m.setAccessible(true); - Object UQSOFAImpl = m.invoke(f, u); - - // Get "isReadOnly" flag ($VALUES is always read-only even if Field.setAccessible(true) is called) - // Flag is located in superclass (sun.reflect.UnsafeQualifiedStaticFieldAccessorImpl.class (also fucking package-local)) - // Set flag to 'false' to allow for modification against Java's will - Field f1 = UQSOFAImpl.getClass().getSuperclass().getDeclaredField("isReadOnly"); - f1.setAccessible(true); - f1.setBoolean(UQSOFAImpl, false); - - // Invoke set() method on UnsafeQualifiedStaticObjectFieldAccessorImpl object which sets the - // private field $VALUES to our new array - m = UQSOFAImpl.getClass().getDeclaredMethod("set", Object.class, Object.class); - m.setAccessible(true); - m.invoke(UQSOFAImpl, f, $VALUES); - } catch (Exception wrongParams) { throw new RuntimeException(wrongParams); } - return u; - } - - /** - * Create a new object without the hassle of having to construct it. WARNING: Not usually a good idea. - * @param clazz Class to instantiate. - * @return An object instance of the supplied class that hasn't been constructed. - * @throws InstantiationException - */ - public static T create(Class clazz) throws InstantiationException{ - try { - Field f = Unsafe.class.getDeclaredField("theUnsafe"); - f.setAccessible(true); - //noinspection unchecked - return (T) ((Unsafe) f.get(null)).allocateInstance(clazz); - } catch (NoSuchFieldException | IllegalAccessException e) { - System.err.println("Whatever you did, it was VERY wrong! Stop it!"); - e.printStackTrace(); - return null; - } - } -} diff --git a/src/com/tofvesson/reflection/SafeReflection.java b/src/com/tofvesson/reflection/SafeReflection.java index 548a9b5..3754d38 100644 --- a/src/com/tofvesson/reflection/SafeReflection.java +++ b/src/com/tofvesson/reflection/SafeReflection.java @@ -1,13 +1,32 @@ package com.tofvesson.reflection; +import sun.misc.Unsafe; +import sun.reflect.ConstructorAccessor; + +import java.lang.reflect.Array; +import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; +import java.util.HashMap; /** * Safe tools to help simplify code when dealing with reflection. */ @SuppressWarnings("unused") public class SafeReflection { + + private static final Unsafe unsafe; + + static{ + Unsafe u = null; + try{ + Field f = Unsafe.class.getDeclaredField("theUnsafe"); + f.setAccessible(true); + u = (Unsafe) f.get(null); + }catch(Exception ignored){} // Exception is never thrown + unsafe = u; + } + /** * Gets the method from the defined class by name and parameters. * Method is accessible. @@ -131,4 +150,210 @@ public class SafeReflection { * @return File if class file exists. */ public static java.io.File getClassFile(Class c){ return new java.io.File(c.getResource(c.getSimpleName()+".class").getFile()); } + + /** + * Allows you to create a completely custom Enum value. If the supplied value already exists, it will be returned. + * will not be created; the value of the existing one will be updated. + * @param clazz The class to attribute the new enum to. + * @param addToValuesArray Whether or not to update the internal, "immutable" values array with the new value (ignored if value already exists). + * @return A new/existing enum. + */ + @SuppressWarnings("unchecked") + public static > T customEnum(Class clazz, boolean addToValuesArray, String name, EnumDefinition def){ + T u; + try { + // Get a reference to the static method values() and get values + Method v = clazz.getDeclaredMethod("values"); + v.setAccessible(true); + T[] values = (T[]) v.invoke(null); + + // Return object if it already exists + for(T u2 : values) if(u2.name().equals(name)) return u2; + + // Generate enum parameter definition + Class[] paramList = new Class[def.params.size()+2]; + paramList[0] = String.class; // Name + paramList[1] = int.class; // Ordinal + int iterator = paramList.length; + for(Class c : def.params.values()) paramList[--iterator] = c; // Shit's fucking reversed + + // Get enum constructor (inherited from Enum.class) + Constructor c = clazz.getDeclaredConstructor(paramList); + c.setAccessible(true); + Method m = Constructor.class.getDeclaredMethod("acquireConstructorAccessor"); + m.setAccessible(true); + + // Get constructor accessor since Constructor.newInstance throws an exception because Enums are "immutable" + ConstructorAccessor access = (ConstructorAccessor) m.invoke(c); + + Object[] parameters = new Object[def.params.size()+2]; + parameters[0] = name; + parameters[1] = values.length; + iterator = parameters.length; + for(Object o : def.params.keySet()) parameters[--iterator] = o; + + // Create new instance of enum with valid name and ordinal + u = (T) access.newInstance(parameters); + + // Get the final name field from Enum.class and make it temporarily modifiable + Field f = Enum.class.getDeclaredField("name"); + f.setAccessible(true); + + // Rename the newly created enum to the requested name + f.set(u, name); + + if(!addToValuesArray) return u; // Stops here if + + // Get the current values field from Enum (a bitch to modify) + f = clazz.getDeclaredField("$VALUES"); + f.setAccessible(true); + T[] $VALUES = (T[]) Array.newInstance(clazz, values.length+1); + System.arraycopy(values, 0, $VALUES, 0, values.length); // Copy over values from old array + $VALUES[values.length] = u; // Add out custom enum to our local array + + // Start doing magic by getting an instance of sun.reflect.UnsafeQualifiedStaticObjectFieldAccessorImpl.class + // Class is package-local so we can't reference it by anything other than Object + m = Field.class.getDeclaredMethod("getFieldAccessor", Object.class); + m.setAccessible(true); + Object UQSOFAImpl = m.invoke(f, u); + + // Get "isReadOnly" flag ($VALUES is always read-only even if Field.setAccessible(true) is called) + // Flag is located in superclass (sun.reflect.UnsafeQualifiedStaticFieldAccessorImpl.class (also fucking package-local)) + // Set flag to 'false' to allow for modification against Java's will + Field f1 = UQSOFAImpl.getClass().getSuperclass().getDeclaredField("isReadOnly"); + f1.setAccessible(true); + f1.setBoolean(UQSOFAImpl, false); + + // Invoke set() method on UnsafeQualifiedStaticObjectFieldAccessorImpl object which sets the + // private field $VALUES to our new array + m = UQSOFAImpl.getClass().getDeclaredMethod("set", Object.class, Object.class); + m.setAccessible(true); + m.invoke(UQSOFAImpl, f, $VALUES); + } catch (Exception wrongParams) { throw new RuntimeException(wrongParams); } + return u; + } + + /** + * Create a new object without the hassle of having to construct it. WARNING: Not usually a good idea. + * @param clazz Class to instantiate. + * @return An object instance of the supplied class that hasn't been constructed. + * @throws InstantiationException Thrown if instantiating the object fails for some reason. + */ + public static T createNewObject(Class clazz) throws InstantiationException{ + //noinspection unchecked + return (T) unsafe.allocateInstance(clazz); + } + + + /** + * A definition for custom enum creation. + */ + public static class EnumDefinition { + HashMap params = new HashMap<>(); // Assign a specific type to each parameter + + /** + * Put an object in the parameter list. + * @param value The parameter to supply. (Type is derived automatically) + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putObject(Object value){ + params.put(value, value.getClass()); + return this; + } + + /** + * Put a primitive value in the parameter list. + * @param autoBoxed An autoboxed version of the parameter. (For example putPrimitive(5) will automatically become Integer) + * @return A reference to the EnumDefinition object this method was called on (for chaining). + * @throws NotAutoBoxedException Thrown if a value expected to be autoboxed isn't autoboxed. + */ + public EnumDefinition putPrimitive(Object autoBoxed) throws NotAutoBoxedException{ + // All autoboxed versions of primitives have a reference to their boxed primitive + try { + params.put(autoBoxed, autoBoxed.getClass().getDeclaredField("value").getType()); + }catch(Exception e){ throw new NotAutoBoxedException(); } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for boolean. + * @param b Boolean parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putBoolean(boolean b){ + try { return putPrimitive(b); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for byte. + * @param b Byte parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putByte(byte b){ + try { return putPrimitive(b); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for char. + * @param c Character parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putChar(char c){ + try { return putPrimitive(c); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for short. + * @param s Short parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putShort(short s){ + try { return putPrimitive(s); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for int. + * @param i Integer parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putInt(int i){ + try { return putPrimitive(i); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for long. + * @param l Long parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putLong(long l){ + try { return putPrimitive(l); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for float. + * @param f Floating point parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putFloat(float f){ + try { return putPrimitive(f); } catch (NotAutoBoxedException e) { } + return this; + } + + /** + * Safe wrapper of {@link #putPrimitive(Object)} for double. + * @param d Double parameter. + * @return A reference to the EnumDefinition object this method was called on (for chaining). + */ + public EnumDefinition putDouble(double d){ + try { return putPrimitive(d); } catch (NotAutoBoxedException e) { } + return this; + } + } + }