/* Copyright (c) 2015, The Linux Foundation. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * Neither the name of The Linux Foundation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include "RenderEngine/RenderEngine.h" #include "DisplayHardware/FramebufferSurface.h" #include "DisplayUtils.h" #if QTI_BSP #include #include #include #include #include #include #endif namespace android { DisplayUtils* DisplayUtils::sDisplayUtils = NULL; bool DisplayUtils::sUseExtendedImpls = false; DisplayUtils::DisplayUtils() { #ifdef QTI_BSP sUseExtendedImpls = true; hasWbNode(); #endif } DisplayUtils* DisplayUtils::getInstance() { if(sDisplayUtils == NULL) { sDisplayUtils = new DisplayUtils(); } return sDisplayUtils; } SurfaceFlinger* DisplayUtils::getSFInstance() { #ifdef QTI_BSP if(sUseExtendedImpls) { return new ExSurfaceFlinger(); } #endif return new SurfaceFlinger(); } Layer* DisplayUtils::getLayerInstance(SurfaceFlinger* flinger, const sp& client, const String8& name, uint32_t w, uint32_t h, uint32_t flags) { #ifdef QTI_BSP if(sUseExtendedImpls) { return new ExLayer(flinger, client, name, w, h, flags); } #endif return new Layer(flinger, client, name, w, h, flags); } #ifndef USE_HWC2 HWComposer* DisplayUtils::getHWCInstance( const sp& flinger, HWComposer::EventHandler& handler) { #ifdef QTI_BSP if(sUseExtendedImpls) { return new ExHWComposer(flinger, handler); } #endif return new HWComposer(flinger,handler); } #else HWComposer* DisplayUtils::getHWCInstance( const sp& flinger) { #ifdef QTI_BSP if(sUseExtendedImpls) { return new ExHWComposer(flinger); } #endif return new HWComposer(flinger); } #endif void DisplayUtils::initVDSInstance(HWComposer* hwc, int32_t hwcDisplayId, sp currentStateSurface, sp &dispSurface, sp &producer, sp bqProducer, sp bqConsumer, String8 currentStateDisplayName, bool currentStateIsSecure, int currentStateType) { #ifdef QTI_BSP if(sUseExtendedImpls) { if(hwc->isVDSEnabled()) { VirtualDisplaySurface* vds = new ExVirtualDisplaySurface(*hwc, hwcDisplayId, currentStateSurface, bqProducer, bqConsumer, currentStateDisplayName, currentStateIsSecure); dispSurface = vds; producer = vds; } else if(!createV4L2BasedVirtualDisplay(hwc, hwcDisplayId, dispSurface, producer, currentStateSurface, bqProducer, bqConsumer, currentStateType)) { VirtualDisplaySurface* vds = new VirtualDisplaySurface(*hwc, hwcDisplayId, currentStateSurface, bqProducer, bqConsumer, currentStateDisplayName); dispSurface = vds; producer = vds; } } else { #endif (void)currentStateIsSecure; (void)currentStateType; VirtualDisplaySurface* vds = new VirtualDisplaySurface(*hwc, hwcDisplayId, currentStateSurface, bqProducer, bqConsumer, currentStateDisplayName); dispSurface = vds; producer = vds; #ifdef QTI_BSP } #endif } bool DisplayUtils::createV4L2BasedVirtualDisplay(HWComposer* hwc, int32_t &hwcDisplayId, sp &dispSurface, sp &producer, sp currentStateSurface, sp bqProducer, sp bqConsumer, int currentStateType) { char value[PROPERTY_VALUE_MAX]; property_get("persist.sys.wfd.virtual", value, "0"); int wfdVirtual = atoi(value); if(wfdVirtual && hwcDisplayId > 0) { //Read virtual display properties and create a //rendering surface for it inorder to be handled //by hwc. sp mNativeWindow = new Surface(currentStateSurface); ANativeWindow* const window = mNativeWindow.get(); int format; window->query(window, NATIVE_WINDOW_FORMAT, &format); EGLSurface surface; EGLint w, h; EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY); // In M AOSP getEGLConfig() always returns EGL_NO_CONFIG as // EGL_ANDROIDX_no_config_context active now. EGLConfig config = RenderEngine::chooseEglConfig(display, format); surface = eglCreateWindowSurface(display, config, window, NULL); eglQuerySurface(display, surface, EGL_WIDTH, &w); eglQuerySurface(display, surface, EGL_HEIGHT, &h); #ifndef USE_HWC2 if(hwc->setVirtualDisplayProperties(hwcDisplayId, w, h, format) != NO_ERROR) return false; #endif dispSurface = new FramebufferSurface(*hwc, currentStateType, bqConsumer); producer = bqProducer; return true; } return false; } bool DisplayUtils::canAllocateHwcDisplayIdForVDS(int usage) { // on AOSP builds with QTI_BSP disabled, we should allocate hwc display id for virtual display int flag_mask = 0xffffffff; char value[PROPERTY_VALUE_MAX]; property_get("debug.vds.allow_hwc", value, "0"); int allowHwcForVDS = atoi(value); #ifdef QTI_BSP // Do not allow hardware acceleration flag_mask = GRALLOC_USAGE_PRIVATE_WFD; #endif return ((mHasWbNode) && (allowHwcForVDS || (usage & flag_mask))); } int DisplayUtils::getNumFbNodes() { int i = 0; while (hasFbNode(i)) i++; return i; } bool DisplayUtils::hasFbNode(int index) { char filename[kMaxStringLength]; snprintf(filename, kMaxStringLength, "/sys/class/graphics/fb%d", index); if (access(filename, F_OK) < 0) { return false; } return true; } bool DisplayUtils::hasWbNode() { int fbNum = getNumFbNodes(); char msmFbType [kMaxStringLength]; char fbType [kMaxStringLength]; FILE *displayPanelFP = NULL; mHasWbNode = false; for (int i = 0; i < fbNum; i++) { snprintf (msmFbType,MAX_FRAME_BUFFER_NAME_SIZE, "/sys/class/graphics/fb%d/msm_fb_type", i); displayPanelFP = fopen(msmFbType, "r"); if(displayPanelFP) { fread(fbType, sizeof(char), MAX_FRAME_BUFFER_NAME_SIZE, displayPanelFP); fclose(displayPanelFP); if(strncmp(fbType, "writeback panel", strlen("writeback panel")) == 0) { mHasWbNode = true; break; } } } return mHasWbNode; } }; // namespace android