blob: d7930561da9ca20070a3954ad23e82fc1ad5a0f7 [file] [log] [blame]
/*
* Copyright 2000-2012 JetBrains s.r.o.
*
* 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 com.intellij.util;
import com.intellij.openapi.Disposable;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.util.Disposer;
import org.jetbrains.annotations.Nullable;
import java.util.Collections;
import java.util.Map;
import java.util.WeakHashMap;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
/**
* @author mike
*/
@SuppressWarnings({"NonPrivateFieldAccessedInSynchronizedContext"})
abstract class Timed<T> implements Disposable {
private static final Logger LOG = Logger.getInstance("#com.intellij.util.Timed");
private static final Map<Timed, Boolean> ourReferences = Collections.synchronizedMap(new WeakHashMap<Timed, Boolean>());
protected static final int SERVICE_DELAY = 60;
int myLastCheckedAccessCount;
int myAccessCount;
protected T myT;
boolean myPolled;
protected Timed(@Nullable final Disposable parentDisposable) {
if (parentDisposable != null) {
Disposer.register(parentDisposable, this);
}
}
@Override
public synchronized void dispose() {
final Object t = myT;
myT = null;
if (t instanceof Disposable) {
Disposable disposable = (Disposable)t;
Disposer.dispose(disposable);
}
remove();
}
protected final void poll() {
if (!myPolled) {
ourReferences.put(this, Boolean.TRUE);
myPolled = true;
}
}
protected final void remove() {
ourReferences.remove(this);
myPolled = false;
}
protected synchronized boolean isLocked() {
return false;
}
protected synchronized boolean checkLocked() {
return isLocked();
}
static {
ScheduledExecutorService service = ConcurrencyUtil.newSingleScheduledThreadExecutor("timed reference disposer", Thread.MIN_PRIORITY + 1);
service.scheduleWithFixedDelay(new Runnable() {
@Override
public void run() {
try {
disposeTimed();
}
catch (Throwable e) {
LOG.error(e);
}
}
}, SERVICE_DELAY, SERVICE_DELAY, TimeUnit.SECONDS);
}
static void disposeTimed() {
final Timed[] references = ourReferences.keySet().toArray(new Timed[ourReferences.size()]);
for (Timed timed : references) {
if (timed == null) continue;
synchronized (timed) {
if (timed.myLastCheckedAccessCount == timed.myAccessCount && !timed.checkLocked()) {
timed.dispose();
}
else {
timed.myLastCheckedAccessCount = timed.myAccessCount;
}
}
}
}
}