1adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project/*
2adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
3adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
4adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * you may not use this file except in compliance with the License.
6adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * You may obtain a copy of the License at
7adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
8adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *     http://www.apache.org/licenses/LICENSE-2.0
9adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
10adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * See the License for the specific language governing permissions and
14adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * limitations under the License.
15adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project */
16adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
17adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Projectpackage tests.support;
18adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
19adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Projectimport java.util.LinkedList;
20adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Projectimport java.util.logging.Logger;
21adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
22adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Projectpublic class ThreadPool extends ThreadGroup {
23adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
24adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    private boolean isAlive;
25adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
26adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    private LinkedList<Runnable> taskQueue;
27adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
28adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    private int threadID;
29adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
30adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    private static int threadPoolID;
31adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
32adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    public ThreadPool(int numThreads) {
33adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        super("ThreadPool-" + (threadPoolID++));
34adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        setDaemon(true);
35adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
36adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        isAlive = true;
37adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
38adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        taskQueue = new LinkedList<Runnable>();
39adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        for (int i = 0; i < numThreads; i++) {
40adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            new PooledThread().start();
41adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
42adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
43adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
44adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    public synchronized void runTask(Runnable task) {
45adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        if (!isAlive) {
46adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            throw new IllegalStateException();
47adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
48adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        if (task != null) {
49adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            taskQueue.add(task);
50adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            notify();
51adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
52adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
53adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
54adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    protected synchronized Runnable getTask() throws InterruptedException {
55adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        while (taskQueue.size() == 0) {
56adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            if (!isAlive) {
57adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                return null;
58adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            }
59adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            wait();
60adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
61adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        Logger.global.info("1 Task is removed");
62adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return (Runnable) taskQueue.removeFirst();
63adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
64adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
65adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    public synchronized void close() {
66adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        if (isAlive) {
67adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            isAlive = false;
68adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            taskQueue.clear();
69adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            interrupt();
70adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
71adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
72adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
73adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    public void join() {
74adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        synchronized (this) {
75adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            isAlive = false;
76adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            notifyAll();
77adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
78adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
79adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        Thread[] threads = new Thread[activeCount()];
80adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        int count = enumerate(threads);
81adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        for (int i = 0; i < count; i++) {
82adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            try {
83adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                threads[i].join();
84adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            } catch (InterruptedException ex) {
85adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                System.err.println(ex.getMessage());
86adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            }
87adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
88adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
89adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
90adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    private class PooledThread extends Thread {
91adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
92adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        public PooledThread() {
93adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            super(ThreadPool.this, "PooledThread-" + (threadID++));
94adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
95adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
96adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        public void run() {
97adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            while (!isInterrupted()) {
98adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
99adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                Runnable task = null;
100adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                try {
101adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    task = getTask();
102adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                } catch (InterruptedException ex) {
103adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    System.err.println(ex.getMessage());
104adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                }
105adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
106adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                if (task == null) {
107adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    return;
108adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                }
109adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
110adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                try {
111adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    Logger.global.info("Task is run");
112adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    task.run();
113f33eae7e84eb6d3b0f4e86b59605bb3de73009f3Elliott Hughes
114adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                } catch (Throwable t) {
115adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                    System.err.println(t.getMessage());
116adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project                }
117adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project            }
118adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        }
119adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
120adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
121