| /* |
| * Copyright (C) 2008 The Android Open Source Project |
| * Copyright (c) 2010-2012, The Linux Foundation. All rights reserved. |
| * Not a Contribution, Apache license notifications and license are retained |
| * for attribution purposes only. |
| * |
| * 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 "overlayRotator.h" |
| #include "overlayUtils.h" |
| #include "mdp_version.h" |
| #include "sync/sync.h" |
| #include "gr.h" |
| |
| namespace ovutils = overlay::utils; |
| |
| namespace overlay { |
| |
| //============Rotator========================= |
| |
| Rotator::Rotator() { |
| char property[PROPERTY_VALUE_MAX]; |
| mRotCacheDisabled = false; |
| if((property_get("debug.rotcache.disable", property, NULL) > 0) && |
| (!strncmp(property, "1", PROPERTY_VALUE_MAX ) || |
| (!strncasecmp(property,"true", PROPERTY_VALUE_MAX )))) { |
| /* Used in debugging to turnoff rotator caching */ |
| mRotCacheDisabled = true; |
| } |
| } |
| |
| Rotator::~Rotator() {} |
| |
| Rotator* Rotator::getRotator() { |
| int type = getRotatorHwType(); |
| if(type == TYPE_MDP) { |
| return new MdpRot(); //will do reset |
| } else if(type == TYPE_MDSS) { |
| return new MdssRot(); |
| } else { |
| ALOGE("%s Unknown h/w type %d", __FUNCTION__, type); |
| return NULL; |
| } |
| } |
| |
| int Rotator::getDownscaleFactor(const int& srcW, const int& srcH, |
| const int& dstW, const int& dstH, const uint32_t& mdpFormat, |
| const bool& isInterlaced) { |
| if(getRotatorHwType() == TYPE_MDSS) { |
| return MdssRot::getDownscaleFactor(srcW, srcH, dstW, dstH, |
| mdpFormat, isInterlaced); |
| } |
| return MdpRot::getDownscaleFactor(srcW, srcH, dstW, dstH, |
| mdpFormat, isInterlaced); |
| } |
| |
| uint32_t Rotator::calcOutputBufSize(const utils::Whf& destWhf) { |
| //dummy aligned w & h. |
| int alW = 0, alH = 0; |
| int halFormat = ovutils::getHALFormat(destWhf.format); |
| //A call into gralloc/memalloc |
| return getBufferSizeAndDimensions( |
| destWhf.w, destWhf.h, halFormat, alW, alH); |
| } |
| |
| int Rotator::getRotatorHwType() { |
| int mdpVersion = qdutils::MDPVersion::getInstance().getMDPVersion(); |
| if (mdpVersion == qdutils::MDSS_V5) |
| return TYPE_MDSS; |
| return TYPE_MDP; |
| } |
| |
| bool Rotator::isRotCached(int fd, uint32_t offset) const { |
| if(mRotCacheDisabled or rotConfChanged() or rotDataChanged(fd,offset)) |
| return false; |
| return true; |
| } |
| |
| bool Rotator::rotDataChanged(int fd, uint32_t offset) const { |
| /* fd and offset are the attributes of the current rotator input buffer. |
| * At this instance, getSrcMemId() and getSrcOffset() return the |
| * attributes of the previous rotator input buffer */ |
| if( (fd == getSrcMemId()) and (offset == getSrcOffset()) ) |
| return false; |
| return true; |
| } |
| |
| //============RotMem========================= |
| |
| bool RotMem::close() { |
| bool ret = true; |
| if(valid()) { |
| if(mem.close() == false) { |
| ALOGE("%s error in closing rot mem", __FUNCTION__); |
| ret = false; |
| } |
| } |
| return ret; |
| } |
| |
| RotMem::RotMem() : mCurrIndex(0) { |
| utils::memset0(mRotOffset); |
| for(int i = 0; i < ROT_NUM_BUFS; i++) { |
| mRelFence[i] = -1; |
| } |
| } |
| |
| RotMem::~RotMem() { |
| for(int i = 0; i < ROT_NUM_BUFS; i++) { |
| ::close(mRelFence[i]); |
| mRelFence[i] = -1; |
| } |
| } |
| |
| void RotMem::setCurrBufReleaseFd(const int& fence) { |
| int ret = 0; |
| |
| if(mRelFence[mCurrIndex] >= 0) { |
| //Wait for previous usage of this buffer to be over. |
| //Can happen if rotation takes > vsync and a fast producer. i.e queue |
| //happens in subsequent vsyncs either because content is 60fps or |
| //because the producer is hasty sometimes. |
| ret = sync_wait(mRelFence[mCurrIndex], 1000); |
| if(ret < 0) { |
| ALOGE("%s: sync_wait error!! error no = %d err str = %s", |
| __FUNCTION__, errno, strerror(errno)); |
| } |
| ::close(mRelFence[mCurrIndex]); |
| } |
| mRelFence[mCurrIndex] = fence; |
| } |
| |
| void RotMem::setPrevBufReleaseFd(const int& fence) { |
| uint32_t numRotBufs = mem.numBufs(); |
| uint32_t prevIndex = (mCurrIndex + numRotBufs - 1) % (numRotBufs); |
| |
| if(mRelFence[prevIndex] >= 0) { |
| /* No need of any wait as nothing will be written into this |
| * buffer by the rotator (this func is called when rotator is |
| * in cache mode) */ |
| ::close(mRelFence[prevIndex]); |
| } |
| |
| mRelFence[prevIndex] = fence; |
| } |
| |
| //============RotMgr========================= |
| RotMgr * RotMgr::sRotMgr = NULL; |
| |
| RotMgr* RotMgr::getInstance() { |
| if(sRotMgr == NULL) { |
| sRotMgr = new RotMgr(); |
| } |
| return sRotMgr; |
| } |
| |
| RotMgr::RotMgr() { |
| for(int i = 0; i < MAX_ROT_SESS; i++) { |
| mRot[i] = 0; |
| } |
| mUseCount = 0; |
| mRotDevFd = -1; |
| } |
| |
| RotMgr::~RotMgr() { |
| clear(); |
| } |
| |
| void RotMgr::configBegin() { |
| //Reset the number of objects used |
| mUseCount = 0; |
| } |
| |
| void RotMgr::configDone() { |
| //Remove the top most unused objects. Videos come and go. |
| for(int i = mUseCount; i < MAX_ROT_SESS; i++) { |
| if(mRot[i]) { |
| delete mRot[i]; |
| mRot[i] = 0; |
| } |
| } |
| } |
| |
| Rotator* RotMgr::getNext() { |
| //Return a rot object, creating one if necessary |
| overlay::Rotator *rot = NULL; |
| if(mUseCount >= MAX_ROT_SESS) { |
| ALOGW("%s, MAX rotator sessions reached, request rejected", __func__); |
| } else { |
| if(mRot[mUseCount] == NULL) |
| mRot[mUseCount] = overlay::Rotator::getRotator(); |
| rot = mRot[mUseCount++]; |
| } |
| return rot; |
| } |
| |
| void RotMgr::clear() { |
| //Brute force obj destruction, helpful in suspend. |
| for(int i = 0; i < MAX_ROT_SESS; i++) { |
| if(mRot[i]) { |
| delete mRot[i]; |
| mRot[i] = 0; |
| } |
| } |
| mUseCount = 0; |
| ::close(mRotDevFd); |
| mRotDevFd = -1; |
| } |
| |
| void RotMgr::getDump(char *buf, size_t len) { |
| for(int i = 0; i < MAX_ROT_SESS; i++) { |
| if(mRot[i]) { |
| mRot[i]->getDump(buf, len); |
| } |
| } |
| char str[4] = {'\0'}; |
| snprintf(str, 4, "\n"); |
| strlcat(buf, str, len); |
| } |
| |
| int RotMgr::getRotDevFd() { |
| if(mRotDevFd < 0 && Rotator::getRotatorHwType() == Rotator::TYPE_MDSS) { |
| mRotDevFd = ::open("/dev/graphics/fb0", O_RDWR, 0); |
| if(mRotDevFd < 0) { |
| ALOGE("%s failed to open fb0", __FUNCTION__); |
| } |
| } |
| return mRotDevFd; |
| } |
| |
| } |