/* * Copyright (C) 2007 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 <binder/IMemory.h> #include <binder/Parcel.h> #include <utils/Errors.h> #include <binder/MemoryHeapBase.h> #include <ui/IOverlay.h> #include <ui/Overlay.h> #include <hardware/overlay.h> namespace android { Overlay::Overlay(const sp<OverlayRef>& overlayRef) : mOverlayRef(overlayRef), mOverlayData(0), mStatus(NO_INIT) { mOverlayData = NULL; hw_module_t const* module; if (overlayRef != 0) { if (hw_get_module(OVERLAY_HARDWARE_MODULE_ID, &module) == 0) { if (overlay_data_open(module, &mOverlayData) == NO_ERROR) { mStatus = mOverlayData->initialize(mOverlayData, overlayRef->mOverlayHandle); } } } } Overlay::~Overlay() { if (mOverlayData) { overlay_data_close(mOverlayData); } } status_t Overlay::dequeueBuffer(overlay_buffer_t* buffer) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->dequeueBuffer(mOverlayData, buffer); } status_t Overlay::queueBuffer(overlay_buffer_t buffer) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->queueBuffer(mOverlayData, buffer); } status_t Overlay::resizeInput(uint32_t width, uint32_t height) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->resizeInput(mOverlayData, width, height); } status_t Overlay::setParameter(int param, int value) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->setParameter(mOverlayData, param, value); } status_t Overlay::setCrop(uint32_t x, uint32_t y, uint32_t w, uint32_t h) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->setCrop(mOverlayData, x, y, w, h); } status_t Overlay::getCrop(uint32_t* x, uint32_t* y, uint32_t* w, uint32_t* h) { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->getCrop(mOverlayData, x, y, w, h); } int32_t Overlay::getBufferCount() const { if (mStatus != NO_ERROR) return mStatus; return mOverlayData->getBufferCount(mOverlayData); } void* Overlay::getBufferAddress(overlay_buffer_t buffer) { if (mStatus != NO_ERROR) return NULL; return mOverlayData->getBufferAddress(mOverlayData, buffer); } void Overlay::destroy() { if (mStatus != NO_ERROR) return; // Must delete the objects in reverse creation order, thus the // data side must be closed first and then the destroy send to // the control side. if (mOverlayData) { overlay_data_close(mOverlayData); mOverlayData = NULL; } mOverlayRef->mOverlayChannel->destroy(); } status_t Overlay::getStatus() const { return mStatus; } overlay_handle_t Overlay::getHandleRef() const { if (mStatus != NO_ERROR) return NULL; return mOverlayRef->mOverlayHandle; } uint32_t Overlay::getWidth() const { if (mStatus != NO_ERROR) return 0; return mOverlayRef->mWidth; } uint32_t Overlay::getHeight() const { if (mStatus != NO_ERROR) return 0; return mOverlayRef->mHeight; } int32_t Overlay::getFormat() const { if (mStatus != NO_ERROR) return -1; return mOverlayRef->mFormat; } int32_t Overlay::getWidthStride() const { if (mStatus != NO_ERROR) return 0; return mOverlayRef->mWidthStride; } int32_t Overlay::getHeightStride() const { if (mStatus != NO_ERROR) return 0; return mOverlayRef->mHeightStride; } // ---------------------------------------------------------------------------- OverlayRef::OverlayRef() : mOverlayHandle(0), mWidth(0), mHeight(0), mFormat(0), mWidthStride(0), mHeightStride(0), mOwnHandle(true) { } OverlayRef::OverlayRef(overlay_handle_t handle, const sp<IOverlay>& channel, uint32_t w, uint32_t h, int32_t f, uint32_t ws, uint32_t hs) : mOverlayHandle(handle), mOverlayChannel(channel), mWidth(w), mHeight(h), mFormat(f), mWidthStride(ws), mHeightStride(hs), mOwnHandle(false) { } OverlayRef::~OverlayRef() { if (mOwnHandle) { native_handle_close(mOverlayHandle); native_handle_delete(const_cast<native_handle*>(mOverlayHandle)); } } sp<OverlayRef> OverlayRef::readFromParcel(const Parcel& data) { sp<OverlayRef> result; sp<IOverlay> overlay = IOverlay::asInterface(data.readStrongBinder()); if (overlay != NULL) { uint32_t w = data.readInt32(); uint32_t h = data.readInt32(); uint32_t f = data.readInt32(); uint32_t ws = data.readInt32(); uint32_t hs = data.readInt32(); native_handle* handle = data.readNativeHandle(); result = new OverlayRef(); result->mOverlayHandle = handle; result->mOverlayChannel = overlay; result->mWidth = w; result->mHeight = h; result->mFormat = f; result->mWidthStride = ws; result->mHeightStride = hs; } return result; } status_t OverlayRef::writeToParcel(Parcel* reply, const sp<OverlayRef>& o) { if (o != NULL) { reply->writeStrongBinder(o->mOverlayChannel->asBinder()); reply->writeInt32(o->mWidth); reply->writeInt32(o->mHeight); reply->writeInt32(o->mFormat); reply->writeInt32(o->mWidthStride); reply->writeInt32(o->mHeightStride); reply->writeNativeHandle(o->mOverlayHandle); } else { reply->writeStrongBinder(NULL); } return NO_ERROR; } // ---------------------------------------------------------------------------- }; // namespace android