/*
// 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 <math.h>
#include <common/utils/HwcTrace.h>
#include <common/base/Drm.h>
#include <Hwcomposer.h>
#include <ips/anniedale/AnnOverlayPlane.h>
#include <ips/tangier/TngGrallocBuffer.h>
#include <khronos/openmax/OMX_IntelVideoExt.h>
#include <DisplayQuery.h>
namespace android {
namespace intel {
AnnOverlayPlane::AnnOverlayPlane(int index, int disp)
: OverlayPlaneBase(index, disp),
mRotationBufProvider(NULL),
mRotationConfig(0),
mZOrderConfig(0),
mUseOverlayRotation(true)
{
CTRACE();
memset(&mContext, 0, sizeof(mContext));
}
AnnOverlayPlane::~AnnOverlayPlane()
{
CTRACE();
}
bool AnnOverlayPlane::setDataBuffer(uint32_t handle)
{
if (handle == 0) {
ELOGTRACE("handle == 0");
return true;
}
return DisplayPlane::setDataBuffer(handle);
}
void AnnOverlayPlane::setZOrderConfig(ZOrderConfig& /* zorderConfig */,
void *nativeConfig)
{
int slot = (int)nativeConfig;
CTRACE();
switch (slot) {
case 0:
mZOrderConfig = 0;
break;
case 1:
mZOrderConfig = (1 << 8);
break;
case 2:
mZOrderConfig = (2 << 8);
break;
case 3:
mZOrderConfig = (3 << 8);
break;
default:
ELOGTRACE("Invalid overlay plane zorder %d", slot);
return;
}
}
bool AnnOverlayPlane::reset()
{
OverlayPlaneBase::reset();
if (mRotationBufProvider) {
mRotationBufProvider->reset();
}
return true;
}
bool AnnOverlayPlane::enable()
{
RETURN_FALSE_IF_NOT_INIT();
// by default always use overlay rotation
mUseOverlayRotation = true;
if (mContext.ctx.ov_ctx.ovadd & (0x1 << 15))
return true;
mContext.ctx.ov_ctx.ovadd |= (0x1 << 15);
// flush
flush(PLANE_ENABLE);
return true;
}
bool AnnOverlayPlane::disable()
{
RETURN_FALSE_IF_NOT_INIT();
if (!(mContext.ctx.ov_ctx.ovadd & (0x1 << 15)))
return true;
mContext.ctx.ov_ctx.ovadd &= ~(0x1 << 15);
mContext.ctx.ov_ctx.ovadd &= ~(0x300);
mContext.ctx.ov_ctx.ovadd |= mPipeConfig;
// flush
flush(PLANE_DISABLE);
return true;
}
void AnnOverlayPlane::postFlip()
{
// when using AnnOverlayPlane through AnnDisplayPlane as proxy, postFlip is never
// called so mUpdateMasks is never reset.
// When using AnnOverlayPlane directly, postFlip is invoked and mUpdateMasks is reset
// post-flip.
// need to check why mUpdateMasks = 0 causes video freeze.
//DisplayPlane::postFlip();
}
void AnnOverlayPlane::resetBackBuffer(int buf)
{
CTRACE();
if (!mBackBuffer[buf] || !mBackBuffer[buf]->buf)
return;
OverlayBackBufferBlk *backBuffer = mBackBuffer[buf]->buf;
memset(backBuffer, 0, sizeof(OverlayBackBufferBlk));
// reset overlay
backBuffer->OCLRC0 = (OVERLAY_INIT_CONTRAST << 18) |
(OVERLAY_INIT_BRIGHTNESS & 0xff);
backBuffer->OCLRC1 = OVERLAY_INIT_SATURATION;
backBuffer->DCLRKV = OVERLAY_INIT_COLORKEY;
backBuffer->DCLRKM = OVERLAY_INIT_COLORKEYMASK;
backBuffer->OCONFIG = 0;
backBuffer->OCONFIG |= (0x1 << 27);
// use 3 line buffers
backBuffer->OCONFIG |= 0x1;
backBuffer->SCHRKEN &= ~(0x7 << 24);
backBuffer->SCHRKEN |= 0xff;
}
bool AnnOverlayPlane::bufferOffsetSetup(BufferMapper& mapper)
{
CTRACE();
OverlayBackBufferBlk *backBuffer = mBackBuffer[mCurrent]->buf;
if (!backBuffer) {
ELOGTRACE("invalid back buffer");
return false;
}
uint32_t format = mapper.getFormat();
uint32_t gttOffsetInBytes = (mapper.getGttOffsetInPage(0) << 12);
if (format == HAL_PIXEL_FORMAT_BGRX_8888 ||
format == HAL_PIXEL_FORMAT_BGRA_8888) {
backBuffer->OCMD = 1 << 10;
// by pass YUV->RGB conversion, 8-bit output
backBuffer->OCONFIG |= (0x1 << 4) | (0x1 << 3);
backBuffer->OSTART_0Y = gttOffsetInBytes;
backBuffer->OSTART_1Y = gttOffsetInBytes;
backBuffer->OBUF_0Y = 0;
backBuffer->OBUF_1Y = 0;
return true;
}
uint32_t yStride = mapper.getStride().yuv.yStride;
uint32_t uvStride = mapper.getStride().yuv.uvStride;
uint32_t h = mapper.getHeight();
uint32_t srcX= mapper.getCrop().x;
uint32_t srcY= mapper.getCrop().y;
uint32_t ySurface, uSurface, vSurface;
uint32_t yTileOffsetX, yTileOffsetY;
uint32_t uTileOffsetX, uTileOffsetY;
uint32_t vTileOffsetX, vTileOffsetY;
// clear original format setting
backBuffer->OCMD &= ~(0xf << 10);
backBuffer->OCMD &= ~OVERLAY_MEMORY_LAYOUT_TILED;
backBuffer->OBUF_0Y = 0;
backBuffer->OBUF_0V = 0;
backBuffer->OBUF_0U = 0;
// Y/U/V plane must be 4k bytes aligned.
ySurface = gttOffsetInBytes;
if (mIsProtectedBuffer) {
// temporary workaround until vsync event logic is corrected.
// it seems that overlay buffer update and renderring can be overlapped,
// as such encryption bit may be cleared during HW rendering
ySurface |= 0x01;
}
switch(format) {
case HAL_PIXEL_FORMAT_YV12: // YV12
vSurface = ySurface + yStride * h;
uSurface = vSurface + uvStride * (h / 2);
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = srcX / 2;
uTileOffsetY = srcY / 2;
vTileOffsetX = uTileOffsetX;
vTileOffsetY = uTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PLANAR_YUV420;
break;
case HAL_PIXEL_FORMAT_I420: // I420
uSurface = ySurface + yStride * h;
vSurface = uSurface + uvStride * (h / 2);
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = srcX / 2;
uTileOffsetY = srcY / 2;
vTileOffsetX = uTileOffsetX;
vTileOffsetY = uTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PLANAR_YUV420;
break;
case HAL_PIXEL_FORMAT_NV12: // NV12
uSurface = ySurface;
vSurface = ySurface;
backBuffer->OBUF_0U = yStride * h;
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = srcX / 2;
uTileOffsetY = srcY / 2 + h;
vTileOffsetX = uTileOffsetX;
vTileOffsetY = uTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PLANAR_NV12_2;
break;
// NOTE: this is the decoded video format, align the height to 32B
//as it's defined by video driver
case OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar: // NV12
uSurface = ySurface + yStride * align_to(h, 32);
vSurface = ySurface + yStride * align_to(h, 32);
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = srcX;
uTileOffsetY = srcY / 2;
vTileOffsetX = uTileOffsetX;
vTileOffsetY = uTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PLANAR_NV12_2;
break;
case OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar_Tiled: //NV12_tiled
uSurface = ySurface + yStride * align_to(h, 32);
vSurface = ySurface + yStride * align_to(h, 32);
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = srcX;
uTileOffsetY = srcY / 2;
vTileOffsetX = uTileOffsetX;
vTileOffsetY = uTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PLANAR_NV12_2;
backBuffer->OCMD |= OVERLAY_MEMORY_LAYOUT_TILED;
break;
case HAL_PIXEL_FORMAT_YUY2: // YUY2
uSurface = ySurface;
vSurface = ySurface;
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = yTileOffsetX;
uTileOffsetY = yTileOffsetY;
vTileOffsetX = yTileOffsetX;
vTileOffsetY = yTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PACKED_YUV422;
backBuffer->OCMD |= OVERLAY_PACKED_ORDER_YUY2;
break;
case HAL_PIXEL_FORMAT_UYVY: // UYVY
uSurface = ySurface;
vSurface = ySurface;
yTileOffsetX = srcX;
yTileOffsetY = srcY;
uTileOffsetX = yTileOffsetX;
uTileOffsetY = yTileOffsetY;
vTileOffsetX = yTileOffsetX;
vTileOffsetY = yTileOffsetY;
backBuffer->OCMD |= OVERLAY_FORMAT_PACKED_YUV422;
backBuffer->OCMD |= OVERLAY_PACKED_ORDER_UYVY;
break;
default:
ELOGTRACE("unsupported format %d", format);
return false;
}
backBuffer->OSTART_0Y = ySurface;
backBuffer->OSTART_0U = uSurface;
backBuffer->OSTART_0V = vSurface;
backBuffer->OBUF_0Y += srcY * yStride + srcX;
backBuffer->OBUF_0V += (srcY / 2) * uvStride + srcX;
backBuffer->OBUF_0U += (srcY / 2) * uvStride + srcX;
backBuffer->OTILEOFF_0Y = yTileOffsetY << 16 | yTileOffsetX;
backBuffer->OTILEOFF_0U = uTileOffsetY << 16 | uTileOffsetX;
backBuffer->OTILEOFF_0V = vTileOffsetY << 16 | vTileOffsetX;
VLOGTRACE("done. offset (%d, %d, %d)",
backBuffer->OBUF_0Y,
backBuffer->OBUF_0U,
backBuffer->OBUF_0V);
return true;
}
bool AnnOverlayPlane::coordinateSetup(BufferMapper& mapper)
{
CTRACE();
uint32_t format = mapper.getFormat();
if (format != HAL_PIXEL_FORMAT_BGRX_8888 &&
format != HAL_PIXEL_FORMAT_BGRA_8888) {
return OverlayPlaneBase::coordinateSetup(mapper);
}
OverlayBackBufferBlk *backBuffer = mBackBuffer[mCurrent]->buf;
if (!backBuffer) {
ELOGTRACE("invalid back buffer");
return false;
}
backBuffer->SWIDTH = mapper.getCrop().w;
backBuffer->SHEIGHT = mapper.getCrop().h;
backBuffer->SWIDTHSW = calculateSWidthSW(0, mapper.getCrop().w) << 2;
backBuffer->OSTRIDE = mapper.getStride().rgb.stride & (~0x3f);
return true;
};
bool AnnOverlayPlane::scalingSetup(BufferMapper& mapper)
{
int xscaleInt, xscaleFract, yscaleInt, yscaleFract;
int xscaleIntUV, xscaleFractUV;
int yscaleIntUV, yscaleFractUV;
// UV is half the size of Y -- YUV420
int uvratio = 2;
uint32_t newval;
coeffRec xcoeffY[N_HORIZ_Y_TAPS * N_PHASES];
coeffRec xcoeffUV[N_HORIZ_UV_TAPS * N_PHASES];
coeffRec ycoeffY[N_VERT_Y_TAPS * N_PHASES];
coeffRec ycoeffUV[N_VERT_UV_TAPS * N_PHASES];
int i, j, pos;
bool scaleChanged = false;
int x, y, w, h;
int deinterlace_factor = 1;
drmModeModeInfoPtr mode = &mModeInfo;
OverlayBackBufferBlk *backBuffer = mBackBuffer[mCurrent]->buf;
if (!backBuffer) {
ELOGTRACE("invalid back buffer");
return false;
}
if (mPanelOrientation == PANEL_ORIENTATION_180) {
if (mode->hdisplay)
x = mode->hdisplay - mPosition.x - mPosition.w;
else
x = mPosition.x;
if (mode->vdisplay)
y = mode->vdisplay - mPosition.y - mPosition.h;
else
y = mPosition.y;
} else {
x = mPosition.x;
y = mPosition.y;
}
w = mPosition.w;
h = mPosition.h;
// check position
checkPosition(x, y, w, h);
VLOGTRACE("final position (%d, %d, %d, %d)", x, y, w, h);
if ((w <= 0) || (h <= 0)) {
ELOGTRACE("invalid dst width/height");
return false;
}
// setup dst position
backBuffer->DWINPOS = (y << 16) | x;
backBuffer->DWINSZ = (h << 16) | w;
uint32_t srcWidth = mapper.getCrop().w;
uint32_t srcHeight = mapper.getCrop().h;
uint32_t dstWidth = w;
uint32_t dstHeight = h;
uint32_t format = mapper.getFormat();
if (format == HAL_PIXEL_FORMAT_BGRX_8888 ||
format == HAL_PIXEL_FORMAT_BGRA_8888) {
backBuffer->YRGBSCALE = 1 << 15 | 0 << 3 | 0 << 20;
backBuffer->UVSCALEV = (1 << 16);
return true;
}
if (mBobDeinterlace && !mTransform)
deinterlace_factor = 2;
VLOGTRACE("src (%dx%d), dst (%dx%d), transform %d",
srcWidth, srcHeight,
dstWidth, dstHeight,
mTransform);
if (mBobDeinterlace) {
float scaleY = (float)(srcHeight >> 1) / dstHeight;
if (scaleY > 4 || scaleY < 0.25) {
VLOGTRACE("Exceed scale limit for interlace, return false");
return false;
}
}
// switch destination width/height for scale factor calculation
// for 90/270 transformation
if (mUseOverlayRotation && ((mTransform == HWC_TRANSFORM_ROT_90) ||
(mTransform == HWC_TRANSFORM_ROT_270))) {
uint32_t tmp = srcHeight;
srcHeight = srcWidth;
srcWidth = tmp;
}
// Y down-scale factor as a multiple of 4096
if (srcWidth == dstWidth && srcHeight == dstHeight) {
xscaleFract = (1 << 12);
yscaleFract = (1 << 12) / deinterlace_factor;
} else {
xscaleFract = ((srcWidth - 1) << 12) / dstWidth;
yscaleFract = ((srcHeight - 1) << 12) / (dstHeight * deinterlace_factor);
}
// Calculate the UV scaling factor
xscaleFractUV = xscaleFract / uvratio;
yscaleFractUV = yscaleFract / uvratio;
// To keep the relative Y and UV ratios exact, round the Y scales
// to a multiple of the Y/UV ratio.
xscaleFract = xscaleFractUV * uvratio;
yscaleFract = yscaleFractUV * uvratio;
// Integer (un-multiplied) values
xscaleInt = xscaleFract >> 12;
yscaleInt = yscaleFract >> 12;
xscaleIntUV = xscaleFractUV >> 12;
yscaleIntUV = yscaleFractUV >> 12;
// Check scaling ratio
if (xscaleInt > INTEL_OVERLAY_MAX_SCALING_RATIO) {
ELOGTRACE("xscaleInt > %d", INTEL_OVERLAY_MAX_SCALING_RATIO);
return false;
}
// shouldn't get here
if (xscaleIntUV > INTEL_OVERLAY_MAX_SCALING_RATIO) {
ELOGTRACE("xscaleIntUV > %d", INTEL_OVERLAY_MAX_SCALING_RATIO);
return false;
}
newval = (xscaleInt << 15) |
((xscaleFract & 0xFFF) << 3) | ((yscaleFract & 0xFFF) << 20);
if (newval != backBuffer->YRGBSCALE) {
scaleChanged = true;
backBuffer->YRGBSCALE = newval;
}
newval = (xscaleIntUV << 15) | ((xscaleFractUV & 0xFFF) << 3) |
((yscaleFractUV & 0xFFF) << 20);
if (newval != backBuffer->UVSCALE) {
scaleChanged = true;
backBuffer->UVSCALE = newval;
}
newval = yscaleInt << 16 | yscaleIntUV;
if (newval != backBuffer->UVSCALEV) {
scaleChanged = true;
backBuffer->UVSCALEV = newval;
}
// Recalculate coefficients if the scaling changed
// Only Horizontal coefficients so far.
if (scaleChanged) {
double fHCutoffY;
double fHCutoffUV;
double fVCutoffY;
double fVCutoffUV;
fHCutoffY = xscaleFract / 4096.0;
fHCutoffUV = xscaleFractUV / 4096.0;
fVCutoffY = yscaleFract / 4096.0;
fVCutoffUV = yscaleFractUV / 4096.0;
// Limit to between 1.0 and 3.0
if (fHCutoffY < MIN_CUTOFF_FREQ)
fHCutoffY = MIN_CUTOFF_FREQ;
if (fHCutoffY > MAX_CUTOFF_FREQ)
fHCutoffY = MAX_CUTOFF_FREQ;
if (fHCutoffUV < MIN_CUTOFF_FREQ)
fHCutoffUV = MIN_CUTOFF_FREQ;
if (fHCutoffUV > MAX_CUTOFF_FREQ)
fHCutoffUV = MAX_CUTOFF_FREQ;
if (fVCutoffY < MIN_CUTOFF_FREQ)
fVCutoffY = MIN_CUTOFF_FREQ;
if (fVCutoffY > MAX_CUTOFF_FREQ)
fVCutoffY = MAX_CUTOFF_FREQ;
if (fVCutoffUV < MIN_CUTOFF_FREQ)
fVCutoffUV = MIN_CUTOFF_FREQ;
if (fVCutoffUV > MAX_CUTOFF_FREQ)
fVCutoffUV = MAX_CUTOFF_FREQ;
updateCoeff(N_HORIZ_Y_TAPS, fHCutoffY, true, true, xcoeffY);
updateCoeff(N_HORIZ_UV_TAPS, fHCutoffUV, true, false, xcoeffUV);
updateCoeff(N_VERT_Y_TAPS, fVCutoffY, false, true, ycoeffY);
updateCoeff(N_VERT_UV_TAPS, fVCutoffUV, false, false, ycoeffUV);
for (i = 0; i < N_PHASES; i++) {
for (j = 0; j < N_HORIZ_Y_TAPS; j++) {
pos = i * N_HORIZ_Y_TAPS + j;
backBuffer->Y_HCOEFS[pos] =
(xcoeffY[pos].sign << 15 |
xcoeffY[pos].exponent << 12 |
xcoeffY[pos].mantissa);
}
}
for (i = 0; i < N_PHASES; i++) {
for (j = 0; j < N_HORIZ_UV_TAPS; j++) {
pos = i * N_HORIZ_UV_TAPS + j;
backBuffer->UV_HCOEFS[pos] =
(xcoeffUV[pos].sign << 15 |
xcoeffUV[pos].exponent << 12 |
xcoeffUV[pos].mantissa);
}
}
for (i = 0; i < N_PHASES; i++) {
for (j = 0; j < N_VERT_Y_TAPS; j++) {
pos = i * N_VERT_Y_TAPS + j;
backBuffer->Y_VCOEFS[pos] =
(ycoeffY[pos].sign << 15 |
ycoeffY[pos].exponent << 12 |
ycoeffY[pos].mantissa);
}
}
for (i = 0; i < N_PHASES; i++) {
for (j = 0; j < N_VERT_UV_TAPS; j++) {
pos = i * N_VERT_UV_TAPS + j;
backBuffer->UV_VCOEFS[pos] =
(ycoeffUV[pos].sign << 15 |
ycoeffUV[pos].exponent << 12 |
ycoeffUV[pos].mantissa);
}
}
}
XLOGTRACE();
return true;
}
void AnnOverlayPlane::setTransform(int transform)
{
RETURN_VOID_IF_NOT_INIT();
if (mPanelOrientation == PANEL_ORIENTATION_180)
transform ^= HWC_TRANSFORM_ROT_180;
DisplayPlane::setTransform(transform);
// setup transform config
switch (mTransform) {
case HWC_TRANSFORM_ROT_90:
mRotationConfig = (0x1 << 10);
break;
case HWC_TRANSFORM_ROT_180:
mRotationConfig = (0x2 << 10);
break;
case HWC_TRANSFORM_ROT_270:
mRotationConfig = (0x3 << 10);
break;
case 0:
mRotationConfig = 0;
break;
default:
ELOGTRACE("Invalid transform %d", mTransform);
mRotationConfig = 0;
break;
}
}
// HSD 4645510:
// This is a SOC limition, that when source buffer width range is
// in (960, 1024] - one cache line length, and rotation bit is set
// in portrait mode, video will show distortion.
bool AnnOverlayPlane::isSettingRotBitAllowed()
{
uint32_t width = mSrcCrop.w;
if ((width > 960 && width <= 1024) &&
(mTransform == 0 || mTransform == HAL_TRANSFORM_ROT_180))
return false;
return true;
}
bool AnnOverlayPlane::flip(void *ctx)
{
uint32_t ovadd = 0;
RETURN_FALSE_IF_NOT_INIT();
if (!DisplayPlane::flip(ctx)) {
ELOGTRACE("failed to flip display plane.");
return false;
}
// update back buffer address
ovadd = (mBackBuffer[mCurrent]->gttOffsetInPage << 12);
// enable rotation mode and setup rotation config
if (mIndex == 0 && mRotationConfig != 0) {
if (isSettingRotBitAllowed())
ovadd |= (1 << 12);
ovadd |= mRotationConfig;
}
// setup z-order config
ovadd |= mZOrderConfig;
// load coefficients
ovadd |= 0x1;
// enable overlay
ovadd |= (1 << 15);
mContext.type = DC_OVERLAY_PLANE;
mContext.ctx.ov_ctx.ovadd = ovadd;
mContext.ctx.ov_ctx.index = mIndex;
mContext.ctx.ov_ctx.pipe = mDevice;
mContext.ctx.ov_ctx.ovadd |= mPipeConfig;
// move to next back buffer
mCurrent = (mCurrent + 1) % OVERLAY_BACK_BUFFER_COUNT;
VLOGTRACE("ovadd = %#x, index = %d, device = %d",
mContext.ctx.ov_ctx.ovadd,
mIndex,
mDevice);
return true;
}
void* AnnOverlayPlane::getContext() const
{
CTRACE();
return (void *)&mContext;
}
bool AnnOverlayPlane::setDataBuffer(BufferMapper& mapper)
{
if (mIsProtectedBuffer) {
// workaround overlay scaling limitation
float scaleX = (float)mSrcCrop.w/mPosition.w;
float scaleY = (float)mSrcCrop.h/mPosition.h;
if (scaleX > 4.0) {
int crop = (mSrcCrop.w - 4 * mPosition.w)/2 + 1;
mSrcCrop.x += crop;
mSrcCrop.w -= 2 * crop;
}
if (scaleY > 4.0) {
int crop = (mSrcCrop.h - 4 * mPosition.h)/2 + 1;
mSrcCrop.y += crop;
mSrcCrop.h -= 2 * crop;
}
if (scaleX > 4.0 || scaleY > 4.0) {
mUpdateMasks |= PLANE_SOURCE_CROP_CHANGED;
mapper.setCrop(mSrcCrop.x, mSrcCrop.y, mSrcCrop.w, mSrcCrop.h);
}
}
if (OverlayPlaneBase::setDataBuffer(mapper) == false) {
return false;
}
signalVideoRotation(mapper);
if (mIsProtectedBuffer) {
// Bit 0: Decryption request, only allowed to change on a synchronous flip
// This request will be qualified with the separate decryption enable bit for OV
mBackBuffer[mCurrent]->buf->OSTART_0Y |= 0x1;
mBackBuffer[mCurrent]->buf->OSTART_1Y |= 0x1;
}
return true;
}
bool AnnOverlayPlane::initialize(uint32_t bufferCount)
{
if (!OverlayPlaneBase::initialize(bufferCount)) {
ELOGTRACE("failed to initialize OverlayPlaneBase");
return false;
}
// setup rotation buffer
mRotationBufProvider = new RotationBufferProvider(mWsbm);
if (!mRotationBufProvider || !mRotationBufProvider->initialize()) {
DEINIT_AND_RETURN_FALSE("failed to initialize RotationBufferProvider");
}
return true;
}
void AnnOverlayPlane::deinitialize()
{
DEINIT_AND_DELETE_OBJ(mRotationBufProvider);
OverlayPlaneBase::deinitialize();
}
bool AnnOverlayPlane::rotatedBufferReady(BufferMapper& mapper, BufferMapper* &rotatedMapper)
{
struct VideoPayloadBuffer *payload;
uint32_t format;
// only NV12_VED has rotated buffer
format = mapper.getFormat();
if (format != OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar &&
format != OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar_Tiled) {
ELOGTRACE("invalid video format %#x", format);
return false;
}
payload = (struct VideoPayloadBuffer *)mapper.getCpuAddress(SUB_BUFFER1);
// check payload
if (!payload) {
ELOGTRACE("no payload found");
return false;
}
if (payload->force_output_method == FORCE_OUTPUT_GPU) {
ELOGTRACE("Output method is not supported!");
return false;
}
if (payload->client_transform != mTransform ||
mBobDeinterlace) {
if (!mRotationBufProvider->setupRotationBuffer(payload, mTransform)) {
DLOGTRACE("failed to setup rotation buffer");
return false;
}
}
rotatedMapper = getTTMMapper(mapper, payload);
return true;
}
void AnnOverlayPlane::signalVideoRotation(BufferMapper& mapper)
{
struct VideoPayloadBuffer *payload;
uint32_t format;
// check if it's video layer
format = mapper.getFormat();
if (format != OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar &&
format != OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar_Tiled) {
return;
}
payload = (struct VideoPayloadBuffer *)mapper.getCpuAddress(SUB_BUFFER1);
if (!payload) {
ELOGTRACE("no payload found");
return;
}
/* if use overlay rotation, signal decoder to stop rotation */
if (mUseOverlayRotation) {
if (payload->client_transform) {
WLOGTRACE("signal decoder to stop generate rotation buffer");
payload->hwc_timestamp = systemTime();
payload->layer_transform = 0;
}
} else {
/* if overlay rotation cannot be used, signal decoder to start rotation */
if (payload->client_transform != mTransform) {
WLOGTRACE("signal decoder to generate rotation buffer with transform %d", mTransform);
payload->hwc_timestamp = systemTime();
payload->layer_transform = mTransform;
}
}
}
bool AnnOverlayPlane::useOverlayRotation(BufferMapper& /* mapper */)
{
if (mTransform == 0)
return true;
if (!isSettingRotBitAllowed()) {
mUseOverlayRotation = false;
mRotationConfig = 0;
return false;
}
// workaround limitation of overlay rotation by falling back to use VA rotated buffer
bool fallback = false;
float scaleX = (float)mSrcCrop.w / mPosition.w;
float scaleY = (float)mSrcCrop.h / mPosition.h;
if (mTransform == HAL_TRANSFORM_ROT_270 || mTransform == HAL_TRANSFORM_ROT_90) {
scaleX = (float)mSrcCrop.w / mPosition.h;
scaleY = (float)mSrcCrop.h / mPosition.w;
}
if (scaleX >= 3 || scaleY >= 3 || scaleX < 1.0/3 || scaleY < 1.0/3) {
if (mUseOverlayRotation) {
DLOGTRACE("overlay rotation with scaling >= 3, use VA rotated buffer");
}
fallback = true;
} else if ((int)mSrcCrop.x & 63) {
if (mUseOverlayRotation) {
DLOGTRACE("offset is not 64 bytes aligned, use VA rotated buffer");
}
fallback = true;
}
#if 0
else if (mTransform != HAL_TRANSFORM_ROT_180 && scaleX != scaleY) {
if (mUseOverlayRotation) {
DLOGTRACE("overlay rotation with uneven scaling, use VA rotated buffer");
}
fallback = true;
}
#endif
// per DC spec, if video is 1080(H)x1920(V), the buffer
// need 1920 of 64-pixel strip if using hw rotation.
// fallback to video ration buffer in such case.
if (mSrcCrop.w == 1080 && mSrcCrop.h == 1920 && mTransform != 0) {
DLOGTRACE("1080(H)x1920(V) cannot use hw rotation, use VA rotated buffer");
fallback = true;
}
if (fallback || mBobDeinterlace) {
mUseOverlayRotation = false;
mRotationConfig = 0;
} else {
mUseOverlayRotation = true;
}
return mUseOverlayRotation;
}
bool AnnOverlayPlane::flush(uint32_t flags)
{
RETURN_FALSE_IF_NOT_INIT();
ALOGTRACE("flags = %#x, type = %d, index = %d", flags, mType, mIndex);
if (!(flags & PLANE_ENABLE) && !(flags & PLANE_DISABLE)) {
ELOGTRACE("invalid flush flags.");
return false;
}
struct drm_psb_register_rw_arg arg;
memset(&arg, 0, sizeof(struct drm_psb_register_rw_arg));
if (flags & PLANE_DISABLE)
arg.plane_disable_mask = 1;
else if (flags & PLANE_ENABLE)
arg.plane_enable_mask = 1;
arg.plane.type = DC_OVERLAY_PLANE;
arg.plane.index = mIndex;
arg.plane.ctx = mContext.ctx.ov_ctx.ovadd;
if (flags & PLANE_DISABLE) {
DLOGTRACE("disabling overlay %d on device %d", mIndex, mDevice);
}
// issue ioctl
Drm *drm = Hwcomposer::getInstance().getDrm();
bool ret = drm->writeReadIoctl(DRM_PSB_REGISTER_RW, &arg, sizeof(arg));
if (ret == false) {
WLOGTRACE("overlay update failed with error code %d", ret);
return false;
}
return true;
}
} // namespace intel
} // namespace android