blob: be4012f0e9e2e66635f05b0074b22619d49c3702 [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.
*/
package com.android.quicksearchbox.util;
import android.util.Log;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadFactory;
/**
* Executor that uses a single thread and an unbounded work queue.
*/
public class SingleThreadNamedTaskExecutor implements NamedTaskExecutor {
private static final boolean DBG = false;
private static final String TAG = "QSB.SingleThreadNamedTaskExecutor";
private final LinkedBlockingQueue<NamedTask> mQueue;
private final Thread mWorker;
private volatile boolean mClosed = false;
public SingleThreadNamedTaskExecutor(ThreadFactory threadFactory) {
mQueue = new LinkedBlockingQueue<NamedTask>();
mWorker = threadFactory.newThread(new Worker());
mWorker.start();
}
public void cancelPendingTasks() {
if (DBG) Log.d(TAG, "Cancelling " + mQueue.size() + " tasks: " + mWorker.getName());
if (mClosed) {
throw new IllegalStateException("cancelPendingTasks() after close()");
}
mQueue.clear();
}
public void close() {
mClosed = true;
mWorker.interrupt();
mQueue.clear();
}
public void execute(NamedTask task) {
if (mClosed) {
throw new IllegalStateException("execute() after close()");
}
mQueue.add(task);
}
private class Worker implements Runnable {
public void run() {
try {
loop();
} finally {
if (!mClosed) Log.w(TAG, "Worker exited before close");
}
}
private void loop() {
Thread currentThread = Thread.currentThread();
String threadName = currentThread.getName();
while (!mClosed) {
NamedTask task;
try {
task = mQueue.take();
} catch (InterruptedException ex) {
continue;
}
currentThread.setName(threadName + " " + task.getName());
try {
if (DBG) Log.d(TAG, "Running task " + task.getName());
task.run();
if (DBG) Log.d(TAG, "Task " + task.getName() + " complete");
} catch (RuntimeException ex) {
Log.e(TAG, "Task " + task.getName() + " failed", ex);
}
}
}
}
public static Factory<NamedTaskExecutor> factory(final ThreadFactory threadFactory) {
return new Factory<NamedTaskExecutor>() {
public NamedTaskExecutor create() {
return new SingleThreadNamedTaskExecutor(threadFactory);
}
};
}
}