1 /*
2 * Copyright (C) 2012-2015, The Linux Foundation. All rights reserved.
3 * Not a Contribution, Apache license notifications and license are retained
4 * for attribution purposes only.
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18
19 #include <math.h>
20 #include "hwc_mdpcomp.h"
21 #include <sys/ioctl.h>
22 #include <dlfcn.h>
23 #include "hdmi.h"
24 #include "qdMetaData.h"
25 #include "mdp_version.h"
26 #include "hwc_fbupdate.h"
27 #include "hwc_ad.h"
28 #include <overlayRotator.h>
29 #include "hwc_copybit.h"
30 #include "qd_utils.h"
31
32 using namespace overlay;
33 using namespace qdutils;
34 using namespace overlay::utils;
35 namespace ovutils = overlay::utils;
36
37 namespace qhwc {
38
39 //==============MDPComp========================================================
40
41 IdleInvalidator *MDPComp::sIdleInvalidator = NULL;
42 bool MDPComp::sIdleFallBack = false;
43 bool MDPComp::sHandleTimeout = false;
44 bool MDPComp::sDebugLogs = false;
45 bool MDPComp::sEnabled = false;
46 bool MDPComp::sEnableMixedMode = true;
47 int MDPComp::sSimulationFlags = 0;
48 int MDPComp::sMaxPipesPerMixer = 0;
49 bool MDPComp::sEnableYUVsplit = false;
50 bool MDPComp::sSrcSplitEnabled = false;
51 int MDPComp::sMaxSecLayers = 1;
52 bool MDPComp::enablePartialUpdateForMDP3 = false;
53 bool MDPComp::sIsPartialUpdateActive = true;
54 bool MDPComp::sIsSingleFullScreenUpdate = false;
55 void *MDPComp::sLibPerfHint = NULL;
56 int MDPComp::sPerfLockHandle = 0;
57 int (*MDPComp::sPerfLockAcquire)(int, int, int*, int) = NULL;
58 int (*MDPComp::sPerfLockRelease)(int value) = NULL;
59 int MDPComp::sPerfHintWindow = -1;
60
getObject(hwc_context_t * ctx,const int & dpy)61 MDPComp* MDPComp::getObject(hwc_context_t *ctx, const int& dpy) {
62 if(qdutils::MDPVersion::getInstance().isSrcSplit()) {
63 sSrcSplitEnabled = true;
64 return new MDPCompSrcSplit(dpy);
65 } else if(isDisplaySplit(ctx, dpy)) {
66 return new MDPCompSplit(dpy);
67 }
68 return new MDPCompNonSplit(dpy);
69 }
70
MDPComp(int dpy)71 MDPComp::MDPComp(int dpy):mDpy(dpy){};
72
dump(android::String8 & buf,hwc_context_t * ctx)73 void MDPComp::dump(android::String8& buf, hwc_context_t *ctx)
74 {
75 if(mCurrentFrame.layerCount > MAX_NUM_APP_LAYERS)
76 return;
77
78 dumpsys_log(buf,"HWC Map for Dpy: %s \n",
79 (mDpy == 0) ? "\"PRIMARY\"" :
80 (mDpy == 1) ? "\"EXTERNAL\"" : "\"VIRTUAL\"");
81 dumpsys_log(buf,"CURR_FRAME: layerCount:%2d mdpCount:%2d "
82 "fbCount:%2d \n", mCurrentFrame.layerCount,
83 mCurrentFrame.mdpCount, mCurrentFrame.fbCount);
84 dumpsys_log(buf,"needsFBRedraw:%3s pipesUsed:%2d MaxPipesPerMixer: %d \n",
85 (mCurrentFrame.needsRedraw? "YES" : "NO"),
86 mCurrentFrame.mdpCount, sMaxPipesPerMixer);
87 if(isDisplaySplit(ctx, mDpy)) {
88 dumpsys_log(buf, "Programmed ROI's: Left: [%d, %d, %d, %d] "
89 "Right: [%d, %d, %d, %d] \n",
90 ctx->listStats[mDpy].lRoi.left, ctx->listStats[mDpy].lRoi.top,
91 ctx->listStats[mDpy].lRoi.right,
92 ctx->listStats[mDpy].lRoi.bottom,
93 ctx->listStats[mDpy].rRoi.left,ctx->listStats[mDpy].rRoi.top,
94 ctx->listStats[mDpy].rRoi.right,
95 ctx->listStats[mDpy].rRoi.bottom);
96 } else {
97 dumpsys_log(buf, "Programmed ROI: [%d, %d, %d, %d] \n",
98 ctx->listStats[mDpy].lRoi.left,ctx->listStats[mDpy].lRoi.top,
99 ctx->listStats[mDpy].lRoi.right,
100 ctx->listStats[mDpy].lRoi.bottom);
101 }
102 dumpsys_log(buf," --------------------------------------------- \n");
103 dumpsys_log(buf," listIdx | cached? | mdpIndex | comptype | Z \n");
104 dumpsys_log(buf," --------------------------------------------- \n");
105 for(int index = 0; index < mCurrentFrame.layerCount; index++ )
106 dumpsys_log(buf," %7d | %7s | %8d | %9s | %2d \n",
107 index,
108 (mCurrentFrame.isFBComposed[index] ? "YES" : "NO"),
109 mCurrentFrame.layerToMDP[index],
110 (mCurrentFrame.isFBComposed[index] ?
111 (mCurrentFrame.drop[index] ? "DROP" :
112 (mCurrentFrame.needsRedraw ? "GLES" : "CACHE")) : "MDP"),
113 (mCurrentFrame.isFBComposed[index] ? mCurrentFrame.fbZ :
114 mCurrentFrame.mdpToLayer[mCurrentFrame.layerToMDP[index]].pipeInfo->zOrder));
115 dumpsys_log(buf,"\n");
116 }
117
init(hwc_context_t * ctx)118 bool MDPComp::init(hwc_context_t *ctx) {
119
120 if(!ctx) {
121 ALOGE("%s: Invalid hwc context!!",__FUNCTION__);
122 return false;
123 }
124
125 char property[PROPERTY_VALUE_MAX] = {0};
126
127 sEnabled = false;
128 if((ctx->mMDP.version >= qdutils::MDP_V4_0) &&
129 (property_get("persist.hwc.mdpcomp.enable", property, NULL) > 0) &&
130 (!strncmp(property, "1", PROPERTY_VALUE_MAX ) ||
131 (!strncasecmp(property,"true", PROPERTY_VALUE_MAX )))) {
132 sEnabled = true;
133 }
134
135 sEnableMixedMode = true;
136 if((property_get("debug.mdpcomp.mixedmode.disable", property, NULL) > 0) &&
137 (!strncmp(property, "1", PROPERTY_VALUE_MAX ) ||
138 (!strncasecmp(property,"true", PROPERTY_VALUE_MAX )))) {
139 sEnableMixedMode = false;
140 }
141
142 qdutils::MDPVersion &mdpVersion = qdutils::MDPVersion::getInstance();
143
144 sMaxPipesPerMixer = (int)mdpVersion.getBlendStages();
145 if(property_get("persist.hwc.mdpcomp.maxpermixer", property, "-1") > 0) {
146 int val = atoi(property);
147 if(val >= 0)
148 sMaxPipesPerMixer = min(val, sMaxPipesPerMixer);
149 }
150
151 /* Maximum layers allowed to use MDP on secondary panels. If property
152 * doesn't exist, default to 1. Using the property it can be set to 0 or
153 * more.
154 */
155 if(property_get("persist.hwc.maxseclayers", property, "1") > 0) {
156 int val = atoi(property);
157 sMaxSecLayers = (val >= 0) ? val : 1;
158 sMaxSecLayers = min(sMaxSecLayers, sMaxPipesPerMixer);
159 }
160
161 if(ctx->mMDP.panel != MIPI_CMD_PANEL) {
162 sIdleInvalidator = IdleInvalidator::getInstance();
163 if(sIdleInvalidator->init(timeout_handler, ctx) < 0) {
164 delete sIdleInvalidator;
165 sIdleInvalidator = NULL;
166 }
167 }
168
169 if(!qdutils::MDPVersion::getInstance().isSrcSplit() &&
170 !qdutils::MDPVersion::getInstance().isRotDownscaleEnabled() &&
171 property_get("persist.mdpcomp.4k2kSplit", property, "0") > 0 &&
172 (!strncmp(property, "1", PROPERTY_VALUE_MAX) ||
173 !strncasecmp(property,"true", PROPERTY_VALUE_MAX))) {
174 sEnableYUVsplit = true;
175 }
176
177 bool defaultPTOR = false;
178 //Enable PTOR when "persist.hwc.ptor.enable" is not defined for
179 //8x16 and 8x39 targets by default
180 if((property_get("persist.hwc.ptor.enable", property, NULL) <= 0) &&
181 (qdutils::MDPVersion::getInstance().is8x16() ||
182 qdutils::MDPVersion::getInstance().is8x39())) {
183 defaultPTOR = true;
184 }
185
186 if (defaultPTOR || (!strncasecmp(property, "true", PROPERTY_VALUE_MAX)) ||
187 (!strncmp(property, "1", PROPERTY_VALUE_MAX ))) {
188 ctx->mCopyBit[HWC_DISPLAY_PRIMARY] = new CopyBit(ctx,
189 HWC_DISPLAY_PRIMARY);
190 }
191
192 if((property_get("persist.mdp3.partialUpdate", property, NULL) <= 0) &&
193 (ctx->mMDP.version == qdutils::MDP_V3_0_5)) {
194 enablePartialUpdateForMDP3 = true;
195 }
196
197 if(!enablePartialUpdateForMDP3 &&
198 (!strncmp(property, "1", PROPERTY_VALUE_MAX ) ||
199 (!strncasecmp(property,"true", PROPERTY_VALUE_MAX )))) {
200 enablePartialUpdateForMDP3 = true;
201 }
202
203 sIsPartialUpdateActive = getPartialUpdatePref(ctx);
204
205 if(property_get("persist.mdpcomp_perfhint", property, "-1") > 0) {
206 int val = atoi(property);
207 if(val > 0 && loadPerfLib()) {
208 sPerfHintWindow = val;
209 ALOGI("PerfHintWindow = %d", sPerfHintWindow);
210 }
211 }
212
213 return true;
214 }
215
reset(hwc_context_t * ctx)216 void MDPComp::reset(hwc_context_t *ctx) {
217 const int numLayers = ctx->listStats[mDpy].numAppLayers;
218 mCurrentFrame.reset(numLayers);
219 ctx->mOverlay->clear(mDpy);
220 ctx->mLayerRotMap[mDpy]->clear();
221 }
222
reset()223 void MDPComp::reset() {
224 sHandleTimeout = false;
225 mModeOn = false;
226 }
227
timeout_handler(void * udata)228 void MDPComp::timeout_handler(void *udata) {
229 struct hwc_context_t* ctx = (struct hwc_context_t*)(udata);
230
231 if(!ctx) {
232 ALOGE("%s: received empty data in timer callback", __FUNCTION__);
233 return;
234 }
235
236 ctx->mDrawLock.lock();
237 // Handle timeout event only if the previous composition is MDP or MIXED.
238 if(!sHandleTimeout) {
239 ALOGD_IF(isDebug(), "%s:Do not handle this timeout", __FUNCTION__);
240 ctx->mDrawLock.unlock();
241 return;
242 }
243 if(!ctx->proc) {
244 ALOGE("%s: HWC proc not registered", __FUNCTION__);
245 ctx->mDrawLock.unlock();
246 return;
247 }
248 sIdleFallBack = true;
249 ctx->mDrawLock.unlock();
250 /* Trigger SF to redraw the current frame */
251 ctx->proc->invalidate(ctx->proc);
252 }
253
setMaxPipesPerMixer(const uint32_t value)254 void MDPComp::setMaxPipesPerMixer(const uint32_t value) {
255 qdutils::MDPVersion &mdpVersion = qdutils::MDPVersion::getInstance();
256 uint32_t maxSupported = (int)mdpVersion.getBlendStages();
257 if(value > maxSupported) {
258 ALOGW("%s: Input exceeds max value supported. Setting to"
259 "max value: %d", __FUNCTION__, maxSupported);
260 }
261 sMaxPipesPerMixer = min(value, maxSupported);
262 }
263
setIdleTimeout(const uint32_t & timeout)264 void MDPComp::setIdleTimeout(const uint32_t& timeout) {
265 enum { ONE_REFRESH_PERIOD_MS = 17, ONE_BILLION_MS = 1000000000 };
266
267 if(sIdleInvalidator) {
268 if(timeout <= ONE_REFRESH_PERIOD_MS) {
269 //If the specified timeout is < 1 draw cycle worth, "virtually"
270 //disable idle timeout. The ideal way for clients to disable
271 //timeout is to set it to 0
272 sIdleInvalidator->setIdleTimeout(ONE_BILLION_MS);
273 ALOGI("Disabled idle timeout");
274 return;
275 }
276 sIdleInvalidator->setIdleTimeout(timeout);
277 ALOGI("Idle timeout set to %u", timeout);
278 } else {
279 ALOGW("Cannot set idle timeout, IdleInvalidator not enabled");
280 }
281 }
282
setMDPCompLayerFlags(hwc_context_t * ctx,hwc_display_contents_1_t * list)283 void MDPComp::setMDPCompLayerFlags(hwc_context_t *ctx,
284 hwc_display_contents_1_t* list) {
285 LayerProp *layerProp = ctx->layerProp[mDpy];
286
287 for(int index = 0; index < ctx->listStats[mDpy].numAppLayers; index++) {
288 hwc_layer_1_t* layer = &(list->hwLayers[index]);
289 if(!mCurrentFrame.isFBComposed[index]) {
290 layerProp[index].mFlags |= HWC_MDPCOMP;
291 layer->compositionType = HWC_OVERLAY;
292 layer->hints |= HWC_HINT_CLEAR_FB;
293 } else {
294 /* Drop the layer when its already present in FB OR when it lies
295 * outside frame's ROI */
296 if(!mCurrentFrame.needsRedraw || mCurrentFrame.drop[index]) {
297 layer->compositionType = HWC_OVERLAY;
298 }
299 }
300 }
301 }
302
setRedraw(hwc_context_t * ctx,hwc_display_contents_1_t * list)303 void MDPComp::setRedraw(hwc_context_t *ctx,
304 hwc_display_contents_1_t* list) {
305 mCurrentFrame.needsRedraw = false;
306 if(!mCachedFrame.isSameFrame(mCurrentFrame, list) ||
307 (list->flags & HWC_GEOMETRY_CHANGED) ||
308 isSkipPresent(ctx, mDpy)) {
309 mCurrentFrame.needsRedraw = true;
310 }
311 }
312
FrameInfo()313 MDPComp::FrameInfo::FrameInfo() {
314 memset(&mdpToLayer, 0, sizeof(mdpToLayer));
315 reset(0);
316 }
317
reset(const int & numLayers)318 void MDPComp::FrameInfo::reset(const int& numLayers) {
319 for(int i = 0 ; i < MAX_NUM_BLEND_STAGES; i++ ) {
320 if(mdpToLayer[i].pipeInfo) {
321 delete mdpToLayer[i].pipeInfo;
322 mdpToLayer[i].pipeInfo = NULL;
323 //We dont own the rotator
324 mdpToLayer[i].rot = NULL;
325 }
326 }
327
328 memset(&mdpToLayer, 0, sizeof(mdpToLayer));
329 memset(&layerToMDP, -1, sizeof(layerToMDP));
330 memset(&isFBComposed, 1, sizeof(isFBComposed));
331
332 layerCount = numLayers;
333 fbCount = numLayers;
334 mdpCount = 0;
335 needsRedraw = true;
336 fbZ = -1;
337 }
338
map()339 void MDPComp::FrameInfo::map() {
340 // populate layer and MDP maps
341 int mdpIdx = 0;
342 for(int idx = 0; idx < layerCount; idx++) {
343 if(!isFBComposed[idx]) {
344 mdpToLayer[mdpIdx].listIndex = idx;
345 layerToMDP[idx] = mdpIdx++;
346 }
347 }
348 }
349
LayerCache()350 MDPComp::LayerCache::LayerCache() {
351 reset();
352 }
353
reset()354 void MDPComp::LayerCache::reset() {
355 memset(&isFBComposed, true, sizeof(isFBComposed));
356 memset(&drop, false, sizeof(drop));
357 layerCount = 0;
358 }
359
updateCounts(const FrameInfo & curFrame)360 void MDPComp::LayerCache::updateCounts(const FrameInfo& curFrame) {
361 layerCount = curFrame.layerCount;
362 memcpy(&isFBComposed, &curFrame.isFBComposed, sizeof(isFBComposed));
363 memcpy(&drop, &curFrame.drop, sizeof(drop));
364 }
365
isSameFrame(const FrameInfo & curFrame,hwc_display_contents_1_t * list)366 bool MDPComp::LayerCache::isSameFrame(const FrameInfo& curFrame,
367 hwc_display_contents_1_t* list) {
368 if(layerCount != curFrame.layerCount)
369 return false;
370 for(int i = 0; i < curFrame.layerCount; i++) {
371 if((curFrame.isFBComposed[i] != isFBComposed[i]) ||
372 (curFrame.drop[i] != drop[i])) {
373 return false;
374 }
375 hwc_layer_1_t const* layer = &list->hwLayers[i];
376 if(curFrame.isFBComposed[i] && layerUpdating(layer)){
377 return false;
378 }
379 }
380 return true;
381 }
382
isSupportedForMDPComp(hwc_context_t * ctx,hwc_layer_1_t * layer)383 bool MDPComp::isSupportedForMDPComp(hwc_context_t *ctx, hwc_layer_1_t* layer) {
384 private_handle_t *hnd = (private_handle_t *)layer->handle;
385 if((has90Transform(layer) and (not isRotationDoable(ctx, hnd))) ||
386 (not isValidDimension(ctx,layer))
387 //More conditions here, SKIP, sRGB+Blend etc
388 ) {
389 return false;
390 }
391 return true;
392 }
393
isValidDimension(hwc_context_t * ctx,hwc_layer_1_t * layer)394 bool MDPComp::isValidDimension(hwc_context_t *ctx, hwc_layer_1_t *layer) {
395 private_handle_t *hnd = (private_handle_t *)layer->handle;
396
397 if(!hnd) {
398 if (layer->flags & HWC_COLOR_FILL) {
399 // Color layer
400 return true;
401 }
402 ALOGD_IF(isDebug(), "%s: layer handle is NULL", __FUNCTION__);
403 return false;
404 }
405
406 //XXX: Investigate doing this with pixel phase on MDSS
407 if(!isSecureBuffer(hnd) && isNonIntegralSourceCrop(layer->sourceCropf))
408 return false;
409
410 hwc_rect_t crop = integerizeSourceCrop(layer->sourceCropf);
411 hwc_rect_t dst = layer->displayFrame;
412 bool rotated90 = (bool)(layer->transform & HAL_TRANSFORM_ROT_90);
413 int crop_w = rotated90 ? crop.bottom - crop.top : crop.right - crop.left;
414 int crop_h = rotated90 ? crop.right - crop.left : crop.bottom - crop.top;
415 int dst_w = dst.right - dst.left;
416 int dst_h = dst.bottom - dst.top;
417 float w_scale = ((float)crop_w / (float)dst_w);
418 float h_scale = ((float)crop_h / (float)dst_h);
419 MDPVersion& mdpHw = MDPVersion::getInstance();
420
421 /* Workaround for MDP HW limitation in DSI command mode panels where
422 * FPS will not go beyond 30 if buffers on RGB pipes are of width or height
423 * less than 5 pixels
424 * There also is a HW limilation in MDP, minimum block size is 2x2
425 * Fallback to GPU if height is less than 2.
426 */
427 if(mdpHw.hasMinCropWidthLimitation() and (crop_w < 5 or crop_h < 5))
428 return false;
429
430 if((w_scale > 1.0f) || (h_scale > 1.0f)) {
431 const uint32_t maxMDPDownscale = mdpHw.getMaxMDPDownscale();
432 const float w_dscale = w_scale;
433 const float h_dscale = h_scale;
434
435 if(ctx->mMDP.version >= qdutils::MDSS_V5) {
436
437 if(!mdpHw.supportsDecimation()) {
438 /* On targets that doesnt support Decimation (eg.,8x26)
439 * maximum downscale support is overlay pipe downscale.
440 */
441 if(crop_w > (int) mdpHw.getMaxPipeWidth() ||
442 w_dscale > maxMDPDownscale ||
443 h_dscale > maxMDPDownscale)
444 return false;
445 } else {
446 // Decimation on macrotile format layers is not supported.
447 if(isTileRendered(hnd)) {
448 /* Bail out if
449 * 1. Src crop > Mixer limit on nonsplit MDPComp
450 * 2. exceeds maximum downscale limit
451 */
452 if(((crop_w > (int) mdpHw.getMaxPipeWidth()) &&
453 !sSrcSplitEnabled) ||
454 w_dscale > maxMDPDownscale ||
455 h_dscale > maxMDPDownscale) {
456 return false;
457 }
458 } else if(w_dscale > 64 || h_dscale > 64)
459 return false;
460 }
461 } else { //A-family
462 if(w_dscale > maxMDPDownscale || h_dscale > maxMDPDownscale)
463 return false;
464 }
465 }
466
467 if((w_scale < 1.0f) || (h_scale < 1.0f)) {
468 const uint32_t upscale = mdpHw.getMaxMDPUpscale();
469 const float w_uscale = 1.0f / w_scale;
470 const float h_uscale = 1.0f / h_scale;
471
472 if(w_uscale > upscale || h_uscale > upscale)
473 return false;
474 }
475
476 return true;
477 }
478
isFrameDoable(hwc_context_t * ctx)479 bool MDPComp::isFrameDoable(hwc_context_t *ctx) {
480 bool ret = true;
481
482 if(!isEnabled()) {
483 ALOGD_IF(isDebug(),"%s: MDP Comp. not enabled.", __FUNCTION__);
484 ret = false;
485 } else if((qdutils::MDPVersion::getInstance().is8x26() ||
486 qdutils::MDPVersion::getInstance().is8x16() ||
487 qdutils::MDPVersion::getInstance().is8x39()) &&
488 ctx->mVideoTransFlag &&
489 isSecondaryConnected(ctx)) {
490 //1 Padding round to shift pipes across mixers
491 ALOGD_IF(isDebug(),"%s: MDP Comp. video transition padding round",
492 __FUNCTION__);
493 ret = false;
494 } else if(qdutils::MDPVersion::getInstance().getTotalPipes() < 8) {
495 /* TODO: freeing up all the resources only for the targets having total
496 number of pipes < 8. Need to analyze number of VIG pipes used
497 for primary in previous draw cycle and accordingly decide
498 whether to fall back to full GPU comp or video only comp
499 */
500 if(isSecondaryConfiguring(ctx)) {
501 ALOGD_IF( isDebug(),"%s: External Display connection is pending",
502 __FUNCTION__);
503 ret = false;
504 } else if(ctx->isPaddingRound) {
505 ALOGD_IF(isDebug(), "%s: padding round invoked for dpy %d",
506 __FUNCTION__,mDpy);
507 ret = false;
508 }
509 } else if (ctx->isDMAStateChanging) {
510 // Bail out if a padding round has been invoked in order to switch DMA
511 // state to block mode. We need this to cater for the case when a layer
512 // requires rotation in the current frame.
513 ALOGD_IF(isDebug(), "%s: padding round invoked to switch DMA state",
514 __FUNCTION__);
515 return false;
516 }
517
518 return ret;
519 }
520
calculateDirtyRect(const hwc_layer_1_t * layer,hwc_rect_t & scissor)521 hwc_rect_t MDPComp::calculateDirtyRect(const hwc_layer_1_t* layer,
522 hwc_rect_t& scissor) {
523 hwc_region_t surfDamage = layer->surfaceDamage;
524 hwc_rect_t src = integerizeSourceCrop(layer->sourceCropf);
525 hwc_rect_t dst = layer->displayFrame;
526 int x_off = dst.left - src.left;
527 int y_off = dst.top - src.top;
528 hwc_rect dirtyRect = (hwc_rect){0, 0, 0, 0};
529 hwc_rect_t updatingRect = dst;
530
531 if (surfDamage.numRects == 0) {
532 // full layer updating, dirty rect is full frame
533 dirtyRect = getIntersection(layer->displayFrame, scissor);
534 } else {
535 for(uint32_t i = 0; i < surfDamage.numRects; i++) {
536 updatingRect = moveRect(surfDamage.rects[i], x_off, y_off);
537 hwc_rect_t intersect = getIntersection(updatingRect, scissor);
538 if(isValidRect(intersect)) {
539 dirtyRect = getUnion(intersect, dirtyRect);
540 }
541 }
542 }
543
544 return dirtyRect;
545 }
546
trimAgainstROI(hwc_context_t * ctx,hwc_rect & crop,hwc_rect & dst)547 void MDPCompNonSplit::trimAgainstROI(hwc_context_t *ctx, hwc_rect &crop,
548 hwc_rect &dst) {
549 hwc_rect_t roi = ctx->listStats[mDpy].lRoi;
550 dst = getIntersection(dst, roi);
551 crop = dst;
552 }
553
554 /* 1) Identify layers that are not visible or lying outside the updating ROI and
555 * drop them from composition.
556 * 2) If we have a scaling layer which needs cropping against generated
557 * ROI, reset ROI to full resolution. */
validateAndApplyROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)558 bool MDPCompNonSplit::validateAndApplyROI(hwc_context_t *ctx,
559 hwc_display_contents_1_t* list) {
560 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
561 hwc_rect_t visibleRect = ctx->listStats[mDpy].lRoi;
562
563 for(int i = numAppLayers - 1; i >= 0; i--){
564 if(!isValidRect(visibleRect)) {
565 mCurrentFrame.drop[i] = true;
566 mCurrentFrame.dropCount++;
567 continue;
568 }
569
570 const hwc_layer_1_t* layer = &list->hwLayers[i];
571 hwc_rect_t dstRect = layer->displayFrame;
572 hwc_rect_t res = getIntersection(visibleRect, dstRect);
573
574 if(!isValidRect(res)) {
575 mCurrentFrame.drop[i] = true;
576 mCurrentFrame.dropCount++;
577 } else {
578 /* Reset frame ROI when any layer which needs scaling also needs ROI
579 * cropping */
580 if(!isSameRect(res, dstRect) && needsScaling (layer)) {
581 ALOGI("%s: Resetting ROI due to scaling", __FUNCTION__);
582 memset(&mCurrentFrame.drop, 0, sizeof(mCurrentFrame.drop));
583 mCurrentFrame.dropCount = 0;
584 return false;
585 }
586
587 /* deduct any opaque region from visibleRect */
588 if (layer->blending == HWC_BLENDING_NONE &&
589 layer->planeAlpha == 0xFF)
590 visibleRect = deductRect(visibleRect, res);
591 }
592 }
593 return true;
594 }
595
596 /* Calculate ROI for the frame by accounting all the layer's dispalyFrame which
597 * are updating. If DirtyRegion is applicable, calculate it by accounting all
598 * the changing layer's dirtyRegion. */
generateROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)599 void MDPCompNonSplit::generateROI(hwc_context_t *ctx,
600 hwc_display_contents_1_t* list) {
601 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
602 if(!canPartialUpdate(ctx, list))
603 return;
604
605 struct hwc_rect roi = (struct hwc_rect){0, 0, 0, 0};
606 hwc_rect fullFrame = (struct hwc_rect) {0, 0,(int)ctx->dpyAttr[mDpy].xres,
607 (int)ctx->dpyAttr[mDpy].yres};
608
609 for(int index = 0; index < numAppLayers; index++ ) {
610 hwc_layer_1_t* layer = &list->hwLayers[index];
611 if (layerUpdating(layer) ||
612 isYuvBuffer((private_handle_t *)layer->handle)) {
613 hwc_rect_t dirtyRect = getIntersection(layer->displayFrame,
614 fullFrame);
615 if(!needsScaling(layer) && !layer->transform) {
616 dirtyRect = calculateDirtyRect(layer, fullFrame);
617 }
618
619 roi = getUnion(roi, dirtyRect);
620 }
621 }
622
623 /* No layer is updating. Still SF wants a refresh.*/
624 if(!isValidRect(roi))
625 return;
626
627 // Align ROI coordinates to panel restrictions
628 roi = getSanitizeROI(roi, fullFrame);
629
630 ctx->listStats[mDpy].lRoi = roi;
631 if(!validateAndApplyROI(ctx, list))
632 resetROI(ctx, mDpy);
633
634 ALOGD_IF(isDebug(),"%s: generated ROI: [%d, %d, %d, %d]", __FUNCTION__,
635 ctx->listStats[mDpy].lRoi.left, ctx->listStats[mDpy].lRoi.top,
636 ctx->listStats[mDpy].lRoi.right, ctx->listStats[mDpy].lRoi.bottom);
637 }
638
trimAgainstROI(hwc_context_t * ctx,hwc_rect & crop,hwc_rect & dst)639 void MDPCompSplit::trimAgainstROI(hwc_context_t *ctx, hwc_rect &crop,
640 hwc_rect &dst) {
641 hwc_rect roi = getUnion(ctx->listStats[mDpy].lRoi,
642 ctx->listStats[mDpy].rRoi);
643 hwc_rect tmpDst = getIntersection(dst, roi);
644 if(!isSameRect(dst, tmpDst)) {
645 crop.left = crop.left + (tmpDst.left - dst.left);
646 crop.top = crop.top + (tmpDst.top - dst.top);
647 crop.right = crop.left + (tmpDst.right - tmpDst.left);
648 crop.bottom = crop.top + (tmpDst.bottom - tmpDst.top);
649 dst = tmpDst;
650 }
651 }
652
653 /* 1) Identify layers that are not visible or lying outside BOTH the updating
654 * ROI's and drop them from composition. If a layer is spanning across both
655 * the halves of the screen but needed by only ROI, the non-contributing
656 * half will not be programmed for MDP.
657 * 2) If we have a scaling layer which needs cropping against generated
658 * ROI, reset ROI to full resolution. */
validateAndApplyROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)659 bool MDPCompSplit::validateAndApplyROI(hwc_context_t *ctx,
660 hwc_display_contents_1_t* list) {
661
662 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
663
664 hwc_rect_t visibleRectL = ctx->listStats[mDpy].lRoi;
665 hwc_rect_t visibleRectR = ctx->listStats[mDpy].rRoi;
666
667 for(int i = numAppLayers - 1; i >= 0; i--){
668 if(!isValidRect(visibleRectL) && !isValidRect(visibleRectR))
669 {
670 mCurrentFrame.drop[i] = true;
671 mCurrentFrame.dropCount++;
672 continue;
673 }
674
675 const hwc_layer_1_t* layer = &list->hwLayers[i];
676 hwc_rect_t dstRect = layer->displayFrame;
677
678 hwc_rect_t l_res = getIntersection(visibleRectL, dstRect);
679 hwc_rect_t r_res = getIntersection(visibleRectR, dstRect);
680 hwc_rect_t res = getUnion(l_res, r_res);
681
682 if(!isValidRect(l_res) && !isValidRect(r_res)) {
683 mCurrentFrame.drop[i] = true;
684 mCurrentFrame.dropCount++;
685 } else {
686 /* Reset frame ROI when any layer which needs scaling also needs ROI
687 * cropping */
688 if(!isSameRect(res, dstRect) && needsScaling (layer)) {
689 memset(&mCurrentFrame.drop, 0, sizeof(mCurrentFrame.drop));
690 mCurrentFrame.dropCount = 0;
691 return false;
692 }
693
694 if (layer->blending == HWC_BLENDING_NONE &&
695 layer->planeAlpha == 0xFF) {
696 visibleRectL = deductRect(visibleRectL, l_res);
697 visibleRectR = deductRect(visibleRectR, r_res);
698 }
699 }
700 }
701 return true;
702 }
703 /* Calculate ROI for the frame by accounting all the layer's dispalyFrame which
704 * are updating. If DirtyRegion is applicable, calculate it by accounting all
705 * the changing layer's dirtyRegion. */
generateROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)706 void MDPCompSplit::generateROI(hwc_context_t *ctx,
707 hwc_display_contents_1_t* list) {
708 if(!canPartialUpdate(ctx, list))
709 return;
710
711 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
712 int lSplit = getLeftSplit(ctx, mDpy);
713
714 int hw_h = (int)ctx->dpyAttr[mDpy].yres;
715 int hw_w = (int)ctx->dpyAttr[mDpy].xres;
716
717 struct hwc_rect l_frame = (struct hwc_rect){0, 0, lSplit, hw_h};
718 struct hwc_rect r_frame = (struct hwc_rect){lSplit, 0, hw_w, hw_h};
719
720 struct hwc_rect l_roi = (struct hwc_rect){0, 0, 0, 0};
721 struct hwc_rect r_roi = (struct hwc_rect){0, 0, 0, 0};
722
723 for(int index = 0; index < numAppLayers; index++ ) {
724 hwc_layer_1_t* layer = &list->hwLayers[index];
725 private_handle_t *hnd = (private_handle_t *)layer->handle;
726
727 if (layerUpdating(layer) || isYuvBuffer(hnd)) {
728 hwc_rect_t l_dirtyRect = getIntersection(layer->displayFrame,
729 l_frame);
730 hwc_rect_t r_dirtyRect = getIntersection(layer->displayFrame,
731 r_frame);
732
733 if(!needsScaling(layer) && !layer->transform) {
734 l_dirtyRect = calculateDirtyRect(layer, l_frame);
735 r_dirtyRect = calculateDirtyRect(layer, r_frame);
736 }
737 if(isValidRect(l_dirtyRect))
738 l_roi = getUnion(l_roi, l_dirtyRect);
739
740 if(isValidRect(r_dirtyRect))
741 r_roi = getUnion(r_roi, r_dirtyRect);
742
743 }
744 }
745
746 /* For panels that cannot accept commands in both the interfaces, we cannot
747 * send two ROI's (for each half). We merge them into single ROI and split
748 * them across lSplit for MDP mixer use. The ROI's will be merged again
749 * finally before udpating the panel in the driver. */
750 if(qdutils::MDPVersion::getInstance().needsROIMerge()) {
751 hwc_rect_t temp_roi = getUnion(l_roi, r_roi);
752 l_roi = getIntersection(temp_roi, l_frame);
753 r_roi = getIntersection(temp_roi, r_frame);
754 }
755
756 /* No layer is updating. Still SF wants a refresh. */
757 if(!isValidRect(l_roi) && !isValidRect(r_roi))
758 return;
759
760 l_roi = getSanitizeROI(l_roi, l_frame);
761 r_roi = getSanitizeROI(r_roi, r_frame);
762
763 ctx->listStats[mDpy].lRoi = l_roi;
764 ctx->listStats[mDpy].rRoi = r_roi;
765
766 if(!validateAndApplyROI(ctx, list))
767 resetROI(ctx, mDpy);
768
769 ALOGD_IF(isDebug(),"%s: generated L_ROI: [%d, %d, %d, %d]"
770 "R_ROI: [%d, %d, %d, %d]", __FUNCTION__,
771 ctx->listStats[mDpy].lRoi.left, ctx->listStats[mDpy].lRoi.top,
772 ctx->listStats[mDpy].lRoi.right, ctx->listStats[mDpy].lRoi.bottom,
773 ctx->listStats[mDpy].rRoi.left, ctx->listStats[mDpy].rRoi.top,
774 ctx->listStats[mDpy].rRoi.right, ctx->listStats[mDpy].rRoi.bottom);
775 }
776
777 /* Checks for conditions where all the layers marked for MDP comp cannot be
778 * bypassed. On such conditions we try to bypass atleast YUV layers */
tryFullFrame(hwc_context_t * ctx,hwc_display_contents_1_t * list)779 bool MDPComp::tryFullFrame(hwc_context_t *ctx,
780 hwc_display_contents_1_t* list){
781
782 const int numAppLayers = ctx->listStats[mDpy].numAppLayers;
783
784 // Fall back to video only composition, if AIV video mode is enabled
785 if(ctx->listStats[mDpy].mAIVVideoMode) {
786 ALOGD_IF(isDebug(), "%s: AIV Video Mode enabled dpy %d",
787 __FUNCTION__, mDpy);
788 return false;
789 }
790
791 /* No Idle fall back if secure display or secure RGB layers are present
792 * or if there is only a single layer being composed */
793 if(sIdleFallBack && !ctx->listStats[mDpy].secureUI &&
794 !ctx->listStats[mDpy].secureRGBCount &&
795 (ctx->listStats[mDpy].numAppLayers > 1)) {
796 ALOGD_IF(isDebug(), "%s: Idle fallback dpy %d",__FUNCTION__, mDpy);
797 return false;
798 }
799
800 if(isSkipPresent(ctx, mDpy)) {
801 ALOGD_IF(isDebug(),"%s: SKIP present: %d",
802 __FUNCTION__,
803 isSkipPresent(ctx, mDpy));
804 return false;
805 }
806
807 // if secondary is configuring or Padding round, fall back to video only
808 // composition and release all assigned non VIG pipes from primary.
809 if(isSecondaryConfiguring(ctx)) {
810 ALOGD_IF( isDebug(),"%s: External Display connection is pending",
811 __FUNCTION__);
812 return false;
813 } else if(ctx->isPaddingRound) {
814 ALOGD_IF(isDebug(), "%s: padding round invoked for dpy %d",
815 __FUNCTION__,mDpy);
816 return false;
817 }
818
819 // check for action safe flag and MDP scaling mode which requires scaling.
820 if(ctx->dpyAttr[mDpy].mActionSafePresent
821 || ctx->dpyAttr[mDpy].mMDPScalingMode) {
822 ALOGD_IF(isDebug(), "%s: Scaling needed for this frame",__FUNCTION__);
823 return false;
824 }
825
826 for(int i = 0; i < numAppLayers; ++i) {
827 hwc_layer_1_t* layer = &list->hwLayers[i];
828 private_handle_t *hnd = (private_handle_t *)layer->handle;
829
830 if(has90Transform(layer) && isRotationDoable(ctx, hnd)) {
831 if(!canUseRotator(ctx, mDpy)) {
832 ALOGD_IF(isDebug(), "%s: Can't use rotator for dpy %d",
833 __FUNCTION__, mDpy);
834 return false;
835 }
836 }
837
838 //For 8x26 with panel width>1k, if RGB layer needs HFLIP fail mdp comp
839 // may not need it if Gfx pre-rotation can handle all flips & rotations
840 MDPVersion& mdpHw = MDPVersion::getInstance();
841 int transform = (layer->flags & HWC_COLOR_FILL) ? 0 : layer->transform;
842 if( mdpHw.is8x26() && (ctx->dpyAttr[mDpy].xres > 1024) &&
843 (transform & HWC_TRANSFORM_FLIP_H) && (!isYuvBuffer(hnd)))
844 return false;
845 }
846
847 if(ctx->mAD->isDoable()) {
848 return false;
849 }
850
851 //If all above hard conditions are met we can do full or partial MDP comp.
852 bool ret = false;
853 if(fullMDPComp(ctx, list)) {
854 ret = true;
855 } else if(fullMDPCompWithPTOR(ctx, list)) {
856 ret = true;
857 } else if(partialMDPComp(ctx, list)) {
858 ret = true;
859 }
860
861 return ret;
862 }
863
fullMDPComp(hwc_context_t * ctx,hwc_display_contents_1_t * list)864 bool MDPComp::fullMDPComp(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
865
866 if(sSimulationFlags & MDPCOMP_AVOID_FULL_MDP)
867 return false;
868
869 const int numAppLayers = ctx->listStats[mDpy].numAppLayers;
870 for(int i = 0; i < numAppLayers; i++) {
871 hwc_layer_1_t* layer = &list->hwLayers[i];
872 if(not mCurrentFrame.drop[i] and
873 not isSupportedForMDPComp(ctx, layer)) {
874 ALOGD_IF(isDebug(), "%s: Unsupported layer in list",__FUNCTION__);
875 return false;
876 }
877 }
878
879 mCurrentFrame.fbCount = 0;
880 memcpy(&mCurrentFrame.isFBComposed, &mCurrentFrame.drop,
881 sizeof(mCurrentFrame.isFBComposed));
882 mCurrentFrame.mdpCount = mCurrentFrame.layerCount - mCurrentFrame.fbCount -
883 mCurrentFrame.dropCount;
884
885 if(sEnableYUVsplit){
886 adjustForSourceSplit(ctx, list);
887 }
888
889 if(!postHeuristicsHandling(ctx, list)) {
890 ALOGD_IF(isDebug(), "post heuristic handling failed");
891 reset(ctx);
892 return false;
893 }
894 ALOGD_IF(sSimulationFlags,"%s: FULL_MDP_COMP SUCCEEDED",
895 __FUNCTION__);
896 return true;
897 }
898
899 /* Full MDP Composition with Peripheral Tiny Overlap Removal.
900 * MDP bandwidth limitations can be avoided, if the overlap region
901 * covered by the smallest layer at a higher z-order, gets composed
902 * by Copybit on a render buffer, which can be queued to MDP.
903 */
fullMDPCompWithPTOR(hwc_context_t * ctx,hwc_display_contents_1_t * list)904 bool MDPComp::fullMDPCompWithPTOR(hwc_context_t *ctx,
905 hwc_display_contents_1_t* list) {
906
907 const int numAppLayers = ctx->listStats[mDpy].numAppLayers;
908 const int stagesForMDP = min(sMaxPipesPerMixer,
909 ctx->mOverlay->availablePipes(mDpy, Overlay::MIXER_DEFAULT));
910
911 // Hard checks where we cannot use this mode
912 if (mDpy || !ctx->mCopyBit[mDpy]) {
913 ALOGD_IF(isDebug(), "%s: Feature not supported!", __FUNCTION__);
914 return false;
915 }
916
917 // Frame level checks
918 if ((numAppLayers > stagesForMDP) || isSkipPresent(ctx, mDpy) ||
919 isYuvPresent(ctx, mDpy) || mCurrentFrame.dropCount ||
920 isSecurePresent(ctx, mDpy)) {
921 ALOGD_IF(isDebug(), "%s: Frame not supported!", __FUNCTION__);
922 return false;
923 }
924 // MDP comp checks
925 for(int i = 0; i < numAppLayers; i++) {
926 hwc_layer_1_t* layer = &list->hwLayers[i];
927 if(not isSupportedForMDPComp(ctx, layer)) {
928 ALOGD_IF(isDebug(), "%s: Unsupported layer in list",__FUNCTION__);
929 return false;
930 }
931 }
932
933 /* We cannot use this composition mode, if:
934 1. A below layer needs scaling.
935 2. Overlap is not peripheral to display.
936 3. Overlap or a below layer has 90 degree transform.
937 4. Overlap area > (1/3 * FrameBuffer) area, based on Perf inputs.
938 */
939
940 int minLayerIndex[MAX_PTOR_LAYERS] = { -1, -1};
941 hwc_rect_t overlapRect[MAX_PTOR_LAYERS];
942 memset(overlapRect, 0, sizeof(overlapRect));
943 int layerPixelCount, minPixelCount = 0;
944 int numPTORLayersFound = 0;
945 for (int i = numAppLayers-1; (i >= 0 &&
946 numPTORLayersFound < MAX_PTOR_LAYERS); i--) {
947 hwc_layer_1_t* layer = &list->hwLayers[i];
948 hwc_rect_t crop = integerizeSourceCrop(layer->sourceCropf);
949 hwc_rect_t dispFrame = layer->displayFrame;
950 layerPixelCount = (crop.right - crop.left) * (crop.bottom - crop.top);
951 // PTOR layer should be peripheral and cannot have transform
952 if (!isPeripheral(dispFrame, ctx->mViewFrame[mDpy]) ||
953 has90Transform(layer)) {
954 continue;
955 }
956 if((3 * (layerPixelCount + minPixelCount)) >
957 ((int)ctx->dpyAttr[mDpy].xres * (int)ctx->dpyAttr[mDpy].yres)) {
958 // Overlap area > (1/3 * FrameBuffer) area, based on Perf inputs.
959 continue;
960 }
961 bool found = false;
962 for (int j = i-1; j >= 0; j--) {
963 // Check if the layers below this layer qualifies for PTOR comp
964 hwc_layer_1_t* layer = &list->hwLayers[j];
965 hwc_rect_t disFrame = layer->displayFrame;
966 // Layer below PTOR is intersecting and has 90 degree transform or
967 // needs scaling cannot be supported.
968 if (isValidRect(getIntersection(dispFrame, disFrame))) {
969 if (has90Transform(layer) || needsScaling(layer)) {
970 found = false;
971 break;
972 }
973 found = true;
974 }
975 }
976 // Store the minLayer Index
977 if(found) {
978 minLayerIndex[numPTORLayersFound] = i;
979 overlapRect[numPTORLayersFound] = list->hwLayers[i].displayFrame;
980 minPixelCount += layerPixelCount;
981 numPTORLayersFound++;
982 }
983 }
984
985 // No overlap layers
986 if (!numPTORLayersFound)
987 return false;
988
989 // Store the displayFrame and the sourceCrops of the layers
990 hwc_rect_t displayFrame[numAppLayers];
991 hwc_rect_t sourceCrop[numAppLayers];
992 for(int i = 0; i < numAppLayers; i++) {
993 hwc_layer_1_t* layer = &list->hwLayers[i];
994 displayFrame[i] = layer->displayFrame;
995 sourceCrop[i] = integerizeSourceCrop(layer->sourceCropf);
996 }
997
998 /**
999 * It's possible that 2 PTOR layers might have overlapping.
1000 * In such case, remove the intersection(again if peripheral)
1001 * from the lower PTOR layer to avoid overlapping.
1002 * If intersection is not on peripheral then compromise
1003 * by reducing number of PTOR layers.
1004 **/
1005 hwc_rect_t commonRect = getIntersection(overlapRect[0], overlapRect[1]);
1006 if(isValidRect(commonRect)) {
1007 overlapRect[1] = deductRect(overlapRect[1], commonRect);
1008 list->hwLayers[minLayerIndex[1]].displayFrame = overlapRect[1];
1009 }
1010
1011 ctx->mPtorInfo.count = numPTORLayersFound;
1012 for(int i = 0; i < MAX_PTOR_LAYERS; i++) {
1013 ctx->mPtorInfo.layerIndex[i] = minLayerIndex[i];
1014 }
1015
1016 if (!ctx->mCopyBit[mDpy]->prepareOverlap(ctx, list)) {
1017 // reset PTOR
1018 ctx->mPtorInfo.count = 0;
1019 if(isValidRect(commonRect)) {
1020 // If PTORs are intersecting restore displayframe of PTOR[1]
1021 // before returning, as we have modified it above.
1022 list->hwLayers[minLayerIndex[1]].displayFrame =
1023 displayFrame[minLayerIndex[1]];
1024 }
1025 return false;
1026 }
1027 private_handle_t *renderBuf = ctx->mCopyBit[mDpy]->getCurrentRenderBuffer();
1028 Whf layerWhf[numPTORLayersFound]; // To store w,h,f of PTOR layers
1029
1030 // Store the blending mode, planeAlpha, and transform of PTOR layers
1031 int32_t blending[numPTORLayersFound];
1032 uint8_t planeAlpha[numPTORLayersFound];
1033 uint32_t transform[numPTORLayersFound];
1034
1035 for(int j = 0; j < numPTORLayersFound; j++) {
1036 int index = ctx->mPtorInfo.layerIndex[j];
1037
1038 // Update src crop of PTOR layer
1039 hwc_layer_1_t* layer = &list->hwLayers[index];
1040 layer->sourceCropf.left = (float)ctx->mPtorInfo.displayFrame[j].left;
1041 layer->sourceCropf.top = (float)ctx->mPtorInfo.displayFrame[j].top;
1042 layer->sourceCropf.right = (float)ctx->mPtorInfo.displayFrame[j].right;
1043 layer->sourceCropf.bottom =(float)ctx->mPtorInfo.displayFrame[j].bottom;
1044
1045 // Store & update w, h, format of PTOR layer
1046 private_handle_t *hnd = (private_handle_t *)layer->handle;
1047 Whf whf(hnd->width, hnd->height, hnd->format, hnd->size);
1048 layerWhf[j] = whf;
1049 hnd->width = renderBuf->width;
1050 hnd->height = renderBuf->height;
1051 hnd->format = renderBuf->format;
1052
1053 // Store & update blending mode, planeAlpha and transform of PTOR layer
1054 blending[j] = layer->blending;
1055 planeAlpha[j] = layer->planeAlpha;
1056 transform[j] = layer->transform;
1057 layer->blending = HWC_BLENDING_NONE;
1058 layer->planeAlpha = 0xFF;
1059 layer->transform = 0;
1060
1061 // Remove overlap from crop & displayFrame of below layers
1062 for (int i = 0; i < index && index !=-1; i++) {
1063 layer = &list->hwLayers[i];
1064 if(!isValidRect(getIntersection(layer->displayFrame,
1065 overlapRect[j]))) {
1066 continue;
1067 }
1068 // Update layer attributes
1069 hwc_rect_t srcCrop = integerizeSourceCrop(layer->sourceCropf);
1070 hwc_rect_t destRect = deductRect(layer->displayFrame,
1071 getIntersection(layer->displayFrame, overlapRect[j]));
1072 qhwc::calculate_crop_rects(srcCrop, layer->displayFrame, destRect,
1073 layer->transform);
1074 layer->sourceCropf.left = (float)srcCrop.left;
1075 layer->sourceCropf.top = (float)srcCrop.top;
1076 layer->sourceCropf.right = (float)srcCrop.right;
1077 layer->sourceCropf.bottom = (float)srcCrop.bottom;
1078 }
1079 }
1080
1081 mCurrentFrame.mdpCount = numAppLayers;
1082 mCurrentFrame.fbCount = 0;
1083 mCurrentFrame.fbZ = -1;
1084
1085 for (int j = 0; j < numAppLayers; j++) {
1086 if(isValidRect(list->hwLayers[j].displayFrame)) {
1087 mCurrentFrame.isFBComposed[j] = false;
1088 } else {
1089 mCurrentFrame.mdpCount--;
1090 mCurrentFrame.drop[j] = true;
1091 }
1092 }
1093
1094 bool result = postHeuristicsHandling(ctx, list);
1095
1096 // Restore layer attributes
1097 for(int i = 0; i < numAppLayers; i++) {
1098 hwc_layer_1_t* layer = &list->hwLayers[i];
1099 layer->displayFrame = displayFrame[i];
1100 layer->sourceCropf.left = (float)sourceCrop[i].left;
1101 layer->sourceCropf.top = (float)sourceCrop[i].top;
1102 layer->sourceCropf.right = (float)sourceCrop[i].right;
1103 layer->sourceCropf.bottom = (float)sourceCrop[i].bottom;
1104 }
1105
1106 // Restore w,h,f, blending attributes, and transform of PTOR layers
1107 for (int i = 0; i < numPTORLayersFound; i++) {
1108 int idx = ctx->mPtorInfo.layerIndex[i];
1109 hwc_layer_1_t* layer = &list->hwLayers[idx];
1110 private_handle_t *hnd = (private_handle_t *)list->hwLayers[idx].handle;
1111 hnd->width = layerWhf[i].w;
1112 hnd->height = layerWhf[i].h;
1113 hnd->format = layerWhf[i].format;
1114 layer->blending = blending[i];
1115 layer->planeAlpha = planeAlpha[i];
1116 layer->transform = transform[i];
1117 }
1118
1119 if (!result) {
1120 // reset PTOR
1121 ctx->mPtorInfo.count = 0;
1122 reset(ctx);
1123 } else {
1124 ALOGD_IF(isDebug(), "%s: PTOR Indexes: %d and %d", __FUNCTION__,
1125 ctx->mPtorInfo.layerIndex[0], ctx->mPtorInfo.layerIndex[1]);
1126 }
1127
1128 ALOGD_IF(isDebug(), "%s: Postheuristics %s!", __FUNCTION__,
1129 (result ? "successful" : "failed"));
1130 return result;
1131 }
1132
partialMDPComp(hwc_context_t * ctx,hwc_display_contents_1_t * list)1133 bool MDPComp::partialMDPComp(hwc_context_t *ctx, hwc_display_contents_1_t* list)
1134 {
1135 if(!sEnableMixedMode) {
1136 //Mixed mode is disabled. No need to even try caching.
1137 return false;
1138 }
1139
1140 bool ret = false;
1141 if(list->flags & HWC_GEOMETRY_CHANGED) { //Try load based first
1142 ret = loadBasedComp(ctx, list) or
1143 cacheBasedComp(ctx, list);
1144 } else {
1145 ret = cacheBasedComp(ctx, list) or
1146 loadBasedComp(ctx, list);
1147 }
1148
1149 return ret;
1150 }
1151
cacheBasedComp(hwc_context_t * ctx,hwc_display_contents_1_t * list)1152 bool MDPComp::cacheBasedComp(hwc_context_t *ctx,
1153 hwc_display_contents_1_t* list) {
1154 if(sSimulationFlags & MDPCOMP_AVOID_CACHE_MDP)
1155 return false;
1156
1157 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
1158 mCurrentFrame.reset(numAppLayers);
1159 updateLayerCache(ctx, list, mCurrentFrame);
1160
1161 //If an MDP marked layer is unsupported cannot do partial MDP Comp
1162 for(int i = 0; i < numAppLayers; i++) {
1163 if(!mCurrentFrame.isFBComposed[i]) {
1164 hwc_layer_1_t* layer = &list->hwLayers[i];
1165 if(not isSupportedForMDPComp(ctx, layer)) {
1166 ALOGD_IF(isDebug(), "%s: Unsupported layer in list",
1167 __FUNCTION__);
1168 reset(ctx);
1169 return false;
1170 }
1171 }
1172 }
1173
1174 updateYUV(ctx, list, false /*secure only*/, mCurrentFrame);
1175 /* mark secure RGB layers for MDP comp */
1176 updateSecureRGB(ctx, list);
1177 bool ret = markLayersForCaching(ctx, list); //sets up fbZ also
1178 if(!ret) {
1179 ALOGD_IF(isDebug(),"%s: batching failed, dpy %d",__FUNCTION__, mDpy);
1180 reset(ctx);
1181 return false;
1182 }
1183
1184 int mdpCount = mCurrentFrame.mdpCount;
1185
1186 if(sEnableYUVsplit){
1187 adjustForSourceSplit(ctx, list);
1188 }
1189
1190 if(!postHeuristicsHandling(ctx, list)) {
1191 ALOGD_IF(isDebug(), "post heuristic handling failed");
1192 reset(ctx);
1193 return false;
1194 }
1195 ALOGD_IF(sSimulationFlags,"%s: CACHE_MDP_COMP SUCCEEDED",
1196 __FUNCTION__);
1197
1198 return true;
1199 }
1200
loadBasedComp(hwc_context_t * ctx,hwc_display_contents_1_t * list)1201 bool MDPComp::loadBasedComp(hwc_context_t *ctx,
1202 hwc_display_contents_1_t* list) {
1203 if(sSimulationFlags & MDPCOMP_AVOID_LOAD_MDP)
1204 return false;
1205
1206 if(not isLoadBasedCompDoable(ctx)) {
1207 return false;
1208 }
1209
1210 const int numAppLayers = ctx->listStats[mDpy].numAppLayers;
1211 const int numNonDroppedLayers = numAppLayers - mCurrentFrame.dropCount;
1212 const int stagesForMDP = min(sMaxPipesPerMixer,
1213 ctx->mOverlay->availablePipes(mDpy, Overlay::MIXER_DEFAULT));
1214
1215 int mdpBatchSize = stagesForMDP - 1; //1 stage for FB
1216 int fbBatchSize = numNonDroppedLayers - mdpBatchSize;
1217 int lastMDPSupportedIndex = numAppLayers;
1218 int dropCount = 0;
1219
1220 //Find the minimum MDP batch size
1221 for(int i = 0; i < numAppLayers;i++) {
1222 if(mCurrentFrame.drop[i]) {
1223 dropCount++;
1224 continue;
1225 }
1226 hwc_layer_1_t* layer = &list->hwLayers[i];
1227 if(not isSupportedForMDPComp(ctx, layer)) {
1228 lastMDPSupportedIndex = i;
1229 mdpBatchSize = min(i - dropCount, stagesForMDP - 1);
1230 fbBatchSize = numNonDroppedLayers - mdpBatchSize;
1231 break;
1232 }
1233 }
1234
1235 ALOGD_IF(isDebug(), "%s:Before optimizing fbBatch, mdpbatch %d, fbbatch %d "
1236 "dropped %d", __FUNCTION__, mdpBatchSize, fbBatchSize,
1237 mCurrentFrame.dropCount);
1238
1239 //Start at a point where the fb batch should at least have 2 layers, for
1240 //this mode to be justified.
1241 while(fbBatchSize < 2) {
1242 ++fbBatchSize;
1243 --mdpBatchSize;
1244 }
1245
1246 //If there are no layers for MDP, this mode doesnt make sense.
1247 if(mdpBatchSize < 1) {
1248 ALOGD_IF(isDebug(), "%s: No MDP layers after optimizing for fbBatch",
1249 __FUNCTION__);
1250 return false;
1251 }
1252
1253 mCurrentFrame.reset(numAppLayers);
1254
1255 //Try with successively smaller mdp batch sizes until we succeed or reach 1
1256 while(mdpBatchSize > 0) {
1257 //Mark layers for MDP comp
1258 int mdpBatchLeft = mdpBatchSize;
1259 for(int i = 0; i < lastMDPSupportedIndex and mdpBatchLeft; i++) {
1260 if(mCurrentFrame.drop[i]) {
1261 continue;
1262 }
1263 mCurrentFrame.isFBComposed[i] = false;
1264 --mdpBatchLeft;
1265 }
1266
1267 mCurrentFrame.fbZ = mdpBatchSize;
1268 mCurrentFrame.fbCount = fbBatchSize;
1269 mCurrentFrame.mdpCount = mdpBatchSize;
1270
1271 ALOGD_IF(isDebug(), "%s:Trying with: mdpbatch %d fbbatch %d dropped %d",
1272 __FUNCTION__, mdpBatchSize, fbBatchSize,
1273 mCurrentFrame.dropCount);
1274
1275 if(postHeuristicsHandling(ctx, list)) {
1276 ALOGD_IF(isDebug(), "%s: Postheuristics handling succeeded",
1277 __FUNCTION__);
1278 ALOGD_IF(sSimulationFlags,"%s: LOAD_MDP_COMP SUCCEEDED",
1279 __FUNCTION__);
1280 return true;
1281 }
1282
1283 reset(ctx);
1284 --mdpBatchSize;
1285 ++fbBatchSize;
1286 }
1287
1288 return false;
1289 }
1290
isLoadBasedCompDoable(hwc_context_t * ctx)1291 bool MDPComp::isLoadBasedCompDoable(hwc_context_t *ctx) {
1292 if(mDpy or isSecurePresent(ctx, mDpy) or
1293 isYuvPresent(ctx, mDpy)) {
1294 return false;
1295 }
1296 return true;
1297 }
1298
canPartialUpdate(hwc_context_t * ctx,hwc_display_contents_1_t * list)1299 bool MDPComp::canPartialUpdate(hwc_context_t *ctx,
1300 hwc_display_contents_1_t* list){
1301 if(!qdutils::MDPVersion::getInstance().isPartialUpdateEnabled() ||
1302 isSkipPresent(ctx, mDpy) || (list->flags & HWC_GEOMETRY_CHANGED) ||
1303 !sIsPartialUpdateActive || mDpy ) {
1304 return false;
1305 }
1306 if(ctx->listStats[mDpy].secureUI)
1307 return false;
1308 if (sIsSingleFullScreenUpdate) {
1309 // make sure one full screen update
1310 sIsSingleFullScreenUpdate = false;
1311 return false;
1312 }
1313 return true;
1314 }
1315
tryVideoOnly(hwc_context_t * ctx,hwc_display_contents_1_t * list)1316 bool MDPComp::tryVideoOnly(hwc_context_t *ctx,
1317 hwc_display_contents_1_t* list) {
1318 const bool secureOnly = true;
1319 return videoOnlyComp(ctx, list, not secureOnly) or
1320 videoOnlyComp(ctx, list, secureOnly);
1321 }
1322
videoOnlyComp(hwc_context_t * ctx,hwc_display_contents_1_t * list,bool secureOnly)1323 bool MDPComp::videoOnlyComp(hwc_context_t *ctx,
1324 hwc_display_contents_1_t* list, bool secureOnly) {
1325 if(sSimulationFlags & MDPCOMP_AVOID_VIDEO_ONLY)
1326 return false;
1327
1328 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
1329 if(!isSecurePresent(ctx, mDpy)) {
1330 /* Bail out if we are processing only secured video layers
1331 * and we dont have any */
1332 if(secureOnly) {
1333 ALOGD_IF(isDebug(),"%s: No Secure Video Layers", __FUNCTION__);
1334 return false;
1335 }
1336 /* No Idle fall back for secure video layers and if there is only
1337 * single layer being composed. */
1338 if(sIdleFallBack && (ctx->listStats[mDpy].numAppLayers > 1)) {
1339 ALOGD_IF(isDebug(), "%s: Idle fallback dpy %d",__FUNCTION__, mDpy);
1340 return false;
1341 }
1342 }
1343
1344 mCurrentFrame.reset(numAppLayers);
1345 mCurrentFrame.fbCount -= mCurrentFrame.dropCount;
1346 updateYUV(ctx, list, secureOnly, mCurrentFrame);
1347 int mdpCount = mCurrentFrame.mdpCount;
1348
1349 if(!isYuvPresent(ctx, mDpy) or (mdpCount == 0)) {
1350 reset(ctx);
1351 return false;
1352 }
1353
1354 if(mCurrentFrame.fbCount)
1355 mCurrentFrame.fbZ = mCurrentFrame.mdpCount;
1356
1357 if(sEnableYUVsplit){
1358 adjustForSourceSplit(ctx, list);
1359 }
1360
1361 if(!postHeuristicsHandling(ctx, list)) {
1362 ALOGD_IF(isDebug(), "post heuristic handling failed");
1363 if(errno == ENOBUFS) {
1364 ALOGD_IF(isDebug(), "SMP Allocation failed");
1365 //On SMP allocation failure in video only comp add padding round
1366 ctx->isPaddingRound = true;
1367 }
1368 reset(ctx);
1369 return false;
1370 }
1371
1372 ALOGD_IF(sSimulationFlags,"%s: VIDEO_ONLY_COMP SUCCEEDED",
1373 __FUNCTION__);
1374 return true;
1375 }
1376
1377 /* if tryFullFrame fails, try to push all video and secure RGB layers to MDP */
tryMDPOnlyLayers(hwc_context_t * ctx,hwc_display_contents_1_t * list)1378 bool MDPComp::tryMDPOnlyLayers(hwc_context_t *ctx,
1379 hwc_display_contents_1_t* list) {
1380 // Fall back to video only composition, if AIV video mode is enabled
1381 if(ctx->listStats[mDpy].mAIVVideoMode) {
1382 ALOGD_IF(isDebug(), "%s: AIV Video Mode enabled dpy %d",
1383 __FUNCTION__, mDpy);
1384 return false;
1385 }
1386
1387 const bool secureOnly = true;
1388 return mdpOnlyLayersComp(ctx, list, not secureOnly) or
1389 mdpOnlyLayersComp(ctx, list, secureOnly);
1390
1391 }
1392
mdpOnlyLayersComp(hwc_context_t * ctx,hwc_display_contents_1_t * list,bool secureOnly)1393 bool MDPComp::mdpOnlyLayersComp(hwc_context_t *ctx,
1394 hwc_display_contents_1_t* list, bool secureOnly) {
1395
1396 if(sSimulationFlags & MDPCOMP_AVOID_MDP_ONLY_LAYERS)
1397 return false;
1398
1399 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
1400 if(!isSecurePresent(ctx, mDpy) && !ctx->listStats[mDpy].secureUI) {
1401 /* Bail out if we are processing only secured video/ui layers
1402 * and we dont have any */
1403 if(secureOnly) {
1404 ALOGD_IF(isDebug(), "%s: No secure video/ui layers", __FUNCTION__);
1405 return false;
1406 }
1407 /* No Idle fall back for secure video/ui layers and if there is only
1408 * single layer being composed. */
1409 if(sIdleFallBack && (ctx->listStats[mDpy].numAppLayers > 1)) {
1410 ALOGD_IF(isDebug(), "%s: Idle fallback dpy %d",__FUNCTION__, mDpy);
1411 return false;
1412 }
1413 }
1414
1415 /* Bail out if we dont have any secure RGB layers */
1416 if (!ctx->listStats[mDpy].secureRGBCount) {
1417 reset(ctx);
1418 return false;
1419 }
1420
1421 mCurrentFrame.reset(numAppLayers);
1422 mCurrentFrame.fbCount -= mCurrentFrame.dropCount;
1423
1424 updateYUV(ctx, list, secureOnly, mCurrentFrame);
1425 /* mark secure RGB layers for MDP comp */
1426 updateSecureRGB(ctx, list);
1427
1428 if(mCurrentFrame.mdpCount == 0) {
1429 reset(ctx);
1430 return false;
1431 }
1432
1433 /* find the maximum batch of layers to be marked for framebuffer */
1434 bool ret = markLayersForCaching(ctx, list); //sets up fbZ also
1435 if(!ret) {
1436 ALOGD_IF(isDebug(),"%s: batching failed, dpy %d",__FUNCTION__, mDpy);
1437 reset(ctx);
1438 return false;
1439 }
1440
1441 if(sEnableYUVsplit){
1442 adjustForSourceSplit(ctx, list);
1443 }
1444
1445 if(!postHeuristicsHandling(ctx, list)) {
1446 ALOGD_IF(isDebug(), "post heuristic handling failed");
1447 reset(ctx);
1448 return false;
1449 }
1450
1451 ALOGD_IF(sSimulationFlags,"%s: MDP_ONLY_LAYERS_COMP SUCCEEDED",
1452 __FUNCTION__);
1453 return true;
1454 }
1455
1456 /* Checks for conditions where YUV layers cannot be bypassed */
isYUVDoable(hwc_context_t * ctx,hwc_layer_1_t * layer)1457 bool MDPComp::isYUVDoable(hwc_context_t* ctx, hwc_layer_1_t* layer) {
1458 if(isSkipLayer(layer)) {
1459 ALOGD_IF(isDebug(), "%s: Video marked SKIP dpy %d", __FUNCTION__, mDpy);
1460 return false;
1461 }
1462
1463 if(has90Transform(layer) && !canUseRotator(ctx, mDpy)) {
1464 ALOGD_IF(isDebug(), "%s: no free DMA pipe",__FUNCTION__);
1465 return false;
1466 }
1467
1468 if(isSecuring(ctx, layer)) {
1469 ALOGD_IF(isDebug(), "%s: MDP securing is active", __FUNCTION__);
1470 return false;
1471 }
1472
1473 if(!isValidDimension(ctx, layer)) {
1474 ALOGD_IF(isDebug(), "%s: Buffer is of invalid width",
1475 __FUNCTION__);
1476 return false;
1477 }
1478
1479 if(layer->planeAlpha < 0xFF) {
1480 ALOGD_IF(isDebug(), "%s: Cannot handle YUV layer with plane alpha\
1481 in video only mode",
1482 __FUNCTION__);
1483 return false;
1484 }
1485
1486 return true;
1487 }
1488
1489 /* Checks for conditions where Secure RGB layers cannot be bypassed */
isSecureRGBDoable(hwc_context_t * ctx,hwc_layer_1_t * layer)1490 bool MDPComp::isSecureRGBDoable(hwc_context_t* ctx, hwc_layer_1_t* layer) {
1491 if(isSkipLayer(layer)) {
1492 ALOGD_IF(isDebug(), "%s: Secure RGB layer marked SKIP dpy %d",
1493 __FUNCTION__, mDpy);
1494 return false;
1495 }
1496
1497 if(isSecuring(ctx, layer)) {
1498 ALOGD_IF(isDebug(), "%s: MDP securing is active", __FUNCTION__);
1499 return false;
1500 }
1501
1502 if(not isSupportedForMDPComp(ctx, layer)) {
1503 ALOGD_IF(isDebug(), "%s: Unsupported secure RGB layer",
1504 __FUNCTION__);
1505 return false;
1506 }
1507 return true;
1508 }
1509
1510 /* starts at fromIndex and check for each layer to find
1511 * if it it has overlapping with any Updating layer above it in zorder
1512 * till the end of the batch. returns true if it finds any intersection */
canPushBatchToTop(const hwc_display_contents_1_t * list,int fromIndex,int toIndex)1513 bool MDPComp::canPushBatchToTop(const hwc_display_contents_1_t* list,
1514 int fromIndex, int toIndex) {
1515 for(int i = fromIndex; i < toIndex; i++) {
1516 if(mCurrentFrame.isFBComposed[i] && !mCurrentFrame.drop[i]) {
1517 if(intersectingUpdatingLayers(list, i+1, toIndex, i)) {
1518 return false;
1519 }
1520 }
1521 }
1522 return true;
1523 }
1524
1525 /* Checks if given layer at targetLayerIndex has any
1526 * intersection with all the updating layers in beween
1527 * fromIndex and toIndex. Returns true if it finds intersectiion */
intersectingUpdatingLayers(const hwc_display_contents_1_t * list,int fromIndex,int toIndex,int targetLayerIndex)1528 bool MDPComp::intersectingUpdatingLayers(const hwc_display_contents_1_t* list,
1529 int fromIndex, int toIndex, int targetLayerIndex) {
1530 for(int i = fromIndex; i <= toIndex; i++) {
1531 if(!mCurrentFrame.isFBComposed[i]) {
1532 if(areLayersIntersecting(&list->hwLayers[i],
1533 &list->hwLayers[targetLayerIndex])) {
1534 return true;
1535 }
1536 }
1537 }
1538 return false;
1539 }
1540
getBatch(hwc_display_contents_1_t * list,int & maxBatchStart,int & maxBatchEnd,int & maxBatchCount)1541 int MDPComp::getBatch(hwc_display_contents_1_t* list,
1542 int& maxBatchStart, int& maxBatchEnd,
1543 int& maxBatchCount) {
1544 int i = 0;
1545 int fbZOrder =-1;
1546 int droppedLayerCt = 0;
1547 while (i < mCurrentFrame.layerCount) {
1548 int batchCount = 0;
1549 int batchStart = i;
1550 int batchEnd = i;
1551 /* Adjust batch Z order with the dropped layers so far */
1552 int fbZ = batchStart - droppedLayerCt;
1553 int firstZReverseIndex = -1;
1554 int updatingLayersAbove = 0;//Updating layer count in middle of batch
1555 while(i < mCurrentFrame.layerCount) {
1556 if(!mCurrentFrame.isFBComposed[i]) {
1557 if(!batchCount) {
1558 i++;
1559 break;
1560 }
1561 updatingLayersAbove++;
1562 i++;
1563 continue;
1564 } else {
1565 if(mCurrentFrame.drop[i]) {
1566 i++;
1567 droppedLayerCt++;
1568 continue;
1569 } else if(updatingLayersAbove <= 0) {
1570 batchCount++;
1571 batchEnd = i;
1572 i++;
1573 continue;
1574 } else { //Layer is FBComposed, not a drop & updatingLayer > 0
1575
1576 // We have a valid updating layer already. If layer-i not
1577 // have overlapping with all updating layers in between
1578 // batch-start and i, then we can add layer i to batch.
1579 if(!intersectingUpdatingLayers(list, batchStart, i-1, i)) {
1580 batchCount++;
1581 batchEnd = i;
1582 i++;
1583 continue;
1584 } else if(canPushBatchToTop(list, batchStart, i)) {
1585 //If All the non-updating layers with in this batch
1586 //does not have intersection with the updating layers
1587 //above in z-order, then we can safely move the batch to
1588 //higher z-order. Increment fbZ as it is moving up.
1589 if( firstZReverseIndex < 0) {
1590 firstZReverseIndex = i;
1591 }
1592 batchCount++;
1593 batchEnd = i;
1594 fbZ += updatingLayersAbove;
1595 i++;
1596 updatingLayersAbove = 0;
1597 continue;
1598 } else {
1599 //both failed.start the loop again from here.
1600 if(firstZReverseIndex >= 0) {
1601 i = firstZReverseIndex;
1602 }
1603 break;
1604 }
1605 }
1606 }
1607 }
1608 if(batchCount > maxBatchCount) {
1609 maxBatchCount = batchCount;
1610 maxBatchStart = batchStart;
1611 maxBatchEnd = batchEnd;
1612 fbZOrder = fbZ;
1613 }
1614 }
1615 return fbZOrder;
1616 }
1617
markLayersForCaching(hwc_context_t * ctx,hwc_display_contents_1_t * list)1618 bool MDPComp::markLayersForCaching(hwc_context_t* ctx,
1619 hwc_display_contents_1_t* list) {
1620 /* Idea is to keep as many non-updating(cached) layers in FB and
1621 * send rest of them through MDP. This is done in 2 steps.
1622 * 1. Find the maximum contiguous batch of non-updating layers.
1623 * 2. See if we can improve this batch size for caching by adding
1624 * opaque layers around the batch, if they don't have
1625 * any overlapping with the updating layers in between.
1626 * NEVER mark an updating layer for caching.
1627 * But cached ones can be marked for MDP */
1628
1629 int maxBatchStart = -1;
1630 int maxBatchEnd = -1;
1631 int maxBatchCount = 0;
1632 int fbZ = -1;
1633
1634 /* Nothing is cached. No batching needed */
1635 if(mCurrentFrame.fbCount == 0) {
1636 return true;
1637 }
1638
1639 /* No MDP comp layers, try to use other comp modes */
1640 if(mCurrentFrame.mdpCount == 0) {
1641 return false;
1642 }
1643
1644 fbZ = getBatch(list, maxBatchStart, maxBatchEnd, maxBatchCount);
1645
1646 /* reset rest of the layers lying inside ROI for MDP comp */
1647 for(int i = 0; i < mCurrentFrame.layerCount; i++) {
1648 hwc_layer_1_t* layer = &list->hwLayers[i];
1649 if((i < maxBatchStart || i > maxBatchEnd) &&
1650 mCurrentFrame.isFBComposed[i]){
1651 if(!mCurrentFrame.drop[i]){
1652 //If an unsupported layer is being attempted to
1653 //be pulled out we should fail
1654 if(not isSupportedForMDPComp(ctx, layer)) {
1655 return false;
1656 }
1657 mCurrentFrame.isFBComposed[i] = false;
1658 }
1659 }
1660 }
1661
1662 // update the frame data
1663 mCurrentFrame.fbZ = fbZ;
1664 mCurrentFrame.fbCount = maxBatchCount;
1665 mCurrentFrame.mdpCount = mCurrentFrame.layerCount -
1666 mCurrentFrame.fbCount - mCurrentFrame.dropCount;
1667
1668 ALOGD_IF(isDebug(),"%s: cached count: %d",__FUNCTION__,
1669 mCurrentFrame.fbCount);
1670
1671 return true;
1672 }
1673
updateLayerCache(hwc_context_t * ctx,hwc_display_contents_1_t * list,FrameInfo & frame)1674 void MDPComp::updateLayerCache(hwc_context_t* ctx,
1675 hwc_display_contents_1_t* list, FrameInfo& frame) {
1676 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
1677 int fbCount = 0;
1678
1679 for(int i = 0; i < numAppLayers; i++) {
1680 hwc_layer_1_t * layer = &list->hwLayers[i];
1681 if (!layerUpdating(layer)) {
1682 if(!frame.drop[i])
1683 fbCount++;
1684 frame.isFBComposed[i] = true;
1685 } else {
1686 frame.isFBComposed[i] = false;
1687 }
1688 }
1689
1690 frame.fbCount = fbCount;
1691 frame.mdpCount = frame.layerCount - frame.fbCount
1692 - frame.dropCount;
1693
1694 ALOGD_IF(isDebug(),"%s: MDP count: %d FB count %d drop count: %d",
1695 __FUNCTION__, frame.mdpCount, frame.fbCount, frame.dropCount);
1696 }
1697
1698 // drop other non-AIV layers from external display list.
dropNonAIVLayers(hwc_context_t * ctx,hwc_display_contents_1_t * list)1699 void MDPComp::dropNonAIVLayers(hwc_context_t* ctx,
1700 hwc_display_contents_1_t* list) {
1701 for (size_t i = 0; i < (size_t)ctx->listStats[mDpy].numAppLayers; i++) {
1702 hwc_layer_1_t * layer = &list->hwLayers[i];
1703 if(!(isAIVVideoLayer(layer) || isAIVCCLayer(layer))) {
1704 mCurrentFrame.dropCount++;
1705 mCurrentFrame.drop[i] = true;
1706 }
1707 }
1708 mCurrentFrame.fbCount -= mCurrentFrame.dropCount;
1709 mCurrentFrame.mdpCount = mCurrentFrame.layerCount -
1710 mCurrentFrame.fbCount - mCurrentFrame.dropCount;
1711 ALOGD_IF(isDebug(),"%s: fb count: %d mdp count %d drop count %d",
1712 __FUNCTION__, mCurrentFrame.fbCount, mCurrentFrame.mdpCount,
1713 mCurrentFrame.dropCount);
1714 }
1715
updateYUV(hwc_context_t * ctx,hwc_display_contents_1_t * list,bool secureOnly,FrameInfo & frame)1716 void MDPComp::updateYUV(hwc_context_t* ctx, hwc_display_contents_1_t* list,
1717 bool secureOnly, FrameInfo& frame) {
1718 int nYuvCount = ctx->listStats[mDpy].yuvCount;
1719 for(int index = 0;index < nYuvCount; index++){
1720 int nYuvIndex = ctx->listStats[mDpy].yuvIndices[index];
1721 hwc_layer_1_t* layer = &list->hwLayers[nYuvIndex];
1722
1723 if(mCurrentFrame.drop[nYuvIndex]) {
1724 continue;
1725 }
1726
1727 if(!isYUVDoable(ctx, layer)) {
1728 if(!frame.isFBComposed[nYuvIndex]) {
1729 frame.isFBComposed[nYuvIndex] = true;
1730 frame.fbCount++;
1731 }
1732 } else {
1733 if(frame.isFBComposed[nYuvIndex]) {
1734 private_handle_t *hnd = (private_handle_t *)layer->handle;
1735 if(!secureOnly || isSecureBuffer(hnd)) {
1736 frame.isFBComposed[nYuvIndex] = false;
1737 frame.fbCount--;
1738 }
1739 }
1740 }
1741 }
1742
1743 frame.mdpCount = frame.layerCount - frame.fbCount - frame.dropCount;
1744 ALOGD_IF(isDebug(),"%s: fb count: %d",__FUNCTION__, frame.fbCount);
1745 }
1746
updateSecureRGB(hwc_context_t * ctx,hwc_display_contents_1_t * list)1747 void MDPComp::updateSecureRGB(hwc_context_t* ctx,
1748 hwc_display_contents_1_t* list) {
1749 int nSecureRGBCount = ctx->listStats[mDpy].secureRGBCount;
1750 for(int index = 0;index < nSecureRGBCount; index++){
1751 int nSecureRGBIndex = ctx->listStats[mDpy].secureRGBIndices[index];
1752 hwc_layer_1_t* layer = &list->hwLayers[nSecureRGBIndex];
1753
1754 if(!isSecureRGBDoable(ctx, layer)) {
1755 if(!mCurrentFrame.isFBComposed[nSecureRGBIndex]) {
1756 mCurrentFrame.isFBComposed[nSecureRGBIndex] = true;
1757 mCurrentFrame.fbCount++;
1758 }
1759 } else {
1760 if(mCurrentFrame.isFBComposed[nSecureRGBIndex]) {
1761 mCurrentFrame.isFBComposed[nSecureRGBIndex] = false;
1762 mCurrentFrame.fbCount--;
1763 }
1764 }
1765 }
1766
1767 mCurrentFrame.mdpCount = mCurrentFrame.layerCount -
1768 mCurrentFrame.fbCount - mCurrentFrame.dropCount;
1769 ALOGD_IF(isDebug(),"%s: fb count: %d",__FUNCTION__,
1770 mCurrentFrame.fbCount);
1771 }
1772
getUpdatingFBRect(hwc_context_t * ctx,hwc_display_contents_1_t * list)1773 hwc_rect_t MDPComp::getUpdatingFBRect(hwc_context_t *ctx,
1774 hwc_display_contents_1_t* list){
1775 hwc_rect_t fbRect = (struct hwc_rect){0, 0, 0, 0};
1776
1777 /* Update only the region of FB needed for composition */
1778 for(int i = 0; i < mCurrentFrame.layerCount; i++ ) {
1779 if(mCurrentFrame.isFBComposed[i] && !mCurrentFrame.drop[i]) {
1780 hwc_layer_1_t* layer = &list->hwLayers[i];
1781 hwc_rect_t dst = layer->displayFrame;
1782 fbRect = getUnion(fbRect, dst);
1783 }
1784 }
1785 trimAgainstROI(ctx, fbRect, fbRect);
1786 return fbRect;
1787 }
1788
postHeuristicsHandling(hwc_context_t * ctx,hwc_display_contents_1_t * list)1789 bool MDPComp::postHeuristicsHandling(hwc_context_t *ctx,
1790 hwc_display_contents_1_t* list) {
1791
1792 //Capability checks
1793 if(!resourceCheck(ctx, list)) {
1794 ALOGD_IF(isDebug(), "%s: resource check failed", __FUNCTION__);
1795 return false;
1796 }
1797
1798 //Limitations checks
1799 if(!hwLimitationsCheck(ctx, list)) {
1800 ALOGD_IF(isDebug(), "%s: HW limitations",__FUNCTION__);
1801 return false;
1802 }
1803
1804 //Configure framebuffer first if applicable
1805 if(mCurrentFrame.fbZ >= 0) {
1806 hwc_rect_t fbRect = getUpdatingFBRect(ctx, list);
1807 if(!ctx->mFBUpdate[mDpy]->prepare(ctx, list, fbRect, mCurrentFrame.fbZ))
1808 {
1809 ALOGD_IF(isDebug(), "%s configure framebuffer failed",
1810 __FUNCTION__);
1811 return false;
1812 }
1813 }
1814
1815 mCurrentFrame.map();
1816
1817 if(!allocLayerPipes(ctx, list)) {
1818 ALOGD_IF(isDebug(), "%s: Unable to allocate MDP pipes", __FUNCTION__);
1819 return false;
1820 }
1821
1822 for (int index = 0, mdpNextZOrder = 0; index < mCurrentFrame.layerCount;
1823 index++) {
1824 if(!mCurrentFrame.isFBComposed[index]) {
1825 int mdpIndex = mCurrentFrame.layerToMDP[index];
1826 hwc_layer_1_t* layer = &list->hwLayers[index];
1827
1828 //Leave fbZ for framebuffer. CACHE/GLES layers go here.
1829 if(mdpNextZOrder == mCurrentFrame.fbZ) {
1830 mdpNextZOrder++;
1831 }
1832 MdpPipeInfo* cur_pipe = mCurrentFrame.mdpToLayer[mdpIndex].pipeInfo;
1833 cur_pipe->zOrder = mdpNextZOrder++;
1834
1835 private_handle_t *hnd = (private_handle_t *)layer->handle;
1836 if(isYUVSplitNeeded(hnd) && sEnableYUVsplit){
1837 if(configure4k2kYuv(ctx, layer,
1838 mCurrentFrame.mdpToLayer[mdpIndex])
1839 != 0 ){
1840 ALOGD_IF(isDebug(), "%s: Failed to configure split pipes \
1841 for layer %d",__FUNCTION__, index);
1842 return false;
1843 }
1844 else{
1845 mdpNextZOrder++;
1846 }
1847 continue;
1848 }
1849 if(configure(ctx, layer, mCurrentFrame.mdpToLayer[mdpIndex]) != 0 ){
1850 ALOGD_IF(isDebug(), "%s: Failed to configure overlay for \
1851 layer %d",__FUNCTION__, index);
1852 return false;
1853 }
1854 }
1855 }
1856
1857 if(!ctx->mOverlay->validateAndSet(mDpy, ctx->dpyAttr[mDpy].fd)) {
1858 ALOGD_IF(isDebug(), "%s: Failed to validate and set overlay for dpy %d"
1859 ,__FUNCTION__, mDpy);
1860 return false;
1861 }
1862
1863 setRedraw(ctx, list);
1864 return true;
1865 }
1866
resourceCheck(hwc_context_t * ctx,hwc_display_contents_1_t * list)1867 bool MDPComp::resourceCheck(hwc_context_t* ctx,
1868 hwc_display_contents_1_t* list) {
1869 const bool fbUsed = mCurrentFrame.fbCount;
1870 if(mCurrentFrame.mdpCount > sMaxPipesPerMixer - fbUsed) {
1871 ALOGD_IF(isDebug(), "%s: Exceeds MAX_PIPES_PER_MIXER",__FUNCTION__);
1872 return false;
1873 }
1874
1875 //Will benefit cases where a video has non-updating background.
1876 if((mDpy > HWC_DISPLAY_PRIMARY) and
1877 (mCurrentFrame.mdpCount > sMaxSecLayers)) {
1878 ALOGD_IF(isDebug(), "%s: Exceeds max secondary pipes",__FUNCTION__);
1879 return false;
1880 }
1881
1882 // Init rotCount to number of rotate sessions used by other displays
1883 int rotCount = ctx->mRotMgr->getNumActiveSessions();
1884 // Count the number of rotator sessions required for current display
1885 for (int index = 0; index < mCurrentFrame.layerCount; index++) {
1886 if(!mCurrentFrame.isFBComposed[index]) {
1887 hwc_layer_1_t* layer = &list->hwLayers[index];
1888 private_handle_t *hnd = (private_handle_t *)layer->handle;
1889 if(has90Transform(layer) && isRotationDoable(ctx, hnd)) {
1890 rotCount++;
1891 }
1892 }
1893 }
1894 // if number of layers to rotate exceeds max rotator sessions, bail out.
1895 if(rotCount > RotMgr::MAX_ROT_SESS) {
1896 ALOGD_IF(isDebug(), "%s: Exceeds max rotator sessions %d",
1897 __FUNCTION__, mDpy);
1898 return false;
1899 }
1900 return true;
1901 }
1902
hwLimitationsCheck(hwc_context_t * ctx,hwc_display_contents_1_t * list)1903 bool MDPComp::hwLimitationsCheck(hwc_context_t* ctx,
1904 hwc_display_contents_1_t* list) {
1905
1906 //A-family hw limitation:
1907 //If a layer need alpha scaling, MDP can not support.
1908 if(ctx->mMDP.version < qdutils::MDSS_V5) {
1909 for(int i = 0; i < mCurrentFrame.layerCount; ++i) {
1910 if(!mCurrentFrame.isFBComposed[i] &&
1911 isAlphaScaled( &list->hwLayers[i])) {
1912 ALOGD_IF(isDebug(), "%s:frame needs alphaScaling",__FUNCTION__);
1913 return false;
1914 }
1915 }
1916 }
1917
1918 // On 8x26 & 8974 hw, we have a limitation of downscaling+blending.
1919 //If multiple layers requires downscaling and also they are overlapping
1920 //fall back to GPU since MDSS can not handle it.
1921 if(qdutils::MDPVersion::getInstance().is8x74v2() ||
1922 qdutils::MDPVersion::getInstance().is8x26()) {
1923 for(int i = 0; i < mCurrentFrame.layerCount-1; ++i) {
1924 hwc_layer_1_t* botLayer = &list->hwLayers[i];
1925 if(!mCurrentFrame.isFBComposed[i] &&
1926 isDownscaleRequired(botLayer)) {
1927 //if layer-i is marked for MDP and needs downscaling
1928 //check if any MDP layer on top of i & overlaps with layer-i
1929 for(int j = i+1; j < mCurrentFrame.layerCount; ++j) {
1930 hwc_layer_1_t* topLayer = &list->hwLayers[j];
1931 if(!mCurrentFrame.isFBComposed[j] &&
1932 isDownscaleRequired(topLayer)) {
1933 hwc_rect_t r = getIntersection(botLayer->displayFrame,
1934 topLayer->displayFrame);
1935 if(isValidRect(r))
1936 return false;
1937 }
1938 }
1939 }
1940 }
1941 }
1942 return true;
1943 }
1944
1945 // Checks only if videos or single layer(RGB) is updating
1946 // which is used for setting dynamic fps or perf hint for single
1947 // layer video playback
onlyVideosUpdating(hwc_context_t * ctx,hwc_display_contents_1_t * list)1948 bool MDPComp::onlyVideosUpdating(hwc_context_t *ctx,
1949 hwc_display_contents_1_t* list) {
1950 bool support = false;
1951 FrameInfo frame;
1952 frame.reset(mCurrentFrame.layerCount);
1953 memset(&frame.drop, 0, sizeof(frame.drop));
1954 frame.dropCount = 0;
1955 ALOGD_IF(isDebug(), "%s: Update Cache and YUVInfo", __FUNCTION__);
1956 updateLayerCache(ctx, list, frame);
1957 updateYUV(ctx, list, false /*secure only*/, frame);
1958 // There are only updating YUV layers or there is single RGB
1959 // Layer(Youtube)
1960 if((ctx->listStats[mDpy].yuvCount == frame.mdpCount) ||
1961 (frame.layerCount == 1)) {
1962 support = true;
1963 }
1964 return support;
1965 }
1966
setDynRefreshRate(hwc_context_t * ctx,hwc_display_contents_1_t * list)1967 void MDPComp::setDynRefreshRate(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
1968 //For primary display, set the dynamic refreshrate
1969 if(!mDpy && qdutils::MDPVersion::getInstance().isDynFpsSupported() &&
1970 ctx->mUseMetaDataRefreshRate) {
1971 uint32_t refreshRate = ctx->dpyAttr[mDpy].refreshRate;
1972 MDPVersion& mdpHw = MDPVersion::getInstance();
1973 if(sIdleFallBack) {
1974 //Set minimum panel refresh rate during idle timeout
1975 refreshRate = mdpHw.getMinFpsSupported();
1976 } else if(onlyVideosUpdating(ctx, list)) {
1977 //Set the new fresh rate, if there is only one updating YUV layer
1978 //or there is one single RGB layer with this request
1979 refreshRate = ctx->listStats[mDpy].refreshRateRequest;
1980 }
1981 setRefreshRate(ctx, mDpy, refreshRate);
1982 }
1983 }
1984
prepare(hwc_context_t * ctx,hwc_display_contents_1_t * list)1985 int MDPComp::prepare(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
1986 int ret = 0;
1987 char property[PROPERTY_VALUE_MAX];
1988
1989 if(!ctx || !list) {
1990 ALOGE("%s: Invalid context or list",__FUNCTION__);
1991 mCachedFrame.reset();
1992 return -1;
1993 }
1994
1995 const int numLayers = ctx->listStats[mDpy].numAppLayers;
1996 if(mDpy == HWC_DISPLAY_PRIMARY) {
1997 sSimulationFlags = 0;
1998 if(property_get("debug.hwc.simulate", property, NULL) > 0) {
1999 int currentFlags = atoi(property);
2000 if(currentFlags != sSimulationFlags) {
2001 sSimulationFlags = currentFlags;
2002 ALOGI("%s: Simulation Flag read: 0x%x (%d)", __FUNCTION__,
2003 sSimulationFlags, sSimulationFlags);
2004 }
2005 }
2006 }
2007 // reset PTOR
2008 if(!mDpy)
2009 memset(&(ctx->mPtorInfo), 0, sizeof(ctx->mPtorInfo));
2010
2011 //reset old data
2012 mCurrentFrame.reset(numLayers);
2013 memset(&mCurrentFrame.drop, 0, sizeof(mCurrentFrame.drop));
2014 mCurrentFrame.dropCount = 0;
2015
2016 //Do not cache the information for next draw cycle.
2017 if(numLayers > MAX_NUM_APP_LAYERS or (!numLayers)) {
2018 ALOGI_IF(numLayers, "%s: Unsupported layer count for mdp composition: %d",
2019 __FUNCTION__, numLayers);
2020 mCachedFrame.reset();
2021 #ifdef DYNAMIC_FPS
2022 // Reset refresh rate
2023 setRefreshRate(ctx, mDpy, ctx->dpyAttr[mDpy].refreshRate);
2024 #endif
2025 return -1;
2026 }
2027
2028 // Detect the start of animation and fall back to GPU only once to cache
2029 // all the layers in FB and display FB content untill animation completes.
2030 if(ctx->listStats[mDpy].isDisplayAnimating) {
2031 mCurrentFrame.needsRedraw = false;
2032 if(ctx->mAnimationState[mDpy] == ANIMATION_STOPPED) {
2033 mCurrentFrame.needsRedraw = true;
2034 ctx->mAnimationState[mDpy] = ANIMATION_STARTED;
2035 }
2036 setMDPCompLayerFlags(ctx, list);
2037 mCachedFrame.updateCounts(mCurrentFrame);
2038 #ifdef DYNAMIC_FPS
2039 // Reset refresh rate
2040 setRefreshRate(ctx, mDpy, ctx->dpyAttr[mDpy].refreshRate);
2041 #endif
2042 ret = -1;
2043 return ret;
2044 } else {
2045 ctx->mAnimationState[mDpy] = ANIMATION_STOPPED;
2046 }
2047
2048 //Hard conditions, if not met, cannot do MDP comp
2049 if(isFrameDoable(ctx)) {
2050 generateROI(ctx, list);
2051 // if AIV Video mode is enabled, drop all non AIV layers from the
2052 // external display list.
2053 if(ctx->listStats[mDpy].mAIVVideoMode) {
2054 dropNonAIVLayers(ctx, list);
2055 }
2056
2057 // if tryFullFrame fails, try to push all video and secure RGB layers
2058 // to MDP for composition.
2059 mModeOn = tryFullFrame(ctx, list) || tryMDPOnlyLayers(ctx, list) ||
2060 tryVideoOnly(ctx, list);
2061 if(mModeOn) {
2062 setMDPCompLayerFlags(ctx, list);
2063 } else {
2064 resetROI(ctx, mDpy);
2065 reset(ctx);
2066 memset(&mCurrentFrame.drop, 0, sizeof(mCurrentFrame.drop));
2067 mCurrentFrame.dropCount = 0;
2068 ret = -1;
2069 ALOGE_IF(sSimulationFlags && (mDpy == HWC_DISPLAY_PRIMARY),
2070 "MDP Composition Strategies Failed");
2071 }
2072 } else {
2073 if ((ctx->mMDP.version == qdutils::MDP_V3_0_5) && ctx->mCopyBit[mDpy] &&
2074 enablePartialUpdateForMDP3) {
2075 generateROI(ctx, list);
2076 for(int i = 0; i < ctx->listStats[mDpy].numAppLayers; i++) {
2077 ctx->copybitDrop[i] = mCurrentFrame.drop[i];
2078 }
2079 }
2080 ALOGD_IF( isDebug(),"%s: MDP Comp not possible for this frame",
2081 __FUNCTION__);
2082 ret = -1;
2083 }
2084
2085 if(isDebug()) {
2086 ALOGD("GEOMETRY change: %d",
2087 (list->flags & HWC_GEOMETRY_CHANGED));
2088 android::String8 sDump("");
2089 dump(sDump, ctx);
2090 ALOGD("%s",sDump.string());
2091 }
2092
2093 #ifdef DYNAMIC_FPS
2094 setDynRefreshRate(ctx, list);
2095 #endif
2096 setPerfHint(ctx, list);
2097
2098 mCachedFrame.updateCounts(mCurrentFrame);
2099 return ret;
2100 }
2101
allocSplitVGPipesfor4k2k(hwc_context_t * ctx,int index)2102 bool MDPComp::allocSplitVGPipesfor4k2k(hwc_context_t *ctx, int index) {
2103
2104 bool bRet = true;
2105 int mdpIndex = mCurrentFrame.layerToMDP[index];
2106 PipeLayerPair& info = mCurrentFrame.mdpToLayer[mdpIndex];
2107 info.pipeInfo = new MdpYUVPipeInfo;
2108 info.rot = NULL;
2109 MdpYUVPipeInfo& pipe_info = *(MdpYUVPipeInfo*)info.pipeInfo;
2110
2111 pipe_info.lIndex = ovutils::OV_INVALID;
2112 pipe_info.rIndex = ovutils::OV_INVALID;
2113
2114 Overlay::PipeSpecs pipeSpecs;
2115 pipeSpecs.formatClass = Overlay::FORMAT_YUV;
2116 pipeSpecs.needsScaling = true;
2117 pipeSpecs.dpy = mDpy;
2118 pipeSpecs.fb = false;
2119
2120 pipe_info.lIndex = ctx->mOverlay->getPipe(pipeSpecs);
2121 if(pipe_info.lIndex == ovutils::OV_INVALID){
2122 bRet = false;
2123 ALOGD_IF(isDebug(),"%s: allocating first VG pipe failed",
2124 __FUNCTION__);
2125 }
2126 pipe_info.rIndex = ctx->mOverlay->getPipe(pipeSpecs);
2127 if(pipe_info.rIndex == ovutils::OV_INVALID){
2128 bRet = false;
2129 ALOGD_IF(isDebug(),"%s: allocating second VG pipe failed",
2130 __FUNCTION__);
2131 }
2132 return bRet;
2133 }
2134
drawOverlap(hwc_context_t * ctx,hwc_display_contents_1_t * list)2135 int MDPComp::drawOverlap(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
2136 int fd = -1;
2137 if (ctx->mPtorInfo.isActive()) {
2138 fd = ctx->mCopyBit[mDpy]->drawOverlap(ctx, list);
2139 if (fd < 0) {
2140 ALOGD_IF(isDebug(),"%s: failed", __FUNCTION__);
2141 }
2142 }
2143 return fd;
2144 }
2145 //=============MDPCompNonSplit==================================================
2146
adjustForSourceSplit(hwc_context_t * ctx,hwc_display_contents_1_t * list)2147 void MDPCompNonSplit::adjustForSourceSplit(hwc_context_t *ctx,
2148 hwc_display_contents_1_t* list) {
2149 //If 4k2k Yuv layer split is possible, and if
2150 //fbz is above 4k2k layer, increment fb zorder by 1
2151 //as we split 4k2k layer and increment zorder for right half
2152 //of the layer
2153 if(!ctx)
2154 return;
2155 if(mCurrentFrame.fbZ >= 0) {
2156 for (int index = 0, mdpNextZOrder = 0; index < mCurrentFrame.layerCount;
2157 index++) {
2158 if(!mCurrentFrame.isFBComposed[index]) {
2159 if(mdpNextZOrder == mCurrentFrame.fbZ) {
2160 mdpNextZOrder++;
2161 }
2162 mdpNextZOrder++;
2163 hwc_layer_1_t* layer = &list->hwLayers[index];
2164 private_handle_t *hnd = (private_handle_t *)layer->handle;
2165 if(isYUVSplitNeeded(hnd)) {
2166 if(mdpNextZOrder <= mCurrentFrame.fbZ)
2167 mCurrentFrame.fbZ += 1;
2168 mdpNextZOrder++;
2169 //As we split 4kx2k yuv layer and program to 2 VG pipes
2170 //(if available) increase mdpcount by 1.
2171 mCurrentFrame.mdpCount++;
2172 }
2173 }
2174 }
2175 }
2176 }
2177
2178 /*
2179 * Configures pipe(s) for MDP composition
2180 */
configure(hwc_context_t * ctx,hwc_layer_1_t * layer,PipeLayerPair & PipeLayerPair)2181 int MDPCompNonSplit::configure(hwc_context_t *ctx, hwc_layer_1_t *layer,
2182 PipeLayerPair& PipeLayerPair) {
2183 MdpPipeInfoNonSplit& mdp_info =
2184 *(static_cast<MdpPipeInfoNonSplit*>(PipeLayerPair.pipeInfo));
2185 eMdpFlags mdpFlags = ovutils::OV_MDP_FLAGS_NONE;
2186 eZorder zOrder = static_cast<eZorder>(mdp_info.zOrder);
2187 eDest dest = mdp_info.index;
2188
2189 ALOGD_IF(isDebug(),"%s: configuring: layer: %p z_order: %d dest_pipe: %d",
2190 __FUNCTION__, layer, zOrder, dest);
2191
2192 return configureNonSplit(ctx, layer, mDpy, mdpFlags, zOrder, dest,
2193 &PipeLayerPair.rot);
2194 }
2195
allocLayerPipes(hwc_context_t * ctx,hwc_display_contents_1_t * list)2196 bool MDPCompNonSplit::allocLayerPipes(hwc_context_t *ctx,
2197 hwc_display_contents_1_t* list) {
2198 for(int index = 0; index < mCurrentFrame.layerCount; index++) {
2199
2200 if(mCurrentFrame.isFBComposed[index]) continue;
2201
2202 hwc_layer_1_t* layer = &list->hwLayers[index];
2203 private_handle_t *hnd = (private_handle_t *)layer->handle;
2204 if(isYUVSplitNeeded(hnd) && sEnableYUVsplit){
2205 if(allocSplitVGPipesfor4k2k(ctx, index)){
2206 continue;
2207 }
2208 }
2209
2210 int mdpIndex = mCurrentFrame.layerToMDP[index];
2211 PipeLayerPair& info = mCurrentFrame.mdpToLayer[mdpIndex];
2212 info.pipeInfo = new MdpPipeInfoNonSplit;
2213 info.rot = NULL;
2214 MdpPipeInfoNonSplit& pipe_info = *(MdpPipeInfoNonSplit*)info.pipeInfo;
2215
2216 Overlay::PipeSpecs pipeSpecs;
2217 pipeSpecs.formatClass = isYuvBuffer(hnd) ?
2218 Overlay::FORMAT_YUV : Overlay::FORMAT_RGB;
2219 pipeSpecs.needsScaling = qhwc::needsScaling(layer) or
2220 (qdutils::MDPVersion::getInstance().is8x26() and
2221 ctx->dpyAttr[HWC_DISPLAY_PRIMARY].xres > 1024);
2222 pipeSpecs.dpy = mDpy;
2223 pipeSpecs.fb = false;
2224 pipeSpecs.numActiveDisplays = ctx->numActiveDisplays;
2225
2226 pipe_info.index = ctx->mOverlay->getPipe(pipeSpecs);
2227
2228 if(pipe_info.index == ovutils::OV_INVALID) {
2229 ALOGD_IF(isDebug(), "%s: Unable to get pipe", __FUNCTION__);
2230 return false;
2231 }
2232 }
2233 return true;
2234 }
2235
configure4k2kYuv(hwc_context_t * ctx,hwc_layer_1_t * layer,PipeLayerPair & PipeLayerPair)2236 int MDPCompNonSplit::configure4k2kYuv(hwc_context_t *ctx, hwc_layer_1_t *layer,
2237 PipeLayerPair& PipeLayerPair) {
2238 MdpYUVPipeInfo& mdp_info =
2239 *(static_cast<MdpYUVPipeInfo*>(PipeLayerPair.pipeInfo));
2240 eZorder zOrder = static_cast<eZorder>(mdp_info.zOrder);
2241 eMdpFlags mdpFlagsL = ovutils::OV_MDP_FLAGS_NONE;
2242 eDest lDest = mdp_info.lIndex;
2243 eDest rDest = mdp_info.rIndex;
2244
2245 return configureSourceSplit(ctx, layer, mDpy, mdpFlagsL, zOrder,
2246 lDest, rDest, &PipeLayerPair.rot);
2247 }
2248
draw(hwc_context_t * ctx,hwc_display_contents_1_t * list)2249 bool MDPCompNonSplit::draw(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
2250
2251 if(!isEnabled() or !mModeOn) {
2252 ALOGD_IF(isDebug(),"%s: MDP Comp not enabled/configured", __FUNCTION__);
2253 return true;
2254 }
2255
2256 // Set the Handle timeout to true for MDP or MIXED composition.
2257 if(sIdleInvalidator && !sIdleFallBack && mCurrentFrame.mdpCount) {
2258 sHandleTimeout = true;
2259 }
2260
2261 overlay::Overlay& ov = *ctx->mOverlay;
2262 LayerProp *layerProp = ctx->layerProp[mDpy];
2263
2264 int numHwLayers = ctx->listStats[mDpy].numAppLayers;
2265 for(int i = 0; i < numHwLayers && mCurrentFrame.mdpCount; i++ )
2266 {
2267 if(mCurrentFrame.isFBComposed[i]) continue;
2268
2269 hwc_layer_1_t *layer = &list->hwLayers[i];
2270 private_handle_t *hnd = (private_handle_t *)layer->handle;
2271 if(!hnd) {
2272 if (!(layer->flags & HWC_COLOR_FILL)) {
2273 ALOGE("%s handle null", __FUNCTION__);
2274 return false;
2275 }
2276 // No PLAY for Color layer
2277 layerProp[i].mFlags &= ~HWC_MDPCOMP;
2278 continue;
2279 }
2280
2281 int mdpIndex = mCurrentFrame.layerToMDP[i];
2282
2283 if(isYUVSplitNeeded(hnd) && sEnableYUVsplit)
2284 {
2285 MdpYUVPipeInfo& pipe_info =
2286 *(MdpYUVPipeInfo*)mCurrentFrame.mdpToLayer[mdpIndex].pipeInfo;
2287 Rotator *rot = mCurrentFrame.mdpToLayer[mdpIndex].rot;
2288 ovutils::eDest indexL = pipe_info.lIndex;
2289 ovutils::eDest indexR = pipe_info.rIndex;
2290 int fd = hnd->fd;
2291 uint32_t offset = (uint32_t)hnd->offset;
2292 if(rot) {
2293 rot->queueBuffer(fd, offset);
2294 fd = rot->getDstMemId();
2295 offset = rot->getDstOffset();
2296 }
2297 if(indexL != ovutils::OV_INVALID) {
2298 ovutils::eDest destL = (ovutils::eDest)indexL;
2299 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2300 using pipe: %d", __FUNCTION__, layer, hnd, indexL );
2301 if (!ov.queueBuffer(fd, offset, destL)) {
2302 ALOGE("%s: queueBuffer failed for display:%d",
2303 __FUNCTION__, mDpy);
2304 return false;
2305 }
2306 }
2307
2308 if(indexR != ovutils::OV_INVALID) {
2309 ovutils::eDest destR = (ovutils::eDest)indexR;
2310 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2311 using pipe: %d", __FUNCTION__, layer, hnd, indexR );
2312 if (!ov.queueBuffer(fd, offset, destR)) {
2313 ALOGE("%s: queueBuffer failed for display:%d",
2314 __FUNCTION__, mDpy);
2315 return false;
2316 }
2317 }
2318 }
2319 else{
2320 MdpPipeInfoNonSplit& pipe_info =
2321 *(MdpPipeInfoNonSplit*)mCurrentFrame.mdpToLayer[mdpIndex].pipeInfo;
2322 ovutils::eDest dest = pipe_info.index;
2323 if(dest == ovutils::OV_INVALID) {
2324 ALOGE("%s: Invalid pipe index (%d)", __FUNCTION__, dest);
2325 return false;
2326 }
2327
2328 if(!(layerProp[i].mFlags & HWC_MDPCOMP)) {
2329 continue;
2330 }
2331
2332 int fd = hnd->fd;
2333 uint32_t offset = (uint32_t)hnd->offset;
2334 int index = ctx->mPtorInfo.getPTORArrayIndex(i);
2335 if (!mDpy && (index != -1)) {
2336 hnd = ctx->mCopyBit[mDpy]->getCurrentRenderBuffer();
2337 fd = hnd->fd;
2338 offset = 0;
2339 }
2340
2341 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2342 using pipe: %d", __FUNCTION__, layer,
2343 hnd, dest );
2344
2345 Rotator *rot = mCurrentFrame.mdpToLayer[mdpIndex].rot;
2346 if(rot) {
2347 if(!rot->queueBuffer(fd, offset))
2348 return false;
2349 fd = rot->getDstMemId();
2350 offset = rot->getDstOffset();
2351 }
2352
2353 if (!ov.queueBuffer(fd, offset, dest)) {
2354 ALOGE("%s: queueBuffer failed for display:%d ",
2355 __FUNCTION__, mDpy);
2356 return false;
2357 }
2358 }
2359
2360 layerProp[i].mFlags &= ~HWC_MDPCOMP;
2361 }
2362 return true;
2363 }
2364
2365 //=============MDPCompSplit===================================================
2366
adjustForSourceSplit(hwc_context_t * ctx,hwc_display_contents_1_t * list)2367 void MDPCompSplit::adjustForSourceSplit(hwc_context_t *ctx,
2368 hwc_display_contents_1_t* list){
2369 //if 4kx2k yuv layer is totally present in either in left half
2370 //or right half then try splitting the yuv layer to avoid decimation
2371 const int lSplit = getLeftSplit(ctx, mDpy);
2372 if(mCurrentFrame.fbZ >= 0) {
2373 for (int index = 0, mdpNextZOrder = 0; index < mCurrentFrame.layerCount;
2374 index++) {
2375 if(!mCurrentFrame.isFBComposed[index]) {
2376 if(mdpNextZOrder == mCurrentFrame.fbZ) {
2377 mdpNextZOrder++;
2378 }
2379 mdpNextZOrder++;
2380 hwc_layer_1_t* layer = &list->hwLayers[index];
2381 private_handle_t *hnd = (private_handle_t *)layer->handle;
2382 if(isYUVSplitNeeded(hnd)) {
2383 hwc_rect_t dst = layer->displayFrame;
2384 if((dst.left > lSplit) || (dst.right < lSplit)) {
2385 mCurrentFrame.mdpCount += 1;
2386 }
2387 if(mdpNextZOrder <= mCurrentFrame.fbZ)
2388 mCurrentFrame.fbZ += 1;
2389 mdpNextZOrder++;
2390 }
2391 }
2392 }
2393 }
2394 }
2395
acquireMDPPipes(hwc_context_t * ctx,hwc_layer_1_t * layer,MdpPipeInfoSplit & pipe_info)2396 bool MDPCompSplit::acquireMDPPipes(hwc_context_t *ctx, hwc_layer_1_t* layer,
2397 MdpPipeInfoSplit& pipe_info) {
2398
2399 const int lSplit = getLeftSplit(ctx, mDpy);
2400 private_handle_t *hnd = (private_handle_t *)layer->handle;
2401 hwc_rect_t dst = layer->displayFrame;
2402 pipe_info.lIndex = ovutils::OV_INVALID;
2403 pipe_info.rIndex = ovutils::OV_INVALID;
2404
2405 Overlay::PipeSpecs pipeSpecs;
2406 pipeSpecs.formatClass = isYuvBuffer(hnd) ?
2407 Overlay::FORMAT_YUV : Overlay::FORMAT_RGB;
2408 pipeSpecs.needsScaling = qhwc::needsScalingWithSplit(ctx, layer, mDpy);
2409 pipeSpecs.dpy = mDpy;
2410 pipeSpecs.mixer = Overlay::MIXER_LEFT;
2411 pipeSpecs.fb = false;
2412
2413 // Acquire pipe only for the updating half
2414 hwc_rect_t l_roi = ctx->listStats[mDpy].lRoi;
2415 hwc_rect_t r_roi = ctx->listStats[mDpy].rRoi;
2416
2417 if (dst.left < lSplit && isValidRect(getIntersection(dst, l_roi))) {
2418 pipe_info.lIndex = ctx->mOverlay->getPipe(pipeSpecs);
2419 if(pipe_info.lIndex == ovutils::OV_INVALID)
2420 return false;
2421 }
2422
2423 if(dst.right > lSplit && isValidRect(getIntersection(dst, r_roi))) {
2424 pipeSpecs.mixer = Overlay::MIXER_RIGHT;
2425 pipe_info.rIndex = ctx->mOverlay->getPipe(pipeSpecs);
2426 if(pipe_info.rIndex == ovutils::OV_INVALID)
2427 return false;
2428 }
2429
2430 return true;
2431 }
2432
allocLayerPipes(hwc_context_t * ctx,hwc_display_contents_1_t * list)2433 bool MDPCompSplit::allocLayerPipes(hwc_context_t *ctx,
2434 hwc_display_contents_1_t* list) {
2435 for(int index = 0 ; index < mCurrentFrame.layerCount; index++) {
2436
2437 if(mCurrentFrame.isFBComposed[index]) continue;
2438
2439 hwc_layer_1_t* layer = &list->hwLayers[index];
2440 private_handle_t *hnd = (private_handle_t *)layer->handle;
2441 hwc_rect_t dst = layer->displayFrame;
2442 const int lSplit = getLeftSplit(ctx, mDpy);
2443 if(isYUVSplitNeeded(hnd) && sEnableYUVsplit){
2444 if((dst.left > lSplit)||(dst.right < lSplit)){
2445 if(allocSplitVGPipesfor4k2k(ctx, index)){
2446 continue;
2447 }
2448 }
2449 }
2450 int mdpIndex = mCurrentFrame.layerToMDP[index];
2451 PipeLayerPair& info = mCurrentFrame.mdpToLayer[mdpIndex];
2452 info.pipeInfo = new MdpPipeInfoSplit;
2453 info.rot = NULL;
2454 MdpPipeInfoSplit& pipe_info = *(MdpPipeInfoSplit*)info.pipeInfo;
2455
2456 if(!acquireMDPPipes(ctx, layer, pipe_info)) {
2457 ALOGD_IF(isDebug(), "%s: Unable to get pipe for type",
2458 __FUNCTION__);
2459 return false;
2460 }
2461 }
2462 return true;
2463 }
2464
configure4k2kYuv(hwc_context_t * ctx,hwc_layer_1_t * layer,PipeLayerPair & PipeLayerPair)2465 int MDPCompSplit::configure4k2kYuv(hwc_context_t *ctx, hwc_layer_1_t *layer,
2466 PipeLayerPair& PipeLayerPair) {
2467 const int lSplit = getLeftSplit(ctx, mDpy);
2468 hwc_rect_t dst = layer->displayFrame;
2469 if((dst.left > lSplit)||(dst.right < lSplit)){
2470 MdpYUVPipeInfo& mdp_info =
2471 *(static_cast<MdpYUVPipeInfo*>(PipeLayerPair.pipeInfo));
2472 eZorder zOrder = static_cast<eZorder>(mdp_info.zOrder);
2473 eMdpFlags mdpFlagsL = ovutils::OV_MDP_FLAGS_NONE;
2474 eDest lDest = mdp_info.lIndex;
2475 eDest rDest = mdp_info.rIndex;
2476
2477 return configureSourceSplit(ctx, layer, mDpy, mdpFlagsL, zOrder,
2478 lDest, rDest, &PipeLayerPair.rot);
2479 }
2480 else{
2481 return configure(ctx, layer, PipeLayerPair);
2482 }
2483 }
2484
2485 /*
2486 * Configures pipe(s) for MDP composition
2487 */
configure(hwc_context_t * ctx,hwc_layer_1_t * layer,PipeLayerPair & PipeLayerPair)2488 int MDPCompSplit::configure(hwc_context_t *ctx, hwc_layer_1_t *layer,
2489 PipeLayerPair& PipeLayerPair) {
2490 MdpPipeInfoSplit& mdp_info =
2491 *(static_cast<MdpPipeInfoSplit*>(PipeLayerPair.pipeInfo));
2492 eZorder zOrder = static_cast<eZorder>(mdp_info.zOrder);
2493 eMdpFlags mdpFlagsL = ovutils::OV_MDP_FLAGS_NONE;
2494 eDest lDest = mdp_info.lIndex;
2495 eDest rDest = mdp_info.rIndex;
2496
2497 ALOGD_IF(isDebug(),"%s: configuring: layer: %p z_order: %d dest_pipeL: %d"
2498 "dest_pipeR: %d",__FUNCTION__, layer, zOrder, lDest, rDest);
2499
2500 return configureSplit(ctx, layer, mDpy, mdpFlagsL, zOrder, lDest,
2501 rDest, &PipeLayerPair.rot);
2502 }
2503
draw(hwc_context_t * ctx,hwc_display_contents_1_t * list)2504 bool MDPCompSplit::draw(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
2505
2506 if(!isEnabled() or !mModeOn) {
2507 ALOGD_IF(isDebug(),"%s: MDP Comp not enabled/configured", __FUNCTION__);
2508 return true;
2509 }
2510
2511 // Set the Handle timeout to true for MDP or MIXED composition.
2512 if(sIdleInvalidator && !sIdleFallBack && mCurrentFrame.mdpCount) {
2513 sHandleTimeout = true;
2514 }
2515
2516 overlay::Overlay& ov = *ctx->mOverlay;
2517 LayerProp *layerProp = ctx->layerProp[mDpy];
2518
2519 int numHwLayers = ctx->listStats[mDpy].numAppLayers;
2520 for(int i = 0; i < numHwLayers && mCurrentFrame.mdpCount; i++ )
2521 {
2522 if(mCurrentFrame.isFBComposed[i]) continue;
2523
2524 hwc_layer_1_t *layer = &list->hwLayers[i];
2525 private_handle_t *hnd = (private_handle_t *)layer->handle;
2526 if(!hnd) {
2527 ALOGE("%s handle null", __FUNCTION__);
2528 return false;
2529 }
2530
2531 if(!(layerProp[i].mFlags & HWC_MDPCOMP)) {
2532 continue;
2533 }
2534
2535 int mdpIndex = mCurrentFrame.layerToMDP[i];
2536
2537 if(isYUVSplitNeeded(hnd) && sEnableYUVsplit)
2538 {
2539 MdpYUVPipeInfo& pipe_info =
2540 *(MdpYUVPipeInfo*)mCurrentFrame.mdpToLayer[mdpIndex].pipeInfo;
2541 Rotator *rot = mCurrentFrame.mdpToLayer[mdpIndex].rot;
2542 ovutils::eDest indexL = pipe_info.lIndex;
2543 ovutils::eDest indexR = pipe_info.rIndex;
2544 int fd = hnd->fd;
2545 uint32_t offset = (uint32_t)hnd->offset;
2546 if(rot) {
2547 rot->queueBuffer(fd, offset);
2548 fd = rot->getDstMemId();
2549 offset = rot->getDstOffset();
2550 }
2551 if(indexL != ovutils::OV_INVALID) {
2552 ovutils::eDest destL = (ovutils::eDest)indexL;
2553 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2554 using pipe: %d", __FUNCTION__, layer, hnd, indexL );
2555 if (!ov.queueBuffer(fd, offset, destL)) {
2556 ALOGE("%s: queueBuffer failed for display:%d",
2557 __FUNCTION__, mDpy);
2558 return false;
2559 }
2560 }
2561
2562 if(indexR != ovutils::OV_INVALID) {
2563 ovutils::eDest destR = (ovutils::eDest)indexR;
2564 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2565 using pipe: %d", __FUNCTION__, layer, hnd, indexR );
2566 if (!ov.queueBuffer(fd, offset, destR)) {
2567 ALOGE("%s: queueBuffer failed for display:%d",
2568 __FUNCTION__, mDpy);
2569 return false;
2570 }
2571 }
2572 }
2573 else{
2574 MdpPipeInfoSplit& pipe_info =
2575 *(MdpPipeInfoSplit*)mCurrentFrame.mdpToLayer[mdpIndex].pipeInfo;
2576 Rotator *rot = mCurrentFrame.mdpToLayer[mdpIndex].rot;
2577
2578 ovutils::eDest indexL = pipe_info.lIndex;
2579 ovutils::eDest indexR = pipe_info.rIndex;
2580
2581 int fd = hnd->fd;
2582 uint32_t offset = (uint32_t)hnd->offset;
2583 int index = ctx->mPtorInfo.getPTORArrayIndex(i);
2584 if (!mDpy && (index != -1)) {
2585 hnd = ctx->mCopyBit[mDpy]->getCurrentRenderBuffer();
2586 fd = hnd->fd;
2587 offset = 0;
2588 }
2589
2590 if(ctx->mAD->draw(ctx, fd, offset)) {
2591 fd = ctx->mAD->getDstFd();
2592 offset = ctx->mAD->getDstOffset();
2593 }
2594
2595 if(rot) {
2596 rot->queueBuffer(fd, offset);
2597 fd = rot->getDstMemId();
2598 offset = rot->getDstOffset();
2599 }
2600
2601 //************* play left mixer **********
2602 if(indexL != ovutils::OV_INVALID) {
2603 ovutils::eDest destL = (ovutils::eDest)indexL;
2604 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2605 using pipe: %d", __FUNCTION__, layer, hnd, indexL );
2606 if (!ov.queueBuffer(fd, offset, destL)) {
2607 ALOGE("%s: queueBuffer failed for left mixer",
2608 __FUNCTION__);
2609 return false;
2610 }
2611 }
2612
2613 //************* play right mixer **********
2614 if(indexR != ovutils::OV_INVALID) {
2615 ovutils::eDest destR = (ovutils::eDest)indexR;
2616 ALOGD_IF(isDebug(),"%s: MDP Comp: Drawing layer: %p hnd: %p \
2617 using pipe: %d", __FUNCTION__, layer, hnd, indexR );
2618 if (!ov.queueBuffer(fd, offset, destR)) {
2619 ALOGE("%s: queueBuffer failed for right mixer",
2620 __FUNCTION__);
2621 return false;
2622 }
2623 }
2624 }
2625
2626 layerProp[i].mFlags &= ~HWC_MDPCOMP;
2627 }
2628
2629 return true;
2630 }
2631
2632 //================MDPCompSrcSplit==============================================
2633
validateAndApplyROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)2634 bool MDPCompSrcSplit::validateAndApplyROI(hwc_context_t *ctx,
2635 hwc_display_contents_1_t* list) {
2636 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
2637 hwc_rect_t visibleRect = ctx->listStats[mDpy].lRoi;
2638
2639 for(int i = numAppLayers - 1; i >= 0; i--) {
2640 if(!isValidRect(visibleRect)) {
2641 mCurrentFrame.drop[i] = true;
2642 mCurrentFrame.dropCount++;
2643 continue;
2644 }
2645
2646 const hwc_layer_1_t* layer = &list->hwLayers[i];
2647 hwc_rect_t dstRect = layer->displayFrame;
2648 hwc_rect_t res = getIntersection(visibleRect, dstRect);
2649
2650 if(!isValidRect(res)) {
2651 mCurrentFrame.drop[i] = true;
2652 mCurrentFrame.dropCount++;
2653 } else {
2654 /* Reset frame ROI when any layer which needs scaling also needs ROI
2655 * cropping */
2656 if(!isSameRect(res, dstRect) && needsScaling (layer)) {
2657 ALOGI("%s: Resetting ROI due to scaling", __FUNCTION__);
2658 memset(&mCurrentFrame.drop, 0, sizeof(mCurrentFrame.drop));
2659 mCurrentFrame.dropCount = 0;
2660 return false;
2661 }
2662
2663 /* deduct any opaque region from visibleRect */
2664 if (layer->blending == HWC_BLENDING_NONE &&
2665 layer->planeAlpha == 0xFF)
2666 visibleRect = deductRect(visibleRect, res);
2667 }
2668 }
2669 return true;
2670 }
2671
2672 /*
2673 * HW Limitation: ping pong split can always split the ping pong output
2674 * equally across two DSI's. So the ROI programmed should be of equal width
2675 * for both the halves
2676 */
generateROI(hwc_context_t * ctx,hwc_display_contents_1_t * list)2677 void MDPCompSrcSplit::generateROI(hwc_context_t *ctx,
2678 hwc_display_contents_1_t* list) {
2679 int numAppLayers = ctx->listStats[mDpy].numAppLayers;
2680
2681 if(!canPartialUpdate(ctx, list))
2682 return;
2683
2684 struct hwc_rect roi = (struct hwc_rect){0, 0, 0, 0};
2685 hwc_rect fullFrame = (struct hwc_rect) {0, 0,(int)ctx->dpyAttr[mDpy].xres,
2686 (int)ctx->dpyAttr[mDpy].yres};
2687
2688 for(int index = 0; index < numAppLayers; index++ ) {
2689 hwc_layer_1_t* layer = &list->hwLayers[index];
2690
2691 // If we have a RGB layer which needs rotation, no partial update
2692 if(!isYuvBuffer((private_handle_t *)layer->handle) && layer->transform)
2693 return;
2694
2695 if (layerUpdating(layer) ||
2696 isYuvBuffer((private_handle_t *)layer->handle)) {
2697 hwc_rect_t dirtyRect = getIntersection(layer->displayFrame,
2698 fullFrame);
2699 if (!needsScaling(layer) && !layer->transform) {
2700 dirtyRect = calculateDirtyRect(layer, fullFrame);
2701 }
2702 roi = getUnion(roi, dirtyRect);
2703 }
2704 }
2705
2706 /* No layer is updating. Still SF wants a refresh.*/
2707 if(!isValidRect(roi))
2708 return;
2709
2710 if (isDisplaySplit(ctx, mDpy)) {
2711 hwc_rect lFrame = fullFrame;
2712 roi = expandROIFromMidPoint(roi, fullFrame);
2713
2714 lFrame.right = fullFrame.right / 2;
2715 hwc_rect lRoi = getIntersection(roi, lFrame);
2716 // Align ROI coordinates to panel restrictions
2717 lRoi = getSanitizeROI(lRoi, lFrame);
2718
2719 hwc_rect rFrame = fullFrame;
2720 rFrame.left = fullFrame.right/2;
2721 hwc_rect rRoi = getIntersection(roi, rFrame);
2722 // Align ROI coordinates to panel restrictions
2723 rRoi = getSanitizeROI(rRoi, rFrame);
2724
2725 roi = getUnion(lRoi, rRoi);
2726
2727 ctx->listStats[mDpy].lRoi = roi;
2728 } else {
2729 hwc_rect lRoi = getIntersection(roi, fullFrame);
2730 // Align ROI coordinates to panel restrictions
2731 lRoi = getSanitizeROI(lRoi, fullFrame);
2732
2733 ctx->listStats[mDpy].lRoi = lRoi;
2734 }
2735
2736 if(!validateAndApplyROI(ctx, list))
2737 resetROI(ctx, mDpy);
2738
2739 ALOGD_IF(isDebug(),"%s: generated ROI: [%d, %d, %d, %d] [%d, %d, %d, %d]",
2740 __FUNCTION__,
2741 ctx->listStats[mDpy].lRoi.left, ctx->listStats[mDpy].lRoi.top,
2742 ctx->listStats[mDpy].lRoi.right, ctx->listStats[mDpy].lRoi.bottom,
2743 ctx->listStats[mDpy].rRoi.left, ctx->listStats[mDpy].rRoi.top,
2744 ctx->listStats[mDpy].rRoi.right, ctx->listStats[mDpy].rRoi.bottom);
2745 }
2746
acquireMDPPipes(hwc_context_t * ctx,hwc_layer_1_t * layer,MdpPipeInfoSplit & pipe_info)2747 bool MDPCompSrcSplit::acquireMDPPipes(hwc_context_t *ctx, hwc_layer_1_t* layer,
2748 MdpPipeInfoSplit& pipe_info) {
2749 private_handle_t *hnd = (private_handle_t *)layer->handle;
2750 hwc_rect_t dst = layer->displayFrame;
2751 hwc_rect_t crop = integerizeSourceCrop(layer->sourceCropf);
2752 pipe_info.lIndex = ovutils::OV_INVALID;
2753 pipe_info.rIndex = ovutils::OV_INVALID;
2754
2755 if(qdutils::MDPVersion::getInstance().isPartialUpdateEnabled() && !mDpy)
2756 trimAgainstROI(ctx,crop, dst);
2757
2758 //If 2 pipes are staged on a single stage of a mixer, then the left pipe
2759 //should have a higher priority than the right one. Pipe priorities are
2760 //starting with VG0, VG1 ... , RGB0 ..., DMA1
2761
2762 Overlay::PipeSpecs pipeSpecs;
2763 pipeSpecs.formatClass = isYuvBuffer(hnd) ?
2764 Overlay::FORMAT_YUV : Overlay::FORMAT_RGB;
2765 pipeSpecs.needsScaling = qhwc::needsScaling(layer);
2766 pipeSpecs.dpy = mDpy;
2767 pipeSpecs.fb = false;
2768
2769 //1 pipe by default for a layer
2770 pipe_info.lIndex = ctx->mOverlay->getPipe(pipeSpecs);
2771 if(pipe_info.lIndex == ovutils::OV_INVALID) {
2772 return false;
2773 }
2774
2775 /* Use 2 pipes IF
2776 a) Layer's crop width is > 2048 or
2777 b) Layer's dest width > 2048 or
2778 c) On primary, driver has indicated with caps to split always. This is
2779 based on an empirically derived value of panel height. Applied only
2780 if the layer's width is > mixer's width
2781 */
2782
2783 MDPVersion& mdpHw = MDPVersion::getInstance();
2784 bool primarySplitAlways = (mDpy == HWC_DISPLAY_PRIMARY) and
2785 mdpHw.isSrcSplitAlways();
2786 const uint32_t lSplit = getLeftSplit(ctx, mDpy);
2787 const uint32_t dstWidth = dst.right - dst.left;
2788 const uint32_t dstHeight = dst.bottom - dst.top;
2789 uint32_t cropWidth = has90Transform(layer) ? crop.bottom - crop.top :
2790 crop.right - crop.left;
2791 uint32_t cropHeight = has90Transform(layer) ? crop.right - crop.left :
2792 crop.bottom - crop.top;
2793 //Approximation to actual clock, ignoring the common factors in pipe and
2794 //mixer cases like line_time
2795 const uint32_t layerClock = getLayerClock(dstWidth, dstHeight, cropHeight);
2796 const uint32_t mixerClock = lSplit;
2797
2798 const uint32_t downscale = getRotDownscale(ctx, layer);
2799 if(downscale) {
2800 cropWidth /= downscale;
2801 cropHeight /= downscale;
2802 }
2803
2804 if(dstWidth > mdpHw.getMaxPipeWidth() or
2805 cropWidth > mdpHw.getMaxPipeWidth() or
2806 (primarySplitAlways and
2807 (cropWidth > lSplit or layerClock > mixerClock))) {
2808 pipe_info.rIndex = ctx->mOverlay->getPipe(pipeSpecs);
2809 if(pipe_info.rIndex == ovutils::OV_INVALID) {
2810 return false;
2811 }
2812
2813 if(ctx->mOverlay->needsPrioritySwap(pipe_info.lIndex,
2814 pipe_info.rIndex)) {
2815 qhwc::swap(pipe_info.lIndex, pipe_info.rIndex);
2816 }
2817 }
2818
2819 return true;
2820 }
2821
configure(hwc_context_t * ctx,hwc_layer_1_t * layer,PipeLayerPair & PipeLayerPair)2822 int MDPCompSrcSplit::configure(hwc_context_t *ctx, hwc_layer_1_t *layer,
2823 PipeLayerPair& PipeLayerPair) {
2824 private_handle_t *hnd = (private_handle_t *)layer->handle;
2825 if(!hnd) {
2826 ALOGE("%s: layer handle is NULL", __FUNCTION__);
2827 return -1;
2828 }
2829 MetaData_t *metadata = (MetaData_t *)hnd->base_metadata;
2830 MdpPipeInfoSplit& mdp_info =
2831 *(static_cast<MdpPipeInfoSplit*>(PipeLayerPair.pipeInfo));
2832 Rotator **rot = &PipeLayerPair.rot;
2833 eZorder z = static_cast<eZorder>(mdp_info.zOrder);
2834 eDest lDest = mdp_info.lIndex;
2835 eDest rDest = mdp_info.rIndex;
2836 hwc_rect_t crop = integerizeSourceCrop(layer->sourceCropf);
2837 hwc_rect_t dst = layer->displayFrame;
2838 int transform = layer->transform;
2839 eTransform orient = static_cast<eTransform>(transform);
2840 int rotFlags = ROT_FLAGS_NONE;
2841 uint32_t format = ovutils::getMdpFormat(hnd->format, hnd->flags);
2842 Whf whf(getWidth(hnd), getHeight(hnd), format, hnd->size);
2843
2844 ALOGD_IF(isDebug(),"%s: configuring: layer: %p z_order: %d dest_pipeL: %d"
2845 "dest_pipeR: %d",__FUNCTION__, layer, z, lDest, rDest);
2846
2847 if(qdutils::MDPVersion::getInstance().isPartialUpdateEnabled() && !mDpy) {
2848 /* MDP driver crops layer coordinates against ROI in Non-Split
2849 * and Split MDP comp. But HWC needs to crop them for source split.
2850 * Reason: 1) Source split is efficient only when the final effective
2851 * load is distributed evenly across mixers.
2852 * 2) We have to know the effective width of the layer that
2853 * the ROI needs to find the no. of pipes the layer needs.
2854 */
2855 trimAgainstROI(ctx, crop, dst);
2856 }
2857
2858 // Handle R/B swap
2859 if (layer->flags & HWC_FORMAT_RB_SWAP) {
2860 if (hnd->format == HAL_PIXEL_FORMAT_RGBA_8888)
2861 whf.format = getMdpFormat(HAL_PIXEL_FORMAT_BGRA_8888);
2862 else if (hnd->format == HAL_PIXEL_FORMAT_RGBX_8888)
2863 whf.format = getMdpFormat(HAL_PIXEL_FORMAT_BGRX_8888);
2864 }
2865 // update source crop and destination position of AIV video layer.
2866 if(ctx->listStats[mDpy].mAIVVideoMode && isYuvBuffer(hnd)) {
2867 updateCoordinates(ctx, crop, dst, mDpy);
2868 }
2869 /* Calculate the external display position based on MDP downscale,
2870 ActionSafe, and extorientation features. */
2871 calcExtDisplayPosition(ctx, hnd, mDpy, crop, dst, transform, orient);
2872
2873 int downscale = getRotDownscale(ctx, layer);
2874 eMdpFlags mdpFlags = ovutils::OV_MDP_FLAGS_NONE;
2875 setMdpFlags(ctx, layer, mdpFlags, downscale, transform);
2876
2877 if(lDest != OV_INVALID && rDest != OV_INVALID) {
2878 //Enable overfetch
2879 setMdpFlags(mdpFlags, OV_MDSS_MDP_DUAL_PIPE);
2880 }
2881
2882 if((has90Transform(layer) or downscale) and isRotationDoable(ctx, hnd)) {
2883 (*rot) = ctx->mRotMgr->getNext();
2884 if((*rot) == NULL) return -1;
2885 ctx->mLayerRotMap[mDpy]->add(layer, *rot);
2886 //If the video is using a single pipe, enable BWC
2887 if(rDest == OV_INVALID) {
2888 BwcPM::setBwc(ctx, mDpy, hnd, crop, dst, transform, downscale,
2889 mdpFlags);
2890 }
2891 //Configure rotator for pre-rotation
2892 if(configRotator(*rot, whf, crop, mdpFlags, orient, downscale) < 0) {
2893 ALOGE("%s: configRotator failed!", __FUNCTION__);
2894 return -1;
2895 }
2896 updateSource(orient, whf, crop, *rot);
2897 rotFlags |= ovutils::ROT_PREROTATED;
2898 }
2899
2900 //If 2 pipes being used, divide layer into half, crop and dst
2901 hwc_rect_t cropL = crop;
2902 hwc_rect_t cropR = crop;
2903 hwc_rect_t dstL = dst;
2904 hwc_rect_t dstR = dst;
2905 if(lDest != OV_INVALID && rDest != OV_INVALID) {
2906 cropL.right = (crop.right + crop.left) / 2;
2907 cropR.left = cropL.right;
2908 sanitizeSourceCrop(cropL, cropR, hnd);
2909
2910 bool cropSwap = false;
2911 //Swap crops on H flip since 2 pipes are being used
2912 if((orient & OVERLAY_TRANSFORM_FLIP_H) && (*rot) == NULL) {
2913 hwc_rect_t tmp = cropL;
2914 cropL = cropR;
2915 cropR = tmp;
2916 cropSwap = true;
2917 }
2918
2919 //cropSwap trick: If the src and dst widths are both odd, let us say
2920 //2507, then splitting both into half would cause left width to be 1253
2921 //and right 1254. If crop is swapped because of H flip, this will cause
2922 //left crop width to be 1254, whereas left dst width remains 1253, thus
2923 //inducing a scaling that is unaccounted for. To overcome that we add 1
2924 //to the dst width if there is a cropSwap. So if the original width was
2925 //2507, the left dst width will be 1254. Even if the original width was
2926 //even for ex: 2508, the left dst width will still remain 1254.
2927 dstL.right = (dst.right + dst.left + cropSwap) / 2;
2928 dstR.left = dstL.right;
2929 }
2930
2931 //For the mdp, since either we are pre-rotating or MDP does flips
2932 orient = OVERLAY_TRANSFORM_0;
2933 transform = 0;
2934
2935 //configure left pipe
2936 if(lDest != OV_INVALID) {
2937 PipeArgs pargL(mdpFlags, whf, z,
2938 static_cast<eRotFlags>(rotFlags), layer->planeAlpha,
2939 (ovutils::eBlending) getBlending(layer->blending));
2940
2941 if(configMdp(ctx->mOverlay, pargL, orient,
2942 cropL, dstL, metadata, lDest) < 0) {
2943 ALOGE("%s: commit failed for left mixer config", __FUNCTION__);
2944 return -1;
2945 }
2946 }
2947
2948 //configure right pipe
2949 if(rDest != OV_INVALID) {
2950 PipeArgs pargR(mdpFlags, whf, z,
2951 static_cast<eRotFlags>(rotFlags),
2952 layer->planeAlpha,
2953 (ovutils::eBlending) getBlending(layer->blending));
2954 if(configMdp(ctx->mOverlay, pargR, orient,
2955 cropR, dstR, metadata, rDest) < 0) {
2956 ALOGE("%s: commit failed for right mixer config", __FUNCTION__);
2957 return -1;
2958 }
2959 }
2960
2961 return 0;
2962 }
2963
getPartialUpdatePref(hwc_context_t * ctx)2964 bool MDPComp::getPartialUpdatePref(hwc_context_t *ctx) {
2965 Locker::Autolock _l(ctx->mDrawLock);
2966 const int fbNum = Overlay::getFbForDpy(Overlay::DPY_PRIMARY);
2967 char path[MAX_SYSFS_FILE_PATH];
2968 snprintf (path, sizeof(path), "sys/class/graphics/fb%d/dyn_pu", fbNum);
2969 int fd = open(path, O_RDONLY);
2970 if(fd < 0) {
2971 ALOGE("%s: Failed to open sysfd node: %s", __FUNCTION__, path);
2972 return -1;
2973 }
2974 char value[4];
2975 ssize_t size_read = read(fd, value, sizeof(value)-1);
2976 if(size_read <= 0) {
2977 ALOGE("%s: Failed to read sysfd node: %s", __FUNCTION__, path);
2978 close(fd);
2979 return -1;
2980 }
2981 close(fd);
2982 value[size_read] = '\0';
2983 return atoi(value);
2984 }
2985
setPartialUpdatePref(hwc_context_t * ctx,bool enable)2986 int MDPComp::setPartialUpdatePref(hwc_context_t *ctx, bool enable) {
2987 Locker::Autolock _l(ctx->mDrawLock);
2988 const int fbNum = Overlay::getFbForDpy(Overlay::DPY_PRIMARY);
2989 char path[MAX_SYSFS_FILE_PATH];
2990 snprintf (path, sizeof(path), "sys/class/graphics/fb%d/dyn_pu", fbNum);
2991 int fd = open(path, O_WRONLY);
2992 if(fd < 0) {
2993 ALOGE("%s: Failed to open sysfd node: %s", __FUNCTION__, path);
2994 return -1;
2995 }
2996 char value[4];
2997 snprintf(value, sizeof(value), "%d", (int)enable);
2998 ssize_t ret = write(fd, value, strlen(value));
2999 if(ret <= 0) {
3000 ALOGE("%s: Failed to write to sysfd nodes: %s", __FUNCTION__, path);
3001 close(fd);
3002 return -1;
3003 }
3004 close(fd);
3005 sIsPartialUpdateActive = enable;
3006 return 0;
3007 }
3008
loadPerfLib()3009 bool MDPComp::loadPerfLib() {
3010 char perfLibPath[PROPERTY_VALUE_MAX] = {0};
3011 bool success = false;
3012 if((property_get("ro.vendor.extension_library", perfLibPath, NULL) <= 0)) {
3013 ALOGE("vendor library not set in ro.vendor.extension_library");
3014 return false;
3015 }
3016
3017 sLibPerfHint = dlopen(perfLibPath, RTLD_NOW);
3018 if(sLibPerfHint) {
3019 *(void **)&sPerfLockAcquire = dlsym(sLibPerfHint, "perf_lock_acq");
3020 *(void **)&sPerfLockRelease = dlsym(sLibPerfHint, "perf_lock_rel");
3021 if (!sPerfLockAcquire || !sPerfLockRelease) {
3022 ALOGE("Failed to load symbols for perfLock");
3023 dlclose(sLibPerfHint);
3024 sLibPerfHint = NULL;
3025 return false;
3026 }
3027 success = true;
3028 ALOGI("Successfully Loaded perf hint API's");
3029 } else {
3030 ALOGE("Failed to open %s : %s", perfLibPath, dlerror());
3031 }
3032 return success;
3033 }
3034
setPerfHint(hwc_context_t * ctx,hwc_display_contents_1_t * list)3035 void MDPComp::setPerfHint(hwc_context_t *ctx, hwc_display_contents_1_t* list) {
3036 if ((sPerfHintWindow < 0) || mDpy || !sLibPerfHint) {
3037 return;
3038 }
3039 static int count = sPerfHintWindow;
3040 static int perflockFlag = 0;
3041
3042 /* Send hint to mpctl when single layer is updated
3043 * for a successful number of windows. Hint release
3044 * happens immediately upon multiple layer update.
3045 */
3046 if (onlyVideosUpdating(ctx, list)) {
3047 if(count) {
3048 count--;
3049 }
3050 } else {
3051 if (perflockFlag) {
3052 perflockFlag = 0;
3053 sPerfLockRelease(sPerfLockHandle);
3054 }
3055 count = sPerfHintWindow;
3056 }
3057 if (count == 0 && !perflockFlag) {
3058 int perfHint = 0x4501; // 45-display layer hint, 01-Enable
3059 sPerfLockHandle = sPerfLockAcquire(0 /*handle*/, 0/*duration*/,
3060 &perfHint, sizeof(perfHint)/sizeof(int));
3061 if(sPerfLockHandle > 0) {
3062 perflockFlag = 1;
3063 }
3064 }
3065 }
3066
3067 }; //namespace
3068
3069