blob: aadf1f6893dd743abf59ed69a5cb3ce31b039cd2 [file] [log] [blame]
/*
* Copyright (C) 2011 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.
*/
package java.lang.ref;
/**
* @hide
*/
public final class FinalizerReference<T> extends Reference<T> {
public static final ReferenceQueue<Object> queue = new ReferenceQueue<Object>();
private static FinalizerReference head = null;
private T zombie;
private FinalizerReference prev;
private FinalizerReference next;
public FinalizerReference(T r, ReferenceQueue<? super T> q) {
super(r, q);
}
@Override
public T get() {
return zombie;
}
@Override
public void clear() {
zombie = null;
}
static void add(Object referent) {
FinalizerReference<?> reference = new FinalizerReference<Object>(referent, queue);
synchronized (FinalizerReference.class) {
reference.prev = null;
reference.next = head;
if (head != null) {
head.prev = reference;
}
head = reference;
}
}
public static void remove(FinalizerReference reference) {
synchronized (FinalizerReference.class) {
FinalizerReference next = reference.next;
FinalizerReference prev = reference.prev;
reference.next = null;
reference.prev = null;
if (prev != null) {
prev.next = next;
} else {
head = next;
}
if (next != null) {
next.prev = prev;
}
}
}
/**
* Returns once all currently-enqueued references have been finalized.
*/
public static void finalizeAllEnqueued() throws InterruptedException {
Sentinel sentinel = new Sentinel();
FinalizerReference<Object> reference = new FinalizerReference<Object>(null, queue);
reference.zombie = sentinel;
reference.enqueueInternal();
sentinel.awaitFinalization();
}
/**
* A marker object that we can immediately enqueue. When this object's
* finalize() method is called, we know all previously-enqueued finalizable
* references have been finalized.
*
* <p>Each instance of this class will be finalized twice as it is enqueued
* directly and by the garbage collector.
*/
private static class Sentinel {
boolean finalized = false;
@Override protected synchronized void finalize() throws Throwable {
finalized = true;
notifyAll();
}
synchronized void awaitFinalization() throws InterruptedException {
while (!finalized) {
wait();
}
}
}
}