ThreadPool.java revision a7f90298629bc6549ffd60adadd387f6caee8c59
1f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin/*
2f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * Copyright (C) 2010 The Android Open Source Project
3f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin *
4f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * Licensed under the Apache License, Version 2.0 (the "License");
5f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * you may not use this file except in compliance with the License.
6f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * You may obtain a copy of the License at
7f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin *
8f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin *      http://www.apache.org/licenses/LICENSE-2.0
9f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin *
10f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * Unless required by applicable law or agreed to in writing, software
11f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * distributed under the License is distributed on an "AS IS" BASIS,
12f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * See the License for the specific language governing permissions and
14f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin * limitations under the License.
15f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin */
16f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
17f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linpackage com.android.gallery3d.util;
18f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
19a7f90298629bc6549ffd60adadd387f6caee8c59Hung-ying Tyanimport android.util.Log;
20a7f90298629bc6549ffd60adadd387f6caee8c59Hung-ying Tyan
21f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linimport java.util.concurrent.Executor;
22f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linimport java.util.concurrent.LinkedBlockingQueue;
23f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linimport java.util.concurrent.ThreadPoolExecutor;
24f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linimport java.util.concurrent.TimeUnit;
25f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
26f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Linpublic class ThreadPool {
277817979db0c52ffeacb951625b1e821eba303285Ahbong Chang    @SuppressWarnings("unused")
28f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static final String TAG = "ThreadPool";
29f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static final int CORE_POOL_SIZE = 4;
30f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static final int MAX_POOL_SIZE = 8;
31f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static final int KEEP_ALIVE_TIME = 10; // 10 seconds
32f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
33f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    // Resource type
34f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public static final int MODE_NONE = 0;
35f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public static final int MODE_CPU = 1;
36f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public static final int MODE_NETWORK = 2;
37f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
38f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public static final JobContext JOB_CONTEXT_STUB = new JobContextStub();
39f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
40f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    ResourceCounter mCpuCounter = new ResourceCounter(2);
41f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    ResourceCounter mNetworkCounter = new ResourceCounter(2);
42f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
43f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    // A Job is like a Callable, but it has an addition JobContext parameter.
44f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public interface Job<T> {
45f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public T run(JobContext jc);
46f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
47f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
48f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public interface JobContext {
49f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        boolean isCancelled();
50f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        void setCancelListener(CancelListener listener);
51f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        boolean setMode(int mode);
52f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
53f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
54f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static class JobContextStub implements JobContext {
55f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        @Override
56f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public boolean isCancelled() {
57f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return false;
58f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
59f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
60f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        @Override
61f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public void setCancelListener(CancelListener listener) {
62f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
63f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
64f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        @Override
65f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public boolean setMode(int mode) {
66f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return true;
67f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
68f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
69f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
70f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public interface CancelListener {
71f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public void onCancel();
72f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
73f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
74f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private static class ResourceCounter {
75f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public int value;
76f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public ResourceCounter(int v) {
77f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            value = v;
78f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
79f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
80f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
81f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private final Executor mExecutor;
82f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
83f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public ThreadPool() {
84f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        mExecutor = new ThreadPoolExecutor(
85f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                CORE_POOL_SIZE, MAX_POOL_SIZE, KEEP_ALIVE_TIME,
86f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                TimeUnit.SECONDS, new LinkedBlockingQueue<Runnable>(),
87f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                new PriorityThreadFactory("thread-pool",
88f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                android.os.Process.THREAD_PRIORITY_BACKGROUND));
89f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
90f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
91f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    // Submit a job to the thread pool. The listener will be called when the
92f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    // job is finished (or cancelled).
93f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public <T> Future<T> submit(Job<T> job, FutureListener<T> listener) {
94f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        Worker<T> w = new Worker<T>(job, listener);
95f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        mExecutor.execute(w);
96f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        return w;
97f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
98f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
99f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    public <T> Future<T> submit(Job<T> job) {
100f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        return submit(job, null);
101f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
102f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
103f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    private class Worker<T> implements Runnable, Future<T>, JobContext {
1047817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @SuppressWarnings("hiding")
105f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private static final String TAG = "Worker";
106f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private Job<T> mJob;
107f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private FutureListener<T> mListener;
108f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private CancelListener mCancelListener;
109f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private ResourceCounter mWaitOnResource;
110f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private volatile boolean mIsCancelled;
111f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private boolean mIsDone;
112f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private T mResult;
113f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private int mMode;
114f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
115f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public Worker(Job<T> job, FutureListener<T> listener) {
116f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            mJob = job;
117f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            mListener = listener;
118f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
119f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
120f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        // This is called by a thread in the thread pool.
1217817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
122f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public void run() {
123f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            T result = null;
124f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
125f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            // A job is in CPU mode by default. setMode returns false
126f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            // if the job is cancelled.
127f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (setMode(MODE_CPU)) {
128f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                try {
129f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    result = mJob.run(this);
130f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                } catch (Throwable ex) {
131f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    Log.w(TAG, "Exception in running a job", ex);
132f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
133f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
134f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
135f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            synchronized(this) {
136f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                setMode(MODE_NONE);
137f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mResult = result;
138f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mIsDone = true;
139f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                notifyAll();
140f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
141f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mListener != null) mListener.onFutureDone(this);
142f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
143f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
144f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        // Below are the methods for Future.
1457817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
146f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public synchronized void cancel() {
147f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mIsCancelled) return;
148f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            mIsCancelled = true;
149f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mWaitOnResource != null) {
150f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                synchronized (mWaitOnResource) {
151f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    mWaitOnResource.notifyAll();
152f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
153f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
154f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mCancelListener != null) {
155f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mCancelListener.onCancel();
156f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
157f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
158f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
1597817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
160f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public boolean isCancelled() {
161f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return mIsCancelled;
162f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
163f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
1647817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
165f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public synchronized boolean isDone() {
166f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return mIsDone;
167f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
168f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
1697817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
170f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public synchronized T get() {
171f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            while (!mIsDone) {
172f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                try {
173f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    wait();
174f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                } catch (Exception ex) {
175f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    Log.w(TAG, "ingore exception", ex);
176f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    // ignore.
177f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
178f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
179f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return mResult;
180f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
181f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
1827817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
183f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public void waitDone() {
184f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            get();
185f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
186f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
187f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        // Below are the methods for JobContext (only called from the
188f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        // thread running the job)
1897817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
190f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public synchronized void setCancelListener(CancelListener listener) {
191f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            mCancelListener = listener;
192f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mIsCancelled && mCancelListener != null) {
193f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mCancelListener.onCancel();
194f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
195f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
196f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
1977817979db0c52ffeacb951625b1e821eba303285Ahbong Chang        @Override
198f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        public boolean setMode(int mode) {
199f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            // Release old resource
200f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            ResourceCounter rc = modeToCounter(mMode);
201f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (rc != null) releaseResource(rc);
202f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            mMode = MODE_NONE;
203f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
204f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            // Acquire new resource
205f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            rc = modeToCounter(mode);
206f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (rc != null) {
207f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                if (!acquireResource(rc)) {
208f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    return false;
209f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
210f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mMode = mode;
211f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
212f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
213f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return true;
214f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
215f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
216f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private ResourceCounter modeToCounter(int mode) {
217f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            if (mode == MODE_CPU) {
218f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                return mCpuCounter;
219f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            } else if (mode == MODE_NETWORK) {
220f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                return mNetworkCounter;
221f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            } else {
222f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                return null;
223f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
224f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
225f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
226f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private boolean acquireResource(ResourceCounter counter) {
227f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            while (true) {
228f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                synchronized (this) {
229f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    if (mIsCancelled) {
230f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        mWaitOnResource = null;
231f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        return false;
232f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    }
233f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    mWaitOnResource = counter;
234f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
235f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
236f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                synchronized (counter) {
237f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    if (counter.value > 0) {
238f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        counter.value--;
239f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        break;
240f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    } else {
241f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        try {
242f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                            counter.wait();
243f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        } catch (InterruptedException ex) {
244f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                            // ignore.
245f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                        }
246f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                    }
247f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                }
248f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
249f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
250f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            synchronized (this) {
251f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                mWaitOnResource = null;
252f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
253f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
254f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            return true;
255f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
256f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin
257f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        private void releaseResource(ResourceCounter counter) {
258f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            synchronized (counter) {
259f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                counter.value++;
260f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin                counter.notifyAll();
261f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin            }
262f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin        }
263f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin    }
264f9a0a4306d589b4a4e20554fed512a603426bfa1Owen Lin}
265