/* * Copyright (c) 2013-2014 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 "hwc_ad.h" #include "hwc_utils.h" #define DEBUG 0 using namespace overlay; using namespace overlay::utils; namespace qhwc { //Helper to write data to ad node static void adWrite(const int& value) { const int wbFbNum = Overlay::getFbForDpy(Overlay::DPY_WRITEBACK); char wbFbPath[256]; snprintf (wbFbPath, sizeof(wbFbPath), "/sys/class/graphics/fb%d/ad", wbFbNum); int adFd = open(wbFbPath, O_WRONLY); if(adFd >= 0) { char opStr[4] = ""; snprintf(opStr, sizeof(opStr), "%d", value); ssize_t ret = write(adFd, opStr, strlen(opStr)); if(ret < 0) { ALOGE("%s: Failed to write %d with error %s", __func__, value, strerror(errno)); } else if (ret == 0){ ALOGE("%s Nothing written to ad", __func__); } else { ALOGD_IF(DEBUG, "%s: Wrote %d to ad", __func__, value); } close(adFd); } else { ALOGE("%s: Failed to open /sys/class/graphics/fb%d/ad with error %s", __func__, wbFbNum, strerror(errno)); } } //Helper to read data from ad node static int adRead() { const int wbFbNum = Overlay::getFbForDpy(Overlay::DPY_WRITEBACK); int ret = -1; char wbFbPath[256]; snprintf (wbFbPath, sizeof(wbFbPath), "/sys/class/graphics/fb%d/ad", wbFbNum); int adFd = open(wbFbPath, O_RDONLY); if(adFd >= 0) { char opStr[4]; ssize_t bytesRead = read(adFd, opStr, sizeof(opStr) - 1); if(bytesRead > 0) { opStr[bytesRead] = '\0'; //Should return -1, 0 or 1 ret = atoi(opStr); ALOGD_IF(DEBUG, "%s: Read %d from ad", __func__, ret); } else if(bytesRead == 0) { ALOGE("%s: ad node empty", __func__); } else { ALOGE("%s: Read from ad node failed with error %s", __func__, strerror(errno)); } close(adFd); } else { ALOGD("%s: /sys/class/graphics/fb%d/ad could not be opened : %s", __func__, wbFbNum, strerror(errno)); } return ret; } AssertiveDisplay::AssertiveDisplay(hwc_context_t *ctx) : mTurnedOff(true), mFeatureEnabled(false), mDest(overlay::utils::OV_INVALID) { //Values in ad node: //-1 means feature is disabled on device // 0 means feature exists but turned off, will be turned on by hwc // 1 means feature is turned on by hwc // Plus, we do this feature only on split primary displays. // Plus, we do this feature only if ro.qcom.ad=2 char property[PROPERTY_VALUE_MAX]; const int ENABLED = 2; int val = 0; if(property_get("ro.qcom.ad", property, "0") > 0) { val = atoi(property); } if(adRead() >= 0 && isDisplaySplit(ctx, HWC_DISPLAY_PRIMARY) && val == ENABLED) { ALOGD_IF(DEBUG, "Assertive display feature supported"); mFeatureEnabled = true; // If feature exists but is turned off, set mTurnedOff to true mTurnedOff = adRead() > 0 ? false : true; } } void AssertiveDisplay::markDoable(hwc_context_t *ctx, const hwc_display_contents_1_t* list) { mDoable = false; if(mFeatureEnabled && !isSecondaryConnected(ctx) && ctx->listStats[HWC_DISPLAY_PRIMARY].yuvCount == 1) { int nYuvIndex = ctx->listStats[HWC_DISPLAY_PRIMARY].yuvIndices[0]; const hwc_layer_1_t* layer = &list->hwLayers[nYuvIndex]; private_handle_t *hnd = (private_handle_t *)layer->handle; qdutils::MDPVersion& mdpHw = qdutils::MDPVersion::getInstance(); if(hnd && hnd->width <= (int) mdpHw.getMaxPipeWidth()) { mDoable = true; } } } void AssertiveDisplay::turnOffAD() { if(mFeatureEnabled) { if(!mTurnedOff) { const int off = 0; adWrite(off); mTurnedOff = true; } } mDoable = false; } bool AssertiveDisplay::prepare(hwc_context_t *ctx, const hwc_rect_t& crop, const Whf& whf, const private_handle_t *hnd) { if(!isDoable()) { //Cleanup one time during this switch turnOffAD(); return false; } Overlay::PipeSpecs pipeSpecs; pipeSpecs.formatClass = Overlay::FORMAT_YUV; pipeSpecs.dpy = overlay::Overlay::DPY_WRITEBACK; pipeSpecs.fb = false; ovutils::eDest dest = ctx->mOverlay->getPipe(pipeSpecs); if(dest == OV_INVALID) { ALOGE("%s failed: No VG pipe available", __func__); mDoable = false; return false; } overlay::Writeback *wb = overlay::Writeback::getInstance(); //Set Security flag on writeback if(isSecureBuffer(hnd)) { if(!wb->setSecure(isSecureBuffer(hnd))) { ALOGE("Failure in setting WB secure flag for ad"); return false; } } if(!wb->configureDpyInfo(hnd->width, hnd->height)) { ALOGE("%s: config display failed", __func__); mDoable = false; return false; } int tmpW, tmpH; size_t size; int format = ovutils::getHALFormat(wb->getOutputFormat()); if(format < 0) { ALOGE("%s invalid format %d", __func__, format); mDoable = false; return false; } size = getBufferSizeAndDimensions(hnd->width, hnd->height, format, tmpW, tmpH); if(!wb->configureMemory((uint32_t)size)) { ALOGE("%s: config memory failed", __func__); mDoable = false; return false; } eMdpFlags mdpFlags = OV_MDP_FLAGS_NONE; if(isSecureBuffer(hnd)) { ovutils::setMdpFlags(mdpFlags, ovutils::OV_MDP_SECURE_OVERLAY_SESSION); } PipeArgs parg(mdpFlags, whf, ZORDER_0, ROT_FLAGS_NONE); hwc_rect_t dst = crop; //input same as output if(configMdp(ctx->mOverlay, parg, OVERLAY_TRANSFORM_0, crop, dst, NULL, dest) < 0) { ALOGE("%s: configMdp failed", __func__); mDoable = false; return false; } mDest = dest; int wbFd = wb->getFbFd(); if(mFeatureEnabled && wbFd >= 0 && !ctx->mOverlay->validateAndSet(overlay::Overlay::DPY_WRITEBACK, wbFd)) { ALOGE("%s: Failed to validate and set overlay for dpy %d" ,__FUNCTION__, overlay::Overlay::DPY_WRITEBACK); turnOffAD(); return false; } // Only turn on AD if there are no errors during configuration stage // and if it was previously in OFF state. if(mFeatureEnabled && mTurnedOff) { //write to sysfs, one time during this switch const int on = 1; adWrite(on); mTurnedOff = false; } return true; } bool AssertiveDisplay::draw(hwc_context_t *ctx, int fd, uint32_t offset) { if(!isDoable()) { return false; } if (!ctx->mOverlay->queueBuffer(fd, offset, mDest)) { ALOGE("%s: queueBuffer failed", __func__); return false; } overlay::Writeback *wb = overlay::Writeback::getInstance(); if(!wb->writeSync()) { return false; } return true; } int AssertiveDisplay::getDstFd() const { overlay::Writeback *wb = overlay::Writeback::getInstance(); return wb->getDstFd(); } uint32_t AssertiveDisplay::getDstOffset() const { overlay::Writeback *wb = overlay::Writeback::getInstance(); return wb->getOffset(); } }