blob: ad6202a06a68d0168d4ff34b0c1049710bd9ab00 [file] [log] [blame]
/*
* Copyright (C) 2021 The Android Open Source Project
*
* 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.
*/
import java.lang.reflect.Field;
import jdk.internal.misc.Unsafe;
public class Main {
private static void check(int actual, int expected, String msg) {
if (actual != expected) {
System.out.println(msg + " : " + actual + " != " + expected);
System.exit(1);
}
}
private static void check(long actual, long expected, String msg) {
if (actual != expected) {
System.out.println(msg + " : " + actual + " != " + expected);
System.exit(1);
}
}
private static void check(float actual, float expected, String msg) {
if (actual != expected) {
System.out.println(msg + " : " + actual + " != " + expected);
System.exit(1);
}
}
private static void check(double actual, double expected, String msg) {
if (actual != expected) {
System.out.println(msg + " : " + actual + " != " + expected);
System.exit(1);
}
}
private static void check(Object actual, Object expected, String msg) {
if (actual != expected) {
System.out.println(msg + " : " + actual + " != " + expected);
System.exit(1);
}
}
private static void expectThrow(Class<?> exceptionClass, String msg) {
System.out.println(msg + " : expected " + exceptionClass.getName());
System.exit(1);
}
private static Unsafe getUnsafe() throws NoSuchFieldException, IllegalAccessException {
Class<?> unsafeClass = Unsafe.class;
Field f = unsafeClass.getDeclaredField("theUnsafe");
f.setAccessible(true);
return (Unsafe) f.get(null);
}
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
System.loadLibrary(args[0]);
Unsafe unsafe = getUnsafe();
testArrayBaseOffset(unsafe);
testArrayIndexScale(unsafe);
testGetAndPutAndCAS(unsafe);
testCompareAndSet(unsafe);
testGetAndPutVolatile(unsafe);
testGetAcquireAndPutRelease(unsafe);
testCopyMemory(unsafe);
}
private static void testArrayBaseOffset(Unsafe unsafe) {
check(unsafe.arrayBaseOffset(boolean[].class), vmJdkArrayBaseOffset(boolean[].class),
"Unsafe.arrayBaseOffset(boolean[])");
check(unsafe.arrayBaseOffset(byte[].class), vmJdkArrayBaseOffset(byte[].class),
"Unsafe.arrayBaseOffset(byte[])");
check(unsafe.arrayBaseOffset(char[].class), vmJdkArrayBaseOffset(char[].class),
"Unsafe.arrayBaseOffset(char[])");
check(unsafe.arrayBaseOffset(double[].class), vmJdkArrayBaseOffset(double[].class),
"Unsafe.arrayBaseOffset(double[])");
check(unsafe.arrayBaseOffset(float[].class), vmJdkArrayBaseOffset(float[].class),
"Unsafe.arrayBaseOffset(float[])");
check(unsafe.arrayBaseOffset(int[].class), vmJdkArrayBaseOffset(int[].class),
"Unsafe.arrayBaseOffset(int[])");
check(unsafe.arrayBaseOffset(long[].class), vmJdkArrayBaseOffset(long[].class),
"Unsafe.arrayBaseOffset(long[])");
check(unsafe.arrayBaseOffset(Object[].class), vmJdkArrayBaseOffset(Object[].class),
"Unsafe.arrayBaseOffset(Object[])");
}
private static void testArrayIndexScale(Unsafe unsafe) {
check(unsafe.arrayIndexScale(boolean[].class), vmJdkArrayIndexScale(boolean[].class),
"Unsafe.arrayIndexScale(boolean[])");
check(unsafe.arrayIndexScale(byte[].class), vmJdkArrayIndexScale(byte[].class),
"Unsafe.arrayIndexScale(byte[])");
check(unsafe.arrayIndexScale(char[].class), vmJdkArrayIndexScale(char[].class),
"Unsafe.arrayIndexScale(char[])");
check(unsafe.arrayIndexScale(double[].class), vmJdkArrayIndexScale(double[].class),
"Unsafe.arrayIndexScale(double[])");
check(unsafe.arrayIndexScale(float[].class), vmJdkArrayIndexScale(float[].class),
"Unsafe.arrayIndexScale(float[])");
check(unsafe.arrayIndexScale(int[].class), vmJdkArrayIndexScale(int[].class),
"Unsafe.arrayIndexScale(int[])");
check(unsafe.arrayIndexScale(long[].class), vmJdkArrayIndexScale(long[].class),
"Unsafe.arrayIndexScale(long[])");
check(unsafe.arrayIndexScale(Object[].class), vmJdkArrayIndexScale(Object[].class),
"Unsafe.arrayIndexScale(Object[])");
}
private static void testGetAndPutAndCAS(Unsafe unsafe) throws NoSuchFieldException {
TestClass t = new TestClass();
int intValue = 12345678;
Field intField = TestClass.class.getDeclaredField("intVar");
long intOffset = unsafe.objectFieldOffset(intField);
check(unsafe.getInt(t, intOffset), 0, "Unsafe.getInt(Object, long) - initial");
unsafe.putInt(t, intOffset, intValue);
check(t.intVar, intValue, "Unsafe.putInt(Object, long, int)");
check(unsafe.getInt(t, intOffset), intValue, "Unsafe.getInt(Object, long)");
long longValue = 1234567887654321L;
Field longField = TestClass.class.getDeclaredField("longVar");
long longOffset = unsafe.objectFieldOffset(longField);
check(unsafe.getLong(t, longOffset), 0, "Unsafe.getLong(Object, long) - initial");
unsafe.putLong(t, longOffset, longValue);
check(t.longVar, longValue, "Unsafe.putLong(Object, long, long)");
check(unsafe.getLong(t, longOffset), longValue, "Unsafe.getLong(Object, long)");
Object objectValue = new Object();
Field objectField = TestClass.class.getDeclaredField("objectVar");
long objectOffset = unsafe.objectFieldOffset(objectField);
check(unsafe.getObject(t, objectOffset), null, "Unsafe.getObject(Object, long) - initial");
unsafe.putObject(t, objectOffset, objectValue);
check(t.objectVar, objectValue, "Unsafe.putObject(Object, long, Object)");
check(unsafe.getObject(t, objectOffset), objectValue, "Unsafe.getObject(Object, long)");
if (unsafe.compareAndSwapInt(t, intOffset, 0, 1)) {
System.out.println("Unexpectedly succeeding compareAndSwapInt(t, intOffset, 0, 1)");
}
if (!unsafe.compareAndSwapInt(t, intOffset, intValue, 0)) {
System.out.println(
"Unexpectedly not succeeding compareAndSwapInt(t, intOffset, intValue, 0)");
}
if (!unsafe.compareAndSwapInt(t, intOffset, 0, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSwapInt(t, intOffset, 0, 1)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapInt using the same
// integer (1) for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSwapInt(t, intOffset, 1, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSwapInt(t, intOffset, 1, 1)");
}
if (unsafe.compareAndSwapLong(t, longOffset, 0, 1)) {
System.out.println("Unexpectedly succeeding compareAndSwapLong(t, longOffset, 0, 1)");
}
if (!unsafe.compareAndSwapLong(t, longOffset, longValue, 0)) {
System.out.println(
"Unexpectedly not succeeding compareAndSwapLong(t, longOffset, longValue, 0)");
}
if (!unsafe.compareAndSwapLong(t, longOffset, 0, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSwapLong(t, longOffset, 0, 1)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapLong using the same
// integer (1) for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSwapLong(t, longOffset, 1, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSwapLong(t, longOffset, 1, 1)");
}
// We do not use `null` as argument to jdk.internal.misc.Unsafe.compareAndSwapObject
// in those tests, as this value is not affected by heap poisoning
// (which uses address negation to poison and unpoison heap object
// references). This way, when heap poisoning is enabled, we can
// better exercise its implementation within that method.
if (unsafe.compareAndSwapObject(t, objectOffset, new Object(), new Object())) {
System.out.println("Unexpectedly succeeding " +
"compareAndSwapObject(t, objectOffset, new Object(), new Object())");
}
Object objectValue2 = new Object();
if (!unsafe.compareAndSwapObject(t, objectOffset, objectValue, objectValue2)) {
System.out.println("Unexpectedly not succeeding " +
"compareAndSwapObject(t, objectOffset, objectValue, objectValue2)");
}
Object objectValue3 = new Object();
if (!unsafe.compareAndSwapObject(t, objectOffset, objectValue2, objectValue3)) {
System.out.println("Unexpectedly not succeeding " +
"compareAndSwapObject(t, objectOffset, objectValue2, objectValue3)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapObject using the same
// object (`objectValue3`) for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSwapObject(t, objectOffset, objectValue3, objectValue3)) {
System.out.println("Unexpectedly not succeeding " +
"compareAndSwapObject(t, objectOffset, objectValue3, objectValue3)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapObject using the same
// object (`t`) for the `obj` and `newValue` arguments.
if (!unsafe.compareAndSwapObject(t, objectOffset, objectValue3, t)) {
System.out.println(
"Unexpectedly not succeeding compareAndSwapObject(t, objectOffset, objectValue3, t)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapObject using the same
// object (`t`) for the `obj`, `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSwapObject(t, objectOffset, t, t)) {
System.out.println("Unexpectedly not succeeding compareAndSwapObject(t, objectOffset, t, t)");
}
// Exercise jdk.internal.misc.Unsafe.compareAndSwapObject using the same
// object (`t`) for the `obj` and `expectedValue` arguments.
if (!unsafe.compareAndSwapObject(t, objectOffset, t, new Object())) {
System.out.println(
"Unexpectedly not succeeding compareAndSwapObject(t, objectOffset, t, new Object())");
}
}
private static void testCompareAndSet(Unsafe unsafe) throws NoSuchFieldException {
TestClass t = new TestClass();
int intValue = 12345678;
Field intField = TestClass.class.getDeclaredField("intVar");
long intOffset = unsafe.objectFieldOffset(intField);
unsafe.putInt(t, intOffset, intValue);
long longValue = 1234567887654321L;
Field longField = TestClass.class.getDeclaredField("longVar");
long longOffset = unsafe.objectFieldOffset(longField);
unsafe.putLong(t, longOffset, longValue);
Object objectValue = new Object();
Field objectField = TestClass.class.getDeclaredField("objectVar");
long objectOffset = unsafe.objectFieldOffset(objectField);
unsafe.putObject(t, objectOffset, objectValue);
if (unsafe.compareAndSetInt(t, intOffset, 0, 1)) {
System.out.println("Unexpectedly succeeding compareAndSetInt(t, intOffset, 0, 1)");
}
check(t.intVar, intValue, "Unsafe.compareAndSetInt(Object, long, int, int) - not set");
if (!unsafe.compareAndSetInt(t, intOffset, intValue, 0)) {
System.out.println(
"Unexpectedly not succeeding compareAndSetInt(t, intOffset, intValue, 0)");
}
check(t.intVar, 0, "Unsafe.compareAndSetInt(Object, long, int, int) - gets set");
if (!unsafe.compareAndSetInt(t, intOffset, 0, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSetInt(t, intOffset, 0, 1)");
}
check(t.intVar, 1, "Unsafe.compareAndSetInt(Object, long, int, int) - gets re-set");
// Exercise jdk.internal.misc.Unsafe.compareAndSetInt using the same
// integer (1) for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSetInt(t, intOffset, 1, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSetInt(t, intOffset, 1, 1)");
}
check(t.intVar, 1, "Unsafe.compareAndSetInt(Object, long, int, int) - gets set to same");
if (unsafe.compareAndSetLong(t, longOffset, 0, 1)) {
System.out.println("Unexpectedly succeeding compareAndSetLong(t, longOffset, 0, 1)");
}
check(t.longVar, longValue, "Unsafe.compareAndSetLong(Object, long, long, long) - not set");
if (!unsafe.compareAndSetLong(t, longOffset, longValue, 0)) {
System.out.println(
"Unexpectedly not succeeding compareAndSetLong(t, longOffset, longValue, 0)");
}
check(t.longVar, 0, "Unsafe.compareAndSetLong(Object, long, long, long) - gets set");
if (!unsafe.compareAndSetLong(t, longOffset, 0, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSetLong(t, longOffset, 0, 1)");
}
check(t.longVar, 1, "Unsafe.compareAndSetLong(Object, long, long, long) - gets re-set");
// Exercise jdk.internal.misc.Unsafe.compareAndSetLong using the same
// integer (1) for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSetLong(t, longOffset, 1, 1)) {
System.out.println("Unexpectedly not succeeding compareAndSetLong(t, longOffset, 1, 1)");
}
check(t.longVar, 1, "Unsafe.compareAndSetLong(Object, long, long, long) - gets set to same");
// We do not use `null` as argument to jdk.internal.misc.Unsafe.compareAndSwapObject
// in those tests, as this value is not affected by heap poisoning
// (which uses address negation to poison and unpoison heap object
// references). This way, when heap poisoning is enabled, we can
// better exercise its implementation within that method.
if (unsafe.compareAndSetObject(t, objectOffset, new Object(), new Object())) {
System.out.println("Unexpectedly succeeding compareAndSetObject(t, objectOffset, 0, 1)");
}
check(t.objectVar, objectValue, "Unsafe.compareAndSetObject(Object, long, Object, Object) - not set");
Object objectValue2 = new Object();
if (!unsafe.compareAndSetObject(t, objectOffset, objectValue, objectValue2)) {
System.out.println(
"Unexpectedly not succeeding compareAndSetObject(t, objectOffset, objectValue, 0)");
}
check(t.objectVar, objectValue2, "Unsafe.compareAndSetObject(Object, long, Object, Object) - gets set");
Object objectValue3 = new Object();
if (!unsafe.compareAndSetObject(t, objectOffset, objectValue2, objectValue3)) {
System.out.println("Unexpectedly not succeeding compareAndSetObject(t, objectOffset, 0, 1)");
}
check(t.objectVar, objectValue3, "Unsafe.compareAndSetObject(Object, long, Object, Object) - gets re-set");
// Exercise jdk.internal.misc.Unsafe.compareAndSetObject using the same
// object for the `expectedValue` and `newValue` arguments.
if (!unsafe.compareAndSetObject(t, objectOffset, objectValue3, objectValue3)) {
System.out.println("Unexpectedly not succeeding compareAndSetObject(t, objectOffset, 1, 1)");
}
check(t.objectVar, objectValue3, "Unsafe.compareAndSetObject(Object, long, Object, Object) - gets set to same");
}
private static void testGetAndPutVolatile(Unsafe unsafe) throws NoSuchFieldException {
TestVolatileClass tv = new TestVolatileClass();
int intValue = 12345678;
Field volatileIntField = TestVolatileClass.class.getDeclaredField("volatileIntVar");
long volatileIntOffset = unsafe.objectFieldOffset(volatileIntField);
check(unsafe.getIntVolatile(tv, volatileIntOffset),
0,
"Unsafe.getIntVolatile(Object, long) - initial");
unsafe.putIntVolatile(tv, volatileIntOffset, intValue);
check(tv.volatileIntVar, intValue, "Unsafe.putIntVolatile(Object, long, int)");
check(unsafe.getIntVolatile(tv, volatileIntOffset),
intValue,
"Unsafe.getIntVolatile(Object, long)");
long longValue = 1234567887654321L;
Field volatileLongField = TestVolatileClass.class.getDeclaredField("volatileLongVar");
long volatileLongOffset = unsafe.objectFieldOffset(volatileLongField);
check(unsafe.getLongVolatile(tv, volatileLongOffset),
0,
"Unsafe.getLongVolatile(Object, long) - initial");
unsafe.putLongVolatile(tv, volatileLongOffset, longValue);
check(tv.volatileLongVar, longValue, "Unsafe.putLongVolatile(Object, long, long)");
check(unsafe.getLongVolatile(tv, volatileLongOffset),
longValue,
"Unsafe.getLongVolatile(Object, long)");
Object objectValue = new Object();
Field volatileObjectField = TestVolatileClass.class.getDeclaredField("volatileObjectVar");
long volatileObjectOffset = unsafe.objectFieldOffset(volatileObjectField);
check(unsafe.getObjectVolatile(tv, volatileObjectOffset),
null,
"Unsafe.getObjectVolatile(Object, long) - initial");
unsafe.putObjectVolatile(tv, volatileObjectOffset, objectValue);
check(tv.volatileObjectVar, objectValue, "Unsafe.putObjectVolatile(Object, long, Object)");
check(unsafe.getObjectVolatile(tv, volatileObjectOffset),
objectValue,
"Unsafe.getObjectVolatile(Object, long)");
}
private static void testGetAcquireAndPutRelease(Unsafe unsafe) throws NoSuchFieldException {
TestVolatileClass tv = new TestVolatileClass();
int intValue = 12345678;
Field volatileIntField = TestVolatileClass.class.getDeclaredField("volatileIntVar");
long volatileIntOffset = unsafe.objectFieldOffset(volatileIntField);
check(unsafe.getIntAcquire(tv, volatileIntOffset),
0,
"Unsafe.getIntAcquire(Object, long) - initial");
unsafe.putIntRelease(tv, volatileIntOffset, intValue);
check(tv.volatileIntVar, intValue, "Unsafe.putIntRelease(Object, long, int)");
check(unsafe.getIntAcquire(tv, volatileIntOffset),
intValue,
"Unsafe.getIntAcquire(Object, long)");
long longValue = 1234567887654321L;
Field volatileLongField = TestVolatileClass.class.getDeclaredField("volatileLongVar");
long volatileLongOffset = unsafe.objectFieldOffset(volatileLongField);
check(unsafe.getLongAcquire(tv, volatileLongOffset),
0,
"Unsafe.getLongAcquire(Object, long) - initial");
unsafe.putLongRelease(tv, volatileLongOffset, longValue);
check(tv.volatileLongVar, longValue, "Unsafe.putLongRelease(Object, long, long)");
check(unsafe.getLongAcquire(tv, volatileLongOffset),
longValue,
"Unsafe.getLongAcquire(Object, long)");
Object objectValue = new Object();
Field volatileObjectField = TestVolatileClass.class.getDeclaredField("volatileObjectVar");
long volatileObjectOffset = unsafe.objectFieldOffset(volatileObjectField);
check(unsafe.getObjectAcquire(tv, volatileObjectOffset),
null,
"Unsafe.getObjectAcquire(Object, long) - initial");
unsafe.putObjectRelease(tv, volatileObjectOffset, objectValue);
check(tv.volatileObjectVar, objectValue, "Unsafe.putObjectRelease(Object, long, Object)");
check(unsafe.getObjectAcquire(tv, volatileObjectOffset),
objectValue,
"Unsafe.getObjectAcquire(Object, long)");
}
private static void testCopyMemory(Unsafe unsafe) {
final int size = 4 * 1024;
final int intSize = 4;
int[] inputInts = new int[size / intSize];
for (int i = 0; i != inputInts.length; ++i) {
inputInts[i] = ((int)i) + 1;
}
int[] outputInts = new int[size / intSize];
unsafe.copyMemory(inputInts, Unsafe.ARRAY_INT_BASE_OFFSET,
outputInts, Unsafe.ARRAY_INT_BASE_OFFSET,
size);
for (int i = 0; i != inputInts.length; ++i) {
check(inputInts[i], outputInts[i], "unsafe.copyMemory/int");
}
final int longSize = 8;
long[] inputLongs = new long[size / longSize];
for (int i = 0; i != inputLongs.length; ++i) {
inputLongs[i] = ((long)i) + 1L;
}
long[] outputLongs = new long[size / longSize];
unsafe.copyMemory(inputLongs, 0, outputLongs, 0, size);
unsafe.copyMemory(inputLongs, Unsafe.ARRAY_LONG_BASE_OFFSET,
outputLongs, Unsafe.ARRAY_LONG_BASE_OFFSET,
size);
for (int i = 0; i != inputLongs.length; ++i) {
check(inputLongs[i], outputLongs[i], "unsafe.copyMemory/long");
}
final int floatSize = 4;
float[] inputFloats = new float[size / floatSize];
for (int i = 0; i != inputFloats.length; ++i) {
inputFloats[i] = ((float)i) + 0.5f;
}
float[] outputFloats = new float[size / floatSize];
unsafe.copyMemory(inputFloats, 0, outputFloats, 0, size);
unsafe.copyMemory(inputFloats, Unsafe.ARRAY_FLOAT_BASE_OFFSET,
outputFloats, Unsafe.ARRAY_FLOAT_BASE_OFFSET,
size);
for (int i = 0; i != inputFloats.length; ++i) {
check(inputFloats[i], outputFloats[i], "unsafe.copyMemory/float");
}
final int doubleSize = 8;
double[] inputDoubles = new double[size / doubleSize];
for (int i = 0; i != inputDoubles.length; ++i) {
inputDoubles[i] = ((double)i) + 0.5;
}
double[] outputDoubles = new double[size / doubleSize];
unsafe.copyMemory(inputDoubles, Unsafe.ARRAY_DOUBLE_BASE_OFFSET,
outputDoubles, Unsafe.ARRAY_DOUBLE_BASE_OFFSET,
size);
for (int i = 0; i != inputDoubles.length; ++i) {
check(inputDoubles[i], outputDoubles[i], "unsafe.copyMemory/double");
}
// check the version that works with memory pointers
try (TestMemoryPtr srcPtr = new TestMemoryPtr(size);
TestMemoryPtr dstPtr = new TestMemoryPtr(size)) {
// use the integer array to fill the source
unsafe.copyMemory(inputInts, Unsafe.ARRAY_INT_BASE_OFFSET,
null, srcPtr.get(),
size);
unsafe.copyMemory(srcPtr.get(), dstPtr.get(), size);
for (int i = 0; i != size; ++i) {
check(unsafe.getByte(srcPtr.get() + i),
unsafe.getByte(dstPtr.get() + i),
"unsafe.copyMemory/memoryAddress");
}
}
try {
TestClass srcObj = new TestClass();
srcObj.intVar = 12345678;
int[] dstArray = new int[1];
unsafe.copyMemory(srcObj, unsafe.objectFieldOffset(TestClass.class, "intVar"),
dstArray, Unsafe.ARRAY_INT_BASE_OFFSET,
4);
expectThrow(RuntimeException.class, "unsafe.copyMemory/non-array-src");
} catch (RuntimeException expected) {
}
try {
int[] srcArray = { 12345678 };
TestClass dstObj = new TestClass();
unsafe.copyMemory(srcArray, Unsafe.ARRAY_INT_BASE_OFFSET,
dstObj, unsafe.objectFieldOffset(TestClass.class, "intVar"),
4);
expectThrow(RuntimeException.class, "unsafe.copyMemory/non-array-dst");
} catch (RuntimeException expected) {
}
}
private static class TestClass {
public int intVar = 0;
public long longVar = 0;
public Object objectVar = null;
}
private static class TestVolatileClass {
public volatile int volatileIntVar = 0;
public volatile long volatileLongVar = 0;
public volatile Object volatileObjectVar = null;
}
private static class TestMemoryPtr implements AutoCloseable {
private long ptr = 0;
public TestMemoryPtr(int size) {
ptr = jdkUnsafeTestMalloc(size);
}
public long get() {
return ptr;
}
@Override
public void close() {
if (ptr != 0) {
jdkUnsafeTestFree(ptr);
ptr = 0;
}
}
}
private static native int vmJdkArrayBaseOffset(Class<?> clazz);
private static native int vmJdkArrayIndexScale(Class<?> clazz);
private static native long jdkUnsafeTestMalloc(long size);
private static native void jdkUnsafeTestFree(long memory);
}