| /* |
| // Copyright (c) 2014 Intel Corporation |
| // |
| // 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 <HwcTrace.h> |
| #include <Hwcomposer.h> |
| #include <Dump.h> |
| #include <UeventObserver.h> |
| |
| namespace android { |
| namespace intel { |
| |
| Hwcomposer* Hwcomposer::sInstance(0); |
| |
| Hwcomposer::Hwcomposer(IPlatFactory *factory) |
| : mProcs(0), |
| mDrm(0), |
| mPlatFactory(factory), |
| mVsyncManager(0), |
| mDisplayAnalyzer(0), |
| mMultiDisplayObserver(0), |
| mUeventObserver(0), |
| mPlaneManager(0), |
| mBufferManager(0), |
| mDisplayContext(0), |
| mInitialized(false) |
| { |
| CTRACE(); |
| |
| mDisplayDevices.setCapacity(IDisplayDevice::DEVICE_COUNT); |
| mDisplayDevices.clear(); |
| } |
| |
| Hwcomposer::~Hwcomposer() |
| { |
| CTRACE(); |
| deinitialize(); |
| } |
| |
| bool Hwcomposer::initCheck() const |
| { |
| return mInitialized; |
| } |
| |
| bool Hwcomposer::prepare(size_t numDisplays, |
| hwc_display_contents_1_t** displays) |
| { |
| bool ret = true; |
| |
| RETURN_FALSE_IF_NOT_INIT(); |
| ATRACE("display count = %d", numDisplays); |
| |
| if (!numDisplays || !displays) { |
| ETRACE("invalid parameters"); |
| return false; |
| } |
| |
| mDisplayAnalyzer->analyzeContents(numDisplays, displays); |
| |
| // disable reclaimed planes |
| mPlaneManager->disableReclaimedPlanes(); |
| |
| if(numDisplays > mDisplayDevices.size()) |
| numDisplays = mDisplayDevices.size(); |
| |
| // reclaim all allocated planes if possible |
| for (size_t i = 0; i < numDisplays; i++) { |
| IDisplayDevice *device = mDisplayDevices.itemAt(i); |
| if (!device) { |
| VTRACE("device %d doesn't exist", i); |
| continue; |
| } |
| if (device->getType() != IDisplayDevice::DEVICE_PRIMARY) |
| continue; |
| |
| device->prePrepare(displays[i]); |
| } |
| |
| for (size_t i = 0; i < numDisplays; i++) { |
| IDisplayDevice *device = mDisplayDevices.itemAt(i); |
| if (!device) { |
| VTRACE("device %d doesn't exist", i); |
| continue; |
| } |
| |
| if (device->getType() != IDisplayDevice::DEVICE_PRIMARY) |
| continue; |
| |
| ret = device->prepare(displays[i]); |
| if (ret == false) { |
| ETRACE("failed to do prepare for device %d", i); |
| continue; |
| } |
| } |
| |
| return ret; |
| } |
| |
| bool Hwcomposer::commit(size_t numDisplays, |
| hwc_display_contents_1_t **displays) |
| { |
| bool ret = true; |
| |
| RETURN_FALSE_IF_NOT_INIT(); |
| ATRACE("display count = %d", numDisplays); |
| |
| if (!numDisplays || !displays) { |
| ETRACE("invalid parameters"); |
| return false; |
| } |
| |
| if(numDisplays > mDisplayDevices.size()) |
| numDisplays = mDisplayDevices.size(); |
| |
| mDisplayContext->commitBegin(numDisplays, displays); |
| |
| for (size_t i = 0; i < numDisplays; i++) { |
| IDisplayDevice *device = mDisplayDevices.itemAt(i); |
| if (!device) { |
| VTRACE("device %d doesn't exist", i); |
| continue; |
| } |
| |
| if (!device->isConnected()) { |
| VTRACE("device %d is disconnected", i); |
| continue; |
| } |
| |
| if (device->getType() != IDisplayDevice::DEVICE_PRIMARY) |
| continue; |
| |
| ret = device->commit(displays[i], mDisplayContext); |
| if (ret == false) { |
| ETRACE("failed to do commit for device %d", i); |
| continue; |
| } |
| } |
| |
| mDisplayContext->commitEnd(numDisplays, displays); |
| // return true always |
| return true; |
| } |
| |
| bool Hwcomposer::setPowerMode(int disp, int mode) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| if(disp >= mDisplayDevices.size()){ |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| return device->setPowerMode(mode); |
| } |
| |
| int Hwcomposer::getActiveConfig(int disp) |
| { |
| RETURN_NULL_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return -1; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return -1; |
| } |
| |
| return device->getActiveConfig(); |
| } |
| |
| bool Hwcomposer::setActiveConfig(int disp, int index) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| return device->setActiveConfig(index); |
| } |
| |
| bool Hwcomposer::setCursorPositionAsync(int disp, int x, int y) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp != HWC_DISPLAY_PRIMARY && disp != HWC_DISPLAY_EXTERNAL) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| return mDisplayContext->setCursorPosition(disp, x, y); |
| } |
| |
| bool Hwcomposer::vsyncControl(int disp, int enabled) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| ATRACE("disp = %d, enabled = %d", disp, enabled); |
| return mVsyncManager->handleVsyncControl(disp, enabled ? true : false); |
| } |
| |
| bool Hwcomposer::blank(int disp, int blank) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| ATRACE("disp = %d, blank = %d", disp, blank); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| return device->blank(blank ? true : false); |
| } |
| |
| bool Hwcomposer::getDisplayConfigs(int disp, |
| uint32_t *configs, |
| size_t *numConfigs) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| if(disp >= mDisplayDevices.size()){ |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device %d found", disp); |
| return false; |
| } |
| |
| return device->getDisplayConfigs(configs, numConfigs); |
| } |
| |
| bool Hwcomposer::getDisplayAttributes(int disp, |
| uint32_t config, |
| const uint32_t *attributes, |
| int32_t *values) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| if(disp >= mDisplayDevices.size()){ |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| return device->getDisplayAttributes(config, attributes, values); |
| } |
| |
| bool Hwcomposer::compositionComplete(int disp) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return false; |
| } |
| |
| mDisplayContext->compositionComplete(); |
| |
| if(disp >= mDisplayDevices.size()){ |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| IDisplayDevice *device = mDisplayDevices.itemAt(disp); |
| if (!device) { |
| ETRACE("no device found"); |
| return false; |
| } |
| |
| return device->compositionComplete(); |
| } |
| |
| void Hwcomposer::vsync(int disp, int64_t timestamp) |
| { |
| RETURN_VOID_IF_NOT_INIT(); |
| |
| if (mProcs && mProcs->vsync) { |
| VTRACE("report vsync on disp %d, timestamp %llu", disp, timestamp); |
| // workaround to pretend vsync is from primary display |
| // Display will freeze if vsync is from external display. |
| mProcs->vsync(const_cast<hwc_procs_t*>(mProcs), IDisplayDevice::DEVICE_PRIMARY, timestamp); |
| } |
| } |
| |
| void Hwcomposer::hotplug(int disp, bool connected) |
| { |
| RETURN_VOID_IF_NOT_INIT(); |
| |
| // TODO: Two fake hotplug events are sent during mode setting. To avoid |
| // unnecessary audio switch, real connection status should be sent to MDS |
| mMultiDisplayObserver->notifyHotPlug(mDrm->isConnected(disp)); |
| |
| if (mProcs && mProcs->hotplug) { |
| DTRACE("report hotplug on disp %d, connected %d", disp, connected); |
| mProcs->hotplug(const_cast<hwc_procs_t*>(mProcs), disp, connected); |
| DTRACE("hotplug callback processed and returned!"); |
| } |
| |
| mDisplayAnalyzer->postHotplugEvent(connected); |
| } |
| |
| void Hwcomposer::invalidate() |
| { |
| RETURN_VOID_IF_NOT_INIT(); |
| |
| if (mProcs && mProcs->invalidate) { |
| DTRACE("invalidating screen..."); |
| mProcs->invalidate(const_cast<hwc_procs_t*>(mProcs)); |
| } |
| } |
| |
| bool Hwcomposer::release() |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| return true; |
| } |
| |
| bool Hwcomposer::dump(char *buff, int buff_len, int *cur_len) |
| { |
| RETURN_FALSE_IF_NOT_INIT(); |
| |
| Dump d(buff, buff_len); |
| |
| // dump composer status |
| d.append("Hardware Composer state:"); |
| // dump device status |
| for (size_t i= 0; i < mDisplayDevices.size(); i++) { |
| IDisplayDevice *device = mDisplayDevices.itemAt(i); |
| if (device) |
| device->dump(d); |
| } |
| |
| // dump plane manager status |
| if (mPlaneManager) |
| mPlaneManager->dump(d); |
| |
| // dump buffer manager status |
| if (mBufferManager) |
| mBufferManager->dump(d); |
| |
| return true; |
| } |
| |
| void Hwcomposer::registerProcs(hwc_procs_t const *procs) |
| { |
| CTRACE(); |
| |
| if (!procs) { |
| WTRACE("procs is NULL"); |
| } |
| mProcs = procs; |
| } |
| |
| bool Hwcomposer::initialize() |
| { |
| CTRACE(); |
| |
| // create drm |
| mDrm = new Drm(); |
| if (!mDrm || !mDrm->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to create DRM"); |
| } |
| |
| if (!mPlatFactory){ |
| DEINIT_AND_RETURN_FALSE("failed to provide a PlatFactory"); |
| } |
| |
| // create buffer manager |
| mBufferManager = mPlatFactory->createBufferManager(); |
| if (!mBufferManager || !mBufferManager->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to create buffer manager"); |
| } |
| |
| // create display plane manager |
| mPlaneManager = mPlatFactory->createDisplayPlaneManager(); |
| if (!mPlaneManager || !mPlaneManager->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to create display plane manager"); |
| } |
| |
| mDisplayContext = mPlatFactory->createDisplayContext(); |
| if (!mDisplayContext || !mDisplayContext->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to create display context"); |
| } |
| |
| mUeventObserver = new UeventObserver(); |
| if (!mUeventObserver || !mUeventObserver->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to initialize uevent observer"); |
| } |
| |
| // create display device |
| mDisplayDevices.clear(); |
| for (int i = 0; i < IDisplayDevice::DEVICE_COUNT; i++) { |
| IDisplayDevice *device = mPlatFactory->createDisplayDevice(i); |
| if (!device || !device->initialize()) { |
| DEINIT_AND_DELETE_OBJ(device); |
| DEINIT_AND_RETURN_FALSE("failed to create device %d", i); |
| } |
| // add this device |
| ETRACE("HWC devices initialize device is %p at %d", device, i); |
| mDisplayDevices.insertAt(device, i, 1); |
| } |
| |
| mVsyncManager = new VsyncManager(*this); |
| if (!mVsyncManager || !mVsyncManager->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to create Vsync Manager"); |
| } |
| |
| mDisplayAnalyzer = new DisplayAnalyzer(); |
| if (!mDisplayAnalyzer || !mDisplayAnalyzer->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to initialize display analyzer"); |
| } |
| |
| mMultiDisplayObserver = new MultiDisplayObserver(); |
| if (!mMultiDisplayObserver || !mMultiDisplayObserver->initialize()) { |
| DEINIT_AND_RETURN_FALSE("failed to initialize display observer"); |
| } |
| |
| // all initialized, starting uevent observer |
| mUeventObserver->start(); |
| |
| mInitialized = true; |
| return true; |
| } |
| |
| void Hwcomposer::deinitialize() |
| { |
| DEINIT_AND_DELETE_OBJ(mMultiDisplayObserver); |
| DEINIT_AND_DELETE_OBJ(mDisplayAnalyzer); |
| // delete mVsyncManager first as it holds reference to display devices. |
| DEINIT_AND_DELETE_OBJ(mVsyncManager); |
| |
| DEINIT_AND_DELETE_OBJ(mUeventObserver); |
| // destroy display devices |
| for (size_t i = 0; i < mDisplayDevices.size(); i++) { |
| IDisplayDevice *device = mDisplayDevices.itemAt(i); |
| DEINIT_AND_DELETE_OBJ(device); |
| } |
| mDisplayDevices.clear(); |
| |
| if (mPlatFactory) { |
| delete mPlatFactory; |
| mPlatFactory = 0; |
| } |
| |
| DEINIT_AND_DELETE_OBJ(mDisplayContext); |
| DEINIT_AND_DELETE_OBJ(mPlaneManager); |
| DEINIT_AND_DELETE_OBJ(mBufferManager); |
| DEINIT_AND_DELETE_OBJ(mDrm); |
| mInitialized = false; |
| } |
| |
| Drm* Hwcomposer::getDrm() |
| { |
| return mDrm; |
| } |
| |
| DisplayPlaneManager* Hwcomposer::getPlaneManager() |
| { |
| return mPlaneManager; |
| } |
| |
| BufferManager* Hwcomposer::getBufferManager() |
| { |
| return mBufferManager; |
| } |
| |
| IDisplayContext* Hwcomposer::getDisplayContext() |
| { |
| return mDisplayContext; |
| } |
| |
| DisplayAnalyzer* Hwcomposer::getDisplayAnalyzer() |
| { |
| return mDisplayAnalyzer; |
| } |
| |
| MultiDisplayObserver* Hwcomposer::getMultiDisplayObserver() |
| { |
| return mMultiDisplayObserver; |
| } |
| |
| IDisplayDevice* Hwcomposer::getDisplayDevice(int disp) |
| { |
| if (disp < 0 || disp >= IDisplayDevice::DEVICE_COUNT) { |
| ETRACE("invalid disp %d", disp); |
| return NULL; |
| } |
| return mDisplayDevices.itemAt(disp); |
| } |
| |
| VsyncManager* Hwcomposer::getVsyncManager() |
| { |
| return mVsyncManager; |
| } |
| |
| UeventObserver* Hwcomposer::getUeventObserver() |
| { |
| return mUeventObserver; |
| } |
| |
| } // namespace intel |
| } // namespace android |