one.nio.util.JavaInternals Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of one-nio Show documentation
Show all versions of one-nio Show documentation
Unconventional Java I/O library
The newest version!
/*
* Copyright 2015 Odnoklassniki Ltd, Mail.Ru Group
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package one.nio.util;
import sun.misc.Unsafe;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
public final class JavaInternals {
public static final Unsafe unsafe = getUnsafe();
public static final long byteArrayOffset = unsafe.arrayBaseOffset(byte[].class);
private static final boolean hasModules = !System.getProperty("java.version").startsWith("1.");
private static final long accessibleOffset = getAccessibleOffset();
public static Unsafe getUnsafe() {
try {
return (Unsafe) getField(Unsafe.class, "theUnsafe").get(null);
} catch (IllegalAccessException e) {
throw new IllegalStateException(e);
}
}
public static boolean hasModules() {
return hasModules;
}
private static long getAccessibleOffset() {
if (hasModules) {
try {
Method m0 = JavaInternals.class.getDeclaredMethod("getAccessibleOffset");
Method m1 = JavaInternals.class.getDeclaredMethod("getAccessibleOffset");
m1.setAccessible(true);
// Work around JDK 17 restrictions on calling setAccessible()
for (long offset = 8; offset < 128; offset++) {
if (unsafe.getByte(m0, offset) == 0 && unsafe.getByte(m1, offset) == 1) {
return offset;
}
}
} catch (Exception e) {
// Fall back to default setAccessible() implementation
}
}
return 0;
}
public static Class> getClass(String className) {
try {
return Class.forName(className);
} catch (ClassNotFoundException e) {
return null;
}
}
public static Field getField(Class> cls, String name) {
Field f = findField(cls, name);
if (f != null) setAccessible(f);
return f;
}
public static Field findField(Class> cls, String name) {
try {
return cls.getDeclaredField(name);
} catch (NoSuchFieldException e) {
return null;
}
}
public static Field findField(String cls, String name) {
try {
return findField(Class.forName(cls), name);
} catch (ClassNotFoundException e) {
return null;
}
}
public static Field findFieldRecursively(Class> cls, String name) {
for (; cls != null; cls = cls.getSuperclass()) {
Field f = findField(cls, name);
if (f != null) {
return f;
}
}
return null;
}
public static Method getMethod(Class> cls, String name, Class>... params) {
Method m = findMethod(cls, name, params);
if (m != null) setAccessible(m);
return m;
}
public static Method getMethodRecursively(Class> cls, String name, Class>... params) {
Method m = findMethodRecursively(cls, name, params);
if (m != null) setAccessible(m);
return m;
}
public static Method findMethod(Class> cls, String name, Class>... params) {
try {
return cls.getDeclaredMethod(name, params);
} catch (NoSuchMethodException e) {
return null;
}
}
public static Method findMethod(String cls, String name, Class>... params) {
try {
return findMethod(Class.forName(cls), name, params);
} catch (ClassNotFoundException e) {
return null;
}
}
public static Method findMethodRecursively(Class> cls, String name, Class>... params) {
for (; cls != null; cls = cls.getSuperclass()) {
Method m = findMethod(cls, name, params);
if (m != null) {
return m;
}
}
return null;
}
public static Constructor getConstructor(Class cls, Class>... params) {
Constructor c = findConstructor(cls, params);
if (c != null) setAccessible(c);
return c;
}
public static Constructor findConstructor(Class cls, Class>... params) {
try {
return cls.getDeclaredConstructor(params);
} catch (NoSuchMethodException e) {
return null;
}
}
public static Constructor> findConstructor(String cls, Class>... params) {
try {
return findConstructor(Class.forName(cls), params);
} catch (ClassNotFoundException e) {
return null;
}
}
public static void setAccessible(AccessibleObject ao) {
if (accessibleOffset != 0) {
unsafe.putByte(ao, accessibleOffset, (byte) 1);
} else {
ao.setAccessible(true);
}
}
public static long fieldOffset(Class> cls, String name) {
try {
return unsafe.objectFieldOffset(cls.getDeclaredField(name));
} catch (NoSuchFieldException e) {
throw new IllegalStateException(e);
}
}
public static long fieldOffset(String cls, String name) {
try {
return fieldOffset(Class.forName(cls), name);
} catch (ClassNotFoundException e) {
throw new IllegalStateException(e);
}
}
// Useful for patching final fields
public static void setStaticField(Class> cls, String name, Object value) {
try {
Field field = cls.getDeclaredField(name);
if (!Modifier.isStatic(field.getModifiers())) {
throw new IllegalArgumentException("Static field expected");
}
unsafe.putObject(unsafe.staticFieldBase(field), unsafe.staticFieldOffset(field), value);
} catch (NoSuchFieldException e) {
throw new IllegalStateException(e);
}
}
// Useful for patching final fields
public static void setObjectField(Object obj, String name, Object value) {
Class> cls = obj.getClass();
setObjectField(obj, cls, name, value);
}
// Useful for patching final fields
public static void setObjectField(Object obj, Class> cls, String name, Object value) {
try {
Field field = cls.getDeclaredField(name);
if (Modifier.isStatic(field.getModifiers())) {
throw new IllegalArgumentException("Object field expected");
}
unsafe.putObject(obj, unsafe.objectFieldOffset(field), value);
} catch (NoSuchFieldException e) {
throw new IllegalStateException(e);
}
}
@SuppressWarnings("unchecked")
public static void uncheckedThrow(Throwable e) throws E {
throw (E) e;
}
// Helpers to be called from generated code
public static void putObject(Object o, Object value, long offset) {
unsafe.putObject(o, offset, value);
}
public static void putInt(Object o, int value, long offset) {
unsafe.putInt(o, offset, value);
}
public static void putLong(Object o, long value, long offset) {
unsafe.putLong(o, offset, value);
}
public static void putBoolean(Object o, boolean value, long offset) {
unsafe.putBoolean(o, offset, value);
}
public static void putByte(Object o, byte value, long offset) {
unsafe.putByte(o, offset, value);
}
public static void putShort(Object o, short value, long offset) {
unsafe.putShort(o, offset, value);
}
public static void putChar(Object o, char value, long offset) {
unsafe.putChar(o, offset, value);
}
public static void putFloat(Object o, float value, long offset) {
unsafe.putFloat(o, offset, value);
}
public static void putDouble(Object o, double value, long offset) {
unsafe.putDouble(o, offset, value);
}
}