You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
192 lines
4.9 KiB
192 lines
4.9 KiB
4 months ago
|
/*
|
||
|
* 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 "gr.h"
|
||
|
|
||
|
namespace ovutils = overlay::utils;
|
||
|
|
||
|
namespace overlay {
|
||
|
|
||
|
//============Rotator=========================
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
|
||
|
//============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::setReleaseFd(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;
|
||
|
}
|
||
|
|
||
|
//============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;
|
||
|
}
|
||
|
|
||
|
}
|