blob: c9fd3610ff84f6e1c9c2197ec9e8b68af9a0a518 [file] [log] [blame]
/*
* Copyright (C) 2009 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.InvocationTargetException;
import java.lang.reflect.Method;
public class Main {
static class ArrayMemEater {
static boolean sawOome;
static void blowup(char[][] holder) {
try {
for (int i = 0; i < holder.length; ++i) {
holder[i] = new char[1024 * 1024];
}
} catch (OutOfMemoryError oome) {
ArrayMemEater.sawOome = true;
}
}
}
static class InstanceMemEater {
static boolean sawOome;
static InstanceMemEater hook;
InstanceMemEater next;
double d1, d2, d3, d4, d5, d6, d7, d8; // Bloat this object so we fill the heap faster.
static InstanceMemEater allocate() {
try {
return new InstanceMemEater();
} catch (OutOfMemoryError e) {
InstanceMemEater.sawOome = true;
return null;
}
}
static void confuseCompilerOptimization(InstanceMemEater instance) {
hook = instance;
}
}
private static int exhaustJavaHeap(Object[] data, int index, int size) {
Runtime.getRuntime().gc();
while (index != data.length) {
try {
data[index] = new byte[size];
++index;
} catch (OutOfMemoryError oome) {
// Rapidly shrink the object size to fill any remaining space.
// Use few different sizes, since detecting out-of-memory is slow.
if (size >= 32) {
size /= 32;
} else if (size > 1) {
size = 1;
} else {
break;
}
}
}
return index;
}
public static Object eatAllMemory() {
Object[] result = null;
int size = 1000000;
// Make sure that there is no reclaimable memory in the heap. Otherwise we may throw
// OOME to prevent GC thrashing, even if later allocations may succeed.
Runtime.getRuntime().gc();
System.runFinalization();
// NOTE: There is a GC invocation in exhaustJavaHeap. So we don't need one here.
while (result == null && size != 0) {
try {
result = new Object[size];
} catch (OutOfMemoryError oome) {
size /= 2;
}
}
if (result != null) {
int index = 0;
// Repeat to ensure there is no space left on the heap.
index = exhaustJavaHeap(result, index, size);
index = exhaustJavaHeap(result, index, /*size*/ 4);
index = exhaustJavaHeap(result, index, /*size*/ 4);
}
return result;
}
static boolean triggerArrayOOM() {
ArrayMemEater.blowup(new char[128 * 1024][]);
return ArrayMemEater.sawOome;
}
static boolean triggerInstanceOOM() {
InstanceMemEater memEater = InstanceMemEater.allocate();
InstanceMemEater lastMemEater = memEater;
do {
lastMemEater.next = InstanceMemEater.allocate();
lastMemEater = lastMemEater.next;
} while (lastMemEater != null);
memEater.confuseCompilerOptimization(memEater);
InstanceMemEater.hook = null;
return InstanceMemEater.sawOome;
}
public static void main(String[] args) {
if (triggerReflectionOOM()) {
System.out.println("Test reflection correctly threw");
}
if (triggerReflectionOOM2()) {
System.out.println("Test reflection2 correctly threw");
}
if (triggerArrayOOM()) {
System.out.println("NEW_ARRAY correctly threw OOME");
}
if (triggerInstanceOOM()) {
System.out.println("NEW_INSTANCE correctly threw OOME");
}
}
static Object[] holder;
public static void blowup() throws Exception {
int size = 2 * 1024 * 1024;
for (int i = 0; i < holder.length; ) {
try {
holder[i] = new char[size];
i++;
} catch (OutOfMemoryError oome) {
size = size / 16;
if (size == 0) {
break;
}
}
}
holder[0] = new char[100000];
}
static boolean triggerReflectionOOM() {
try {
Class<?> c = Main.class;
Method m = c.getMethod("blowup");
holder = new Object[1000000];
m.invoke(null);
holder = null;
System.out.println("Didn't throw from blowup");
} catch (OutOfMemoryError e) {
holder = null;
} catch (InvocationTargetException e) {
holder = null;
if (!(e.getCause() instanceof OutOfMemoryError)) {
System.out.println("InvocationTargetException cause not OOME " + e.getCause());
return false;
}
} catch (Exception e) {
holder = null;
System.out.println("Unexpected exception " + e);
return false;
}
return true;
}
static boolean triggerReflectionOOM2() {
Object memory = eatAllMemory();
boolean result = false;
try {
Main.class.getDeclaredMethods();
} catch (OutOfMemoryError e) {
result = true;
}
if (!result) {
boolean memoryWasAllocated = (memory != null);
memory = null;
System.out.println("memoryWasAllocated = " + memoryWasAllocated);
}
return result;
}
}