Java程序  |  88行  |  3.31 KB

/*
 * Copyright (C) 2013 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.*;
import java.lang.Runtime;

public class Main {
    static Object nativeLock = new Object();
    static Object deadlockLock = new Object();
    static boolean aboutToDeadlockLock = false;
    static int nativeBytes = 0;
    static Object runtime;
    static Method register_native_allocation;
    static Method register_native_free;
    static long maxMem = 0;

    static class NativeAllocation {
        private int bytes;

        NativeAllocation(int bytes, boolean testingDeadlock) throws Exception {
            this.bytes = bytes;
            register_native_allocation.invoke(runtime, bytes);
            synchronized (nativeLock) {
                if (!testingDeadlock) {
                    nativeBytes += bytes;
                    if (nativeBytes > maxMem) {
                        throw new OutOfMemoryError();
                    }
                }
            }
        }

        protected void finalize() throws Exception {
            synchronized (nativeLock) {
                nativeBytes -= bytes;
            }
            register_native_free.invoke(runtime, bytes);
            aboutToDeadlockLock = true;
            synchronized (deadlockLock) {
            }
        }
    }

    public static void main(String[] args) throws Exception {
        Class<?> vm_runtime = Class.forName("dalvik.system.VMRuntime");
        Method get_runtime = vm_runtime.getDeclaredMethod("getRuntime");
        runtime = get_runtime.invoke(null);
        register_native_allocation = vm_runtime.getDeclaredMethod("registerNativeAllocation", Integer.TYPE);
        register_native_free = vm_runtime.getDeclaredMethod("registerNativeFree", Integer.TYPE);
        maxMem = Runtime.getRuntime().maxMemory();
        int count = 16;
        int size = (int)(maxMem / 2 / count);
        int allocation_count = 256;
        NativeAllocation[] allocations = new NativeAllocation[count];
        for (int i = 0; i < allocation_count; ++i) {
            allocations[i % count] = new NativeAllocation(size, false);
        }
        // Test that we don't get a deadlock if we are holding nativeLock. If there is no timeout,
        // then we will get a finalizer timeout exception.
        aboutToDeadlockLock = false;
        synchronized (deadlockLock) {
            for (int i = 0; aboutToDeadlockLock != true; ++i) {
                allocations[i % count] = new NativeAllocation(size, true);
            }
            // Do more allocations now that the finalizer thread is deadlocked so that we force
            // finalization and timeout. 
            for (int i = 0; i < 10; ++i) {
                allocations[i % count] = new NativeAllocation(size, true);
            }
        }
        System.out.println("Test complete");
    }
}