|  | /* | 
|  | * Copyright 2020 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. | 
|  | */ | 
|  |  | 
|  | #include "CommonPool.h" | 
|  |  | 
|  | #undef LOG_TAG | 
|  | #define LOG_TAG "RenderEngine" | 
|  | #define ATRACE_TAG ATRACE_TAG_GRAPHICS | 
|  |  | 
|  | #include <sys/resource.h> | 
|  | #include <utils/Trace.h> | 
|  |  | 
|  | #include <system/thread_defs.h> | 
|  | #include <array> | 
|  |  | 
|  | namespace android { | 
|  | namespace renderengine { | 
|  | namespace skia { | 
|  |  | 
|  | CommonPool::CommonPool() { | 
|  | ATRACE_CALL(); | 
|  |  | 
|  | CommonPool* pool = this; | 
|  | // Create 2 workers | 
|  | for (int i = 0; i < THREAD_COUNT; i++) { | 
|  | std::thread worker([pool, i] { | 
|  | { | 
|  | std::array<char, 20> name{"reTask"}; | 
|  | snprintf(name.data(), name.size(), "reTask%d", i); | 
|  | auto self = pthread_self(); | 
|  | pthread_setname_np(self, name.data()); | 
|  | setpriority(PRIO_PROCESS, 0, ANDROID_PRIORITY_FOREGROUND); | 
|  | } | 
|  | pool->workerLoop(); | 
|  | }); | 
|  | worker.detach(); | 
|  | } | 
|  | } | 
|  |  | 
|  | CommonPool& CommonPool::instance() { | 
|  | static CommonPool pool; | 
|  | return pool; | 
|  | } | 
|  |  | 
|  | void CommonPool::post(Task&& task) { | 
|  | instance().enqueue(std::move(task)); | 
|  | } | 
|  |  | 
|  | void CommonPool::enqueue(Task&& task) { | 
|  | std::unique_lock lock(mLock); | 
|  | while (mWorkQueue.size() > QUEUE_SIZE) { | 
|  | lock.unlock(); | 
|  | ALOGW("Queue is full: %d, waiting before adding more tasks.", QUEUE_SIZE); | 
|  | usleep(100); | 
|  | lock.lock(); | 
|  | } | 
|  | mWorkQueue.push(std::move(task)); | 
|  | if (mWaitingThreads == THREAD_COUNT || (mWaitingThreads > 0 && mWorkQueue.size() > 1)) { | 
|  | mCondition.notify_one(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CommonPool::workerLoop() { | 
|  | std::unique_lock lock(mLock); | 
|  | while (true) { | 
|  | if (mWorkQueue.size() == 0) { | 
|  | mWaitingThreads++; | 
|  | mCondition.wait(lock); | 
|  | mWaitingThreads--; | 
|  | } | 
|  | // Need to double-check that work is still available now that we have the lock | 
|  | // It may have already been grabbed by a different thread | 
|  | while (mWorkQueue.size() > 0) { | 
|  | auto work = mWorkQueue.front(); | 
|  | mWorkQueue.pop(); | 
|  | lock.unlock(); | 
|  | work(); | 
|  | lock.lock(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | } // namespace skia | 
|  | } // namespace renderengine | 
|  | } // namespace android |