Major update

- Added ability to "post" a return value to the Async object before completion
- Added ability to poll the return value of Async without awaiting completion
- Simplified SafeReflection "customEnum" method using Unsafe.
- Modified SafeReflection.<clinit> to support android
This commit is contained in:
Gabriel Tofvesson 2017-01-10 07:24:04 +01:00
parent b89ed6bf77
commit 372acd1f7e
3 changed files with 69 additions and 54 deletions

2
.idea/misc.xml generated
View File

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_5" default="false" project-jdk-name="1.8" project-jdk-type="JavaSDK">
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_5" default="false" project-jdk-name="1.8 (1)" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/out" />
</component>
</project>

View File

@ -1,8 +1,6 @@
package com.tofvesson.async;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.*;
@SuppressWarnings({"WeakerAccess", "unused", "unchecked"})
public class Async<T> {
@ -17,6 +15,11 @@ public class Async<T> {
*/
volatile T ret;
/**
* Whether or not a posted value can be overwritten.
*/
volatile boolean ovw = true;
/**
* Status indicators.
*/
@ -88,7 +91,8 @@ public class Async<T> {
public void run(){
try {
new ThreadLocal<Async>().set(Async.this);
ret = (T) method.invoke(o, params); // Invoke given method
T ret = (T)method.invoke(o, params);// Invoke given method
if(ovw) Async.this.ret = ret; // Checks if a sticky value already has been posted
complete = true; // Notify all threads who are checking
} catch (Throwable t1) { // Prepare for failure
if(!failed) { // Checks if task was canceled
@ -133,7 +137,8 @@ public class Async<T> {
public void run(){
new ThreadLocal<Async>().set(Async.this);
try {
ret = c.newInstance(params); // Create a new instance: invoke "<init>" method
T ret = c.newInstance(params); // Create a new instance: invoke "<init>" method
if(ovw) Async.this.ret = ret; // Checks if a sticky value already has been posted
complete = true; // Notify all threads that async is finished
} catch (Throwable t1) { // Handle crash
if(!failed) { // Ensure that crash wasn't called by cancel()
@ -177,11 +182,12 @@ public class Async<T> {
*/
public boolean isAlive(){ return task.isAlive(); } // Check if thread is still alive which directly determines if process is alive since Async is a wrapper of Thread
/**
* Get async instance pertaining to current thread.
* @return Async owning current thread or null if thread isn't Async.
*/
public static Async current(){
public static <T> Async<T> current(){
try{
Object holder;
Field f = Thread.class.getDeclaredField("threadLocals");
@ -196,7 +202,7 @@ public class Async<T> {
f.setAccessible(true);
containsData = true;
}
if((holder=f.get(o)) instanceof Async) return (Async) holder;
if((holder=f.get(o)) instanceof Async) return (Async<T>) holder;
}
}catch(Exception e){}
return null;
@ -222,6 +228,34 @@ public class Async<T> {
}
}
/**
* Post a return value to Async object.
* Meant to be used from inside an instance of Async as a way to prematurely provide a return value without stopping said async process.
* Posting a value does not inform the Async object that it has completed operations.
* @param value Value to post.
* @param allowOverwrite If the return type of the async process should be overwritten when said process is completed. Calling postReturn overrides this.
*/
public void postReturn(T value, boolean allowOverwrite){
ret = value;
ovw = allowOverwrite;
}
/**
* Post a return value to Async object.
* Meant to be used from inside an instance of Async as a way to prematurely provide a return value without stopping said async process.
* Posting a value does not inform the Async object that it has completed operations.
* @param value Value to post.
*/
public void postReturn(T value){ postReturn(value, false); }
/**
* Get return type posted by object.
* Meant to be used after a value has been posted, not when it has been returned.
*/
public T getReturn(){
return ret;
}
/**
* Checks for dangerous calls to Async methods such as awaiting oneself (which would cause a freeze in that thread).
*/

View File

@ -3,60 +3,54 @@ package com.tofvesson.reflection;
import sun.misc.Unsafe;
import java.lang.reflect.*;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.*;
/**
* Safe tools to help simplify code when dealing with reflection.
*/
@SuppressWarnings("ALL")
public final class SafeReflection {
private static final Unsafe unsafe;
private static final Method newInstance, aConAccess, gFieldAccess, fieldAccess;
private static final Field ro;
private static final Method newInstance, aConAccess;
private static final String version;
private static final long override;
private static final boolean hasAConAccess;
static{
Unsafe u = null;
Method m = null, m1 = null, m2 = null, m3 = null;
Field f = null;
Method m = null, m1 = null;
Field f;
long l = 0;
String ver = "sun.reflect";
String ver = "";
boolean b = true;
//Get package based on java version (Java 9+ use "jdk.internal.reflect" while "sun.reflect" is used by earlier versions)
try{
Class.forName("sun.reflect.DelegatingConstructorAccessorImpl");
ClassLoader.getSystemClassLoader().loadClass("jdk.internal.reflect.DelegatingConstructorAccessorImpl");
}catch(Throwable ignored){
ver="jdk.internal.reflect"; // If class can't be found in sun.reflect; we know that user is running Java 9+
ver="sun.reflect"; // If class can't be found in sun.reflect; we know that user is running Java 9+
}
try{
Class<?> c;
f = Unsafe.class.getDeclaredField("theUnsafe");
f.setAccessible(true);
u = (Unsafe) f.get(null);
m = Class.forName(ver+".DelegatingConstructorAccessorImpl").getDeclaredMethod("newInstance", Object[].class);
u.putBoolean(m, l=u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")), true);
m1 = Constructor.class.getDeclaredMethod("acquireConstructorAccessor");
m1.setAccessible(true);
m2 = Field.class.getDeclaredMethod("getFieldAccessor", Object.class);
m2.setAccessible(true);
m3 = (c=Class.forName(ver+".UnsafeQualifiedStaticObjectFieldAccessorImpl")).getDeclaredMethod("set", Object.class, Object.class);
u.putBoolean(m3, l, true);
f = c.getSuperclass().getDeclaredField("isReadOnly");
l=u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override"));
try {
m1 = Constructor.class.getDeclaredMethod("acquireConstructorAccessor");
m1.setAccessible(true);
m = Class.forName(ver + ".DelegatingConstructorAccessorImpl").getDeclaredMethod("newInstance", Object[].class);
u.putBoolean(m, l, true);
}catch(Exception e){
b = false;
}
u.putBoolean(f, l, true);
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception is never thrown
unsafe = u;
newInstance = m;
aConAccess = m1;
gFieldAccess = m2;
fieldAccess = m3;
ro = f;
override = l;
version = ver;
hasAConAccess = b;
}
/**
@ -239,14 +233,12 @@ public final class SafeReflection {
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
for(Class c : def.params.values()) paramList[--iterator] = c; // The stuff is reversed
// Get enum constructor (inherited from Enum.class)
Constructor c = clazz.getDeclaredConstructor(paramList);
unsafe.putBoolean(c, override, true);
// Get constructor accessor since Constructor.newInstance throws an exception because Enums are "immutable"
Object access = aConAccess.invoke(c);
Constructor<T> c = clazz.getDeclaredConstructor(paramList);
if(hasAConAccess) unsafe.putBoolean(c, override, true);
else c.setAccessible(true);
Object[] parameters = new Object[def.params.size()+2];
parameters[0] = name;
@ -255,7 +247,8 @@ public final class SafeReflection {
for(Object o : def.params.keySet()) parameters[--iterator] = o;
// Create new instance of enum with valid name and ordinal
u = (T) newInstance.invoke(access, (Object) parameters);
if(hasAConAccess) u = (T) newInstance.invoke(aConAccess.invoke(c), (Object) parameters);
else u = c.newInstance(parameters);
// Get the final name field from Enum.class and make it temporarily modifiable
Field f = Enum.class.getDeclaredField("name");
@ -266,26 +259,14 @@ public final class SafeReflection {
if(!addToValuesArray) return u; // Stops here if
// Get the current values field from Enum (a bitch to modify)
// Get the current values field from Enum (a female dog 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
Object UQSOFAImpl = gFieldAccess.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
ro.setBoolean(UQSOFAImpl, false);
// Invoke set() method on UnsafeQualifiedStaticObjectFieldAccessorImpl object which sets the
// private field $VALUES to our new array
System.out.println(UQSOFAImpl.getClass());
fieldAccess.invoke(UQSOFAImpl, f, $VALUES);
unsafe.putObject(clazz, unsafe.staticFieldOffset(f), $VALUES);
} catch (Exception wrongParams) { throw new RuntimeException(wrongParams); }
return u;
}