C++程序  |  110行  |  3.32 KB

/*
 * Copyright 2016 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.
 */
#define LOG_TAG "BufferQueueScheduler"

#include "BufferQueueScheduler.h"

#include <android/native_window.h>
#include <gui/Surface.h>

using namespace android;

BufferQueueScheduler::BufferQueueScheduler(
        const sp<SurfaceControl>& surfaceControl, const HSV& color, int id)
      : mSurfaceControl(surfaceControl), mColor(color), mSurfaceId(id), mContinueScheduling(true) {}

void BufferQueueScheduler::startScheduling() {
    ALOGV("Starting Scheduler for %d Layer", mSurfaceId);
    std::unique_lock<std::mutex> lock(mMutex);
    if (mSurfaceControl == nullptr) {
        mCondition.wait(lock, [&] { return (mSurfaceControl != nullptr); });
    }

    while (mContinueScheduling) {
        while (true) {
            if (mBufferEvents.empty()) {
                break;
            }

            BufferEvent event = mBufferEvents.front();
            lock.unlock();

            bufferUpdate(event.dimensions);
            fillSurface(event.event);
            mColor.modulate();
            lock.lock();
            mBufferEvents.pop();
        }
        mCondition.wait(lock);
    }
}

void BufferQueueScheduler::addEvent(const BufferEvent& event) {
    std::lock_guard<std::mutex> lock(mMutex);
    mBufferEvents.push(event);
    mCondition.notify_one();
}

void BufferQueueScheduler::stopScheduling() {
    std::lock_guard<std::mutex> lock(mMutex);
    mContinueScheduling = false;
    mCondition.notify_one();
}

void BufferQueueScheduler::setSurfaceControl(
        const sp<SurfaceControl>& surfaceControl, const HSV& color) {
    std::lock_guard<std::mutex> lock(mMutex);
    mSurfaceControl = surfaceControl;
    mColor = color;
    mCondition.notify_one();
}

void BufferQueueScheduler::bufferUpdate(const Dimensions& dimensions) {
    sp<Surface> s = mSurfaceControl->getSurface();
    s->setBuffersDimensions(dimensions.width, dimensions.height);
}

void BufferQueueScheduler::fillSurface(const std::shared_ptr<Event>& event) {
    ANativeWindow_Buffer outBuffer;
    sp<Surface> s = mSurfaceControl->getSurface();

    status_t status = s->lock(&outBuffer, nullptr);

    if (status != NO_ERROR) {
        ALOGE("fillSurface: failed to lock buffer, (%d)", status);
        return;
    }

    auto color = mColor.getRGB();

    auto img = reinterpret_cast<uint8_t*>(outBuffer.bits);
    for (int y = 0; y < outBuffer.height; y++) {
        for (int x = 0; x < outBuffer.width; x++) {
            uint8_t* pixel = img + (4 * (y * outBuffer.stride + x));
            pixel[0] = color.r;
            pixel[1] = color.g;
            pixel[2] = color.b;
            pixel[3] = LAYER_ALPHA;
        }
    }

    event->readyToExecute();

    status = s->unlockAndPost();

    ALOGE_IF(status != NO_ERROR, "fillSurface: failed to unlock and post buffer, (%d)", status);
}