1 /*
2 * Copyright 2018 Advanced Micro Devices, Inc.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
21 *
22 * Authors: AMD
23 *
24 */
25
26 #include "../display_mode_lib.h"
27 #include "display_mode_vba_20.h"
28 #include "../dml_inline_defs.h"
29
30 /*
31 * NOTE:
32 * This file is gcc-parseable HW gospel, coming straight from HW engineers.
33 *
34 * It doesn't adhere to Linux kernel style and sometimes will do things in odd
35 * ways. Unless there is something clearly wrong with it the code should
36 * remain as-is as it provides us with a guarantee from HW that it is correct.
37 */
38
39 #define BPP_INVALID 0
40 #define BPP_BLENDED_PIPE 0xffffffff
41 #define DCN20_MAX_420_IMAGE_WIDTH 4096
42
43 static double adjust_ReturnBW(
44 struct display_mode_lib *mode_lib,
45 double ReturnBW,
46 bool DCCEnabledAnyPlane,
47 double ReturnBandwidthToDCN);
48 static unsigned int dscceComputeDelay(
49 unsigned int bpc,
50 double bpp,
51 unsigned int sliceWidth,
52 unsigned int numSlices,
53 enum output_format_class pixelFormat);
54 static unsigned int dscComputeDelay(enum output_format_class pixelFormat);
55 // Super monster function with some 45 argument
56 static bool CalculatePrefetchSchedule(
57 struct display_mode_lib *mode_lib,
58 double DPPCLK,
59 double DISPCLK,
60 double PixelClock,
61 double DCFCLKDeepSleep,
62 unsigned int DSCDelay,
63 unsigned int DPPPerPlane,
64 bool ScalerEnabled,
65 unsigned int NumberOfCursors,
66 double DPPCLKDelaySubtotal,
67 double DPPCLKDelaySCL,
68 double DPPCLKDelaySCLLBOnly,
69 double DPPCLKDelayCNVCFormater,
70 double DPPCLKDelayCNVCCursor,
71 double DISPCLKDelaySubtotal,
72 unsigned int ScalerRecoutWidth,
73 enum output_format_class OutputFormat,
74 unsigned int VBlank,
75 unsigned int HTotal,
76 unsigned int MaxInterDCNTileRepeaters,
77 unsigned int VStartup,
78 unsigned int PageTableLevels,
79 bool GPUVMEnable,
80 bool DynamicMetadataEnable,
81 unsigned int DynamicMetadataLinesBeforeActiveRequired,
82 unsigned int DynamicMetadataTransmittedBytes,
83 bool DCCEnable,
84 double UrgentLatencyPixelDataOnly,
85 double UrgentExtraLatency,
86 double TCalc,
87 unsigned int PDEAndMetaPTEBytesFrame,
88 unsigned int MetaRowByte,
89 unsigned int PixelPTEBytesPerRow,
90 double PrefetchSourceLinesY,
91 unsigned int SwathWidthY,
92 double BytePerPixelDETY,
93 double VInitPreFillY,
94 unsigned int MaxNumSwathY,
95 double PrefetchSourceLinesC,
96 double BytePerPixelDETC,
97 double VInitPreFillC,
98 unsigned int MaxNumSwathC,
99 unsigned int SwathHeightY,
100 unsigned int SwathHeightC,
101 double TWait,
102 bool XFCEnabled,
103 double XFCRemoteSurfaceFlipDelay,
104 bool InterlaceEnable,
105 bool ProgressiveToInterlaceUnitInOPP,
106 double *DSTXAfterScaler,
107 double *DSTYAfterScaler,
108 double *DestinationLinesForPrefetch,
109 double *PrefetchBandwidth,
110 double *DestinationLinesToRequestVMInVBlank,
111 double *DestinationLinesToRequestRowInVBlank,
112 double *VRatioPrefetchY,
113 double *VRatioPrefetchC,
114 double *RequiredPrefetchPixDataBW,
115 unsigned int *VStartupRequiredWhenNotEnoughTimeForDynamicMetadata,
116 double *Tno_bw,
117 unsigned int *VUpdateOffsetPix,
118 double *VUpdateWidthPix,
119 double *VReadyOffsetPix);
120 static double RoundToDFSGranularityUp(double Clock, double VCOSpeed);
121 static double RoundToDFSGranularityDown(double Clock, double VCOSpeed);
122 static double CalculatePrefetchSourceLines(
123 struct display_mode_lib *mode_lib,
124 double VRatio,
125 double vtaps,
126 bool Interlace,
127 bool ProgressiveToInterlaceUnitInOPP,
128 unsigned int SwathHeight,
129 unsigned int ViewportYStart,
130 double *VInitPreFill,
131 unsigned int *MaxNumSwath);
132 static unsigned int CalculateVMAndRowBytes(
133 struct display_mode_lib *mode_lib,
134 bool DCCEnable,
135 unsigned int BlockHeight256Bytes,
136 unsigned int BlockWidth256Bytes,
137 enum source_format_class SourcePixelFormat,
138 unsigned int SurfaceTiling,
139 unsigned int BytePerPixel,
140 enum scan_direction_class ScanDirection,
141 unsigned int ViewportWidth,
142 unsigned int ViewportHeight,
143 unsigned int SwathWidthY,
144 bool GPUVMEnable,
145 unsigned int VMMPageSize,
146 unsigned int PTEBufferSizeInRequestsLuma,
147 unsigned int PDEProcessingBufIn64KBReqs,
148 unsigned int Pitch,
149 unsigned int DCCMetaPitch,
150 unsigned int *MacroTileWidth,
151 unsigned int *MetaRowByte,
152 unsigned int *PixelPTEBytesPerRow,
153 bool *PTEBufferSizeNotExceeded,
154 unsigned int *dpte_row_height,
155 unsigned int *meta_row_height);
156 static double CalculateTWait(
157 unsigned int PrefetchMode,
158 double DRAMClockChangeLatency,
159 double UrgentLatencyPixelDataOnly,
160 double SREnterPlusExitTime);
161 static double CalculateRemoteSurfaceFlipDelay(
162 struct display_mode_lib *mode_lib,
163 double VRatio,
164 double SwathWidth,
165 double Bpp,
166 double LineTime,
167 double XFCTSlvVupdateOffset,
168 double XFCTSlvVupdateWidth,
169 double XFCTSlvVreadyOffset,
170 double XFCXBUFLatencyTolerance,
171 double XFCFillBWOverhead,
172 double XFCSlvChunkSize,
173 double XFCBusTransportTime,
174 double TCalc,
175 double TWait,
176 double *SrcActiveDrainRate,
177 double *TInitXFill,
178 double *TslvChk);
179 static void CalculateActiveRowBandwidth(
180 bool GPUVMEnable,
181 enum source_format_class SourcePixelFormat,
182 double VRatio,
183 bool DCCEnable,
184 double LineTime,
185 unsigned int MetaRowByteLuma,
186 unsigned int MetaRowByteChroma,
187 unsigned int meta_row_height_luma,
188 unsigned int meta_row_height_chroma,
189 unsigned int PixelPTEBytesPerRowLuma,
190 unsigned int PixelPTEBytesPerRowChroma,
191 unsigned int dpte_row_height_luma,
192 unsigned int dpte_row_height_chroma,
193 double *meta_row_bw,
194 double *dpte_row_bw,
195 double *qual_row_bw);
196 static void CalculateFlipSchedule(
197 struct display_mode_lib *mode_lib,
198 double UrgentExtraLatency,
199 double UrgentLatencyPixelDataOnly,
200 unsigned int GPUVMMaxPageTableLevels,
201 bool GPUVMEnable,
202 double BandwidthAvailableForImmediateFlip,
203 unsigned int TotImmediateFlipBytes,
204 enum source_format_class SourcePixelFormat,
205 unsigned int ImmediateFlipBytes,
206 double LineTime,
207 double VRatio,
208 double Tno_bw,
209 double PDEAndMetaPTEBytesFrame,
210 unsigned int MetaRowByte,
211 unsigned int PixelPTEBytesPerRow,
212 bool DCCEnable,
213 unsigned int dpte_row_height,
214 unsigned int meta_row_height,
215 double qual_row_bw,
216 double *DestinationLinesToRequestVMInImmediateFlip,
217 double *DestinationLinesToRequestRowInImmediateFlip,
218 double *final_flip_bw,
219 bool *ImmediateFlipSupportedForPipe);
220 static double CalculateWriteBackDelay(
221 enum source_format_class WritebackPixelFormat,
222 double WritebackHRatio,
223 double WritebackVRatio,
224 unsigned int WritebackLumaHTaps,
225 unsigned int WritebackLumaVTaps,
226 unsigned int WritebackChromaHTaps,
227 unsigned int WritebackChromaVTaps,
228 unsigned int WritebackDestinationWidth);
229
230 static void dml20_DisplayPipeConfiguration(struct display_mode_lib *mode_lib);
231 static void dml20_DISPCLKDPPCLKDCFCLKDeepSleepPrefetchParametersWatermarksAndPerformanceCalculation(
232 struct display_mode_lib *mode_lib);
233
dml20_recalculate(struct display_mode_lib * mode_lib)234 void dml20_recalculate(struct display_mode_lib *mode_lib)
235 {
236 ModeSupportAndSystemConfiguration(mode_lib);
237 mode_lib->vba.FabricAndDRAMBandwidth = dml_min(
238 mode_lib->vba.DRAMSpeed * mode_lib->vba.NumberOfChannels * mode_lib->vba.DRAMChannelWidth,
239 mode_lib->vba.FabricClock * mode_lib->vba.FabricDatapathToDCNDataReturn) / 1000.0;
240 PixelClockAdjustmentForProgressiveToInterlaceUnit(mode_lib);
241 dml20_DisplayPipeConfiguration(mode_lib);
242 dml20_DISPCLKDPPCLKDCFCLKDeepSleepPrefetchParametersWatermarksAndPerformanceCalculation(mode_lib);
243 }
244
adjust_ReturnBW(struct display_mode_lib * mode_lib,double ReturnBW,bool DCCEnabledAnyPlane,double ReturnBandwidthToDCN)245 static double adjust_ReturnBW(
246 struct display_mode_lib *mode_lib,
247 double ReturnBW,
248 bool DCCEnabledAnyPlane,
249 double ReturnBandwidthToDCN)
250 {
251 double CriticalCompression;
252
253 if (DCCEnabledAnyPlane
254 && ReturnBandwidthToDCN
255 > mode_lib->vba.DCFCLK * mode_lib->vba.ReturnBusWidth / 4.0)
256 ReturnBW =
257 dml_min(
258 ReturnBW,
259 ReturnBandwidthToDCN * 4
260 * (1.0
261 - mode_lib->vba.UrgentLatencyPixelDataOnly
262 / ((mode_lib->vba.ROBBufferSizeInKByte
263 - mode_lib->vba.PixelChunkSizeInKByte)
264 * 1024
265 / ReturnBandwidthToDCN
266 - mode_lib->vba.DCFCLK
267 * mode_lib->vba.ReturnBusWidth
268 / 4)
269 + mode_lib->vba.UrgentLatencyPixelDataOnly));
270
271 CriticalCompression = 2.0 * mode_lib->vba.ReturnBusWidth * mode_lib->vba.DCFCLK
272 * mode_lib->vba.UrgentLatencyPixelDataOnly
273 / (ReturnBandwidthToDCN * mode_lib->vba.UrgentLatencyPixelDataOnly
274 + (mode_lib->vba.ROBBufferSizeInKByte
275 - mode_lib->vba.PixelChunkSizeInKByte)
276 * 1024);
277
278 if (DCCEnabledAnyPlane && CriticalCompression > 1.0 && CriticalCompression < 4.0)
279 ReturnBW =
280 dml_min(
281 ReturnBW,
282 4.0 * ReturnBandwidthToDCN
283 * (mode_lib->vba.ROBBufferSizeInKByte
284 - mode_lib->vba.PixelChunkSizeInKByte)
285 * 1024
286 * mode_lib->vba.ReturnBusWidth
287 * mode_lib->vba.DCFCLK
288 * mode_lib->vba.UrgentLatencyPixelDataOnly
289 / dml_pow(
290 (ReturnBandwidthToDCN
291 * mode_lib->vba.UrgentLatencyPixelDataOnly
292 + (mode_lib->vba.ROBBufferSizeInKByte
293 - mode_lib->vba.PixelChunkSizeInKByte)
294 * 1024),
295 2));
296
297 return ReturnBW;
298 }
299
dscceComputeDelay(unsigned int bpc,double bpp,unsigned int sliceWidth,unsigned int numSlices,enum output_format_class pixelFormat)300 static unsigned int dscceComputeDelay(
301 unsigned int bpc,
302 double bpp,
303 unsigned int sliceWidth,
304 unsigned int numSlices,
305 enum output_format_class pixelFormat)
306 {
307 // valid bpc = source bits per component in the set of {8, 10, 12}
308 // valid bpp = increments of 1/16 of a bit
309 // min = 6/7/8 in N420/N422/444, respectively
310 // max = such that compression is 1:1
311 //valid sliceWidth = number of pixels per slice line, must be less than or equal to 5184/numSlices (or 4096/numSlices in 420 mode)
312 //valid numSlices = number of slices in the horiziontal direction per DSC engine in the set of {1, 2, 3, 4}
313 //valid pixelFormat = pixel/color format in the set of {:N444_RGB, :S422, :N422, :N420}
314
315 // fixed value
316 unsigned int rcModelSize = 8192;
317
318 // N422/N420 operate at 2 pixels per clock
319 unsigned int pixelsPerClock, lstall, D, initalXmitDelay, w, s, ix, wx, p, l0, a, ax, l,
320 Delay, pixels;
321
322 if (pixelFormat == dm_n422 || pixelFormat == dm_420)
323 pixelsPerClock = 2;
324 // #all other modes operate at 1 pixel per clock
325 else
326 pixelsPerClock = 1;
327
328 //initial transmit delay as per PPS
329 initalXmitDelay = dml_round(rcModelSize / 2.0 / bpp / pixelsPerClock);
330
331 //compute ssm delay
332 if (bpc == 8)
333 D = 81;
334 else if (bpc == 10)
335 D = 89;
336 else
337 D = 113;
338
339 //divide by pixel per cycle to compute slice width as seen by DSC
340 w = sliceWidth / pixelsPerClock;
341
342 //422 mode has an additional cycle of delay
343 if (pixelFormat == dm_s422)
344 s = 1;
345 else
346 s = 0;
347
348 //main calculation for the dscce
349 ix = initalXmitDelay + 45;
350 wx = (w + 2) / 3;
351 p = 3 * wx - w;
352 l0 = ix / w;
353 a = ix + p * l0;
354 ax = (a + 2) / 3 + D + 6 + 1;
355 l = (ax + wx - 1) / wx;
356 if ((ix % w) == 0 && p != 0)
357 lstall = 1;
358 else
359 lstall = 0;
360 Delay = l * wx * (numSlices - 1) + ax + s + lstall + 22;
361
362 //dsc processes 3 pixel containers per cycle and a container can contain 1 or 2 pixels
363 pixels = Delay * 3 * pixelsPerClock;
364 return pixels;
365 }
366
dscComputeDelay(enum output_format_class pixelFormat)367 static unsigned int dscComputeDelay(enum output_format_class pixelFormat)
368 {
369 unsigned int Delay = 0;
370
371 if (pixelFormat == dm_420) {
372 // sfr
373 Delay = Delay + 2;
374 // dsccif
375 Delay = Delay + 0;
376 // dscc - input deserializer
377 Delay = Delay + 3;
378 // dscc gets pixels every other cycle
379 Delay = Delay + 2;
380 // dscc - input cdc fifo
381 Delay = Delay + 12;
382 // dscc gets pixels every other cycle
383 Delay = Delay + 13;
384 // dscc - cdc uncertainty
385 Delay = Delay + 2;
386 // dscc - output cdc fifo
387 Delay = Delay + 7;
388 // dscc gets pixels every other cycle
389 Delay = Delay + 3;
390 // dscc - cdc uncertainty
391 Delay = Delay + 2;
392 // dscc - output serializer
393 Delay = Delay + 1;
394 // sft
395 Delay = Delay + 1;
396 } else if (pixelFormat == dm_n422) {
397 // sfr
398 Delay = Delay + 2;
399 // dsccif
400 Delay = Delay + 1;
401 // dscc - input deserializer
402 Delay = Delay + 5;
403 // dscc - input cdc fifo
404 Delay = Delay + 25;
405 // dscc - cdc uncertainty
406 Delay = Delay + 2;
407 // dscc - output cdc fifo
408 Delay = Delay + 10;
409 // dscc - cdc uncertainty
410 Delay = Delay + 2;
411 // dscc - output serializer
412 Delay = Delay + 1;
413 // sft
414 Delay = Delay + 1;
415 } else {
416 // sfr
417 Delay = Delay + 2;
418 // dsccif
419 Delay = Delay + 0;
420 // dscc - input deserializer
421 Delay = Delay + 3;
422 // dscc - input cdc fifo
423 Delay = Delay + 12;
424 // dscc - cdc uncertainty
425 Delay = Delay + 2;
426 // dscc - output cdc fifo
427 Delay = Delay + 7;
428 // dscc - output serializer
429 Delay = Delay + 1;
430 // dscc - cdc uncertainty
431 Delay = Delay + 2;
432 // sft
433 Delay = Delay + 1;
434 }
435
436 return Delay;
437 }
438
CalculatePrefetchSchedule(struct display_mode_lib * mode_lib,double DPPCLK,double DISPCLK,double PixelClock,double DCFCLKDeepSleep,unsigned int DSCDelay,unsigned int DPPPerPlane,bool ScalerEnabled,unsigned int NumberOfCursors,double DPPCLKDelaySubtotal,double DPPCLKDelaySCL,double DPPCLKDelaySCLLBOnly,double DPPCLKDelayCNVCFormater,double DPPCLKDelayCNVCCursor,double DISPCLKDelaySubtotal,unsigned int ScalerRecoutWidth,enum output_format_class OutputFormat,unsigned int VBlank,unsigned int HTotal,unsigned int MaxInterDCNTileRepeaters,unsigned int VStartup,unsigned int PageTableLevels,bool GPUVMEnable,bool DynamicMetadataEnable,unsigned int DynamicMetadataLinesBeforeActiveRequired,unsigned int DynamicMetadataTransmittedBytes,bool DCCEnable,double UrgentLatencyPixelDataOnly,double UrgentExtraLatency,double TCalc,unsigned int PDEAndMetaPTEBytesFrame,unsigned int MetaRowByte,unsigned int PixelPTEBytesPerRow,double PrefetchSourceLinesY,unsigned int SwathWidthY,double BytePerPixelDETY,double VInitPreFillY,unsigned int MaxNumSwathY,double PrefetchSourceLinesC,double BytePerPixelDETC,double VInitPreFillC,unsigned int MaxNumSwathC,unsigned int SwathHeightY,unsigned int SwathHeightC,double TWait,bool XFCEnabled,double XFCRemoteSurfaceFlipDelay,bool InterlaceEnable,bool ProgressiveToInterlaceUnitInOPP,double * DSTXAfterScaler,double * DSTYAfterScaler,double * DestinationLinesForPrefetch,double * PrefetchBandwidth,double * DestinationLinesToRequestVMInVBlank,double * DestinationLinesToRequestRowInVBlank,double * VRatioPrefetchY,double * VRatioPrefetchC,double * RequiredPrefetchPixDataBW,unsigned int * VStartupRequiredWhenNotEnoughTimeForDynamicMetadata,double * Tno_bw,unsigned int * VUpdateOffsetPix,double * VUpdateWidthPix,double * VReadyOffsetPix)439 static bool CalculatePrefetchSchedule(
440 struct display_mode_lib *mode_lib,
441 double DPPCLK,
442 double DISPCLK,
443 double PixelClock,
444 double DCFCLKDeepSleep,
445 unsigned int DSCDelay,
446 unsigned int DPPPerPlane,
447 bool ScalerEnabled,
448 unsigned int NumberOfCursors,
449 double DPPCLKDelaySubtotal,
450 double DPPCLKDelaySCL,
451 double DPPCLKDelaySCLLBOnly,
452 double DPPCLKDelayCNVCFormater,
453 double DPPCLKDelayCNVCCursor,
454 double DISPCLKDelaySubtotal,
455 unsigned int ScalerRecoutWidth,
456 enum output_format_class OutputFormat,
457 unsigned int VBlank,
458 unsigned int HTotal,
459 unsigned int MaxInterDCNTileRepeaters,
460 unsigned int VStartup,
461 unsigned int PageTableLevels,
462 bool GPUVMEnable,
463 bool DynamicMetadataEnable,
464 unsigned int DynamicMetadataLinesBeforeActiveRequired,
465 unsigned int DynamicMetadataTransmittedBytes,
466 bool DCCEnable,
467 double UrgentLatencyPixelDataOnly,
468 double UrgentExtraLatency,
469 double TCalc,
470 unsigned int PDEAndMetaPTEBytesFrame,
471 unsigned int MetaRowByte,
472 unsigned int PixelPTEBytesPerRow,
473 double PrefetchSourceLinesY,
474 unsigned int SwathWidthY,
475 double BytePerPixelDETY,
476 double VInitPreFillY,
477 unsigned int MaxNumSwathY,
478 double PrefetchSourceLinesC,
479 double BytePerPixelDETC,
480 double VInitPreFillC,
481 unsigned int MaxNumSwathC,
482 unsigned int SwathHeightY,
483 unsigned int SwathHeightC,
484 double TWait,
485 bool XFCEnabled,
486 double XFCRemoteSurfaceFlipDelay,
487 bool InterlaceEnable,
488 bool ProgressiveToInterlaceUnitInOPP,
489 double *DSTXAfterScaler,
490 double *DSTYAfterScaler,
491 double *DestinationLinesForPrefetch,
492 double *PrefetchBandwidth,
493 double *DestinationLinesToRequestVMInVBlank,
494 double *DestinationLinesToRequestRowInVBlank,
495 double *VRatioPrefetchY,
496 double *VRatioPrefetchC,
497 double *RequiredPrefetchPixDataBW,
498 unsigned int *VStartupRequiredWhenNotEnoughTimeForDynamicMetadata,
499 double *Tno_bw,
500 unsigned int *VUpdateOffsetPix,
501 double *VUpdateWidthPix,
502 double *VReadyOffsetPix)
503 {
504 bool MyError = false;
505 unsigned int DPPCycles, DISPCLKCycles;
506 double DSTTotalPixelsAfterScaler, TotalRepeaterDelayTime;
507 double Tdm, LineTime, Tsetup;
508 double dst_y_prefetch_equ;
509 double Tsw_oto;
510 double prefetch_bw_oto;
511 double Tvm_oto;
512 double Tr0_oto;
513 double Tpre_oto;
514 double dst_y_prefetch_oto;
515 double TimeForFetchingMetaPTE = 0;
516 double TimeForFetchingRowInVBlank = 0;
517 double LinesToRequestPrefetchPixelData = 0;
518
519 if (ScalerEnabled)
520 DPPCycles = DPPCLKDelaySubtotal + DPPCLKDelaySCL;
521 else
522 DPPCycles = DPPCLKDelaySubtotal + DPPCLKDelaySCLLBOnly;
523
524 DPPCycles = DPPCycles + DPPCLKDelayCNVCFormater + NumberOfCursors * DPPCLKDelayCNVCCursor;
525
526 DISPCLKCycles = DISPCLKDelaySubtotal;
527
528 if (DPPCLK == 0.0 || DISPCLK == 0.0)
529 return true;
530
531 *DSTXAfterScaler = DPPCycles * PixelClock / DPPCLK + DISPCLKCycles * PixelClock / DISPCLK
532 + DSCDelay;
533
534 if (DPPPerPlane > 1)
535 *DSTXAfterScaler = *DSTXAfterScaler + ScalerRecoutWidth;
536
537 if (OutputFormat == dm_420 || (InterlaceEnable && ProgressiveToInterlaceUnitInOPP))
538 *DSTYAfterScaler = 1;
539 else
540 *DSTYAfterScaler = 0;
541
542 DSTTotalPixelsAfterScaler = ((double) (*DSTYAfterScaler * HTotal)) + *DSTXAfterScaler;
543 *DSTYAfterScaler = dml_floor(DSTTotalPixelsAfterScaler / HTotal, 1);
544 *DSTXAfterScaler = DSTTotalPixelsAfterScaler - ((double) (*DSTYAfterScaler * HTotal));
545
546 *VUpdateOffsetPix = dml_ceil(HTotal / 4.0, 1);
547 TotalRepeaterDelayTime = MaxInterDCNTileRepeaters * (2.0 / DPPCLK + 3.0 / DISPCLK);
548 *VUpdateWidthPix = (14.0 / DCFCLKDeepSleep + 12.0 / DPPCLK + TotalRepeaterDelayTime)
549 * PixelClock;
550
551 *VReadyOffsetPix = dml_max(
552 150.0 / DPPCLK,
553 TotalRepeaterDelayTime + 20.0 / DCFCLKDeepSleep + 10.0 / DPPCLK)
554 * PixelClock;
555
556 Tsetup = (double) (*VUpdateOffsetPix + *VUpdateWidthPix + *VReadyOffsetPix) / PixelClock;
557
558 LineTime = (double) HTotal / PixelClock;
559
560 if (DynamicMetadataEnable) {
561 double Tdmbf, Tdmec, Tdmsks;
562
563 Tdm = dml_max(0.0, UrgentExtraLatency - TCalc);
564 Tdmbf = DynamicMetadataTransmittedBytes / 4.0 / DISPCLK;
565 Tdmec = LineTime;
566 if (DynamicMetadataLinesBeforeActiveRequired == 0)
567 Tdmsks = VBlank * LineTime / 2.0;
568 else
569 Tdmsks = DynamicMetadataLinesBeforeActiveRequired * LineTime;
570 if (InterlaceEnable && !ProgressiveToInterlaceUnitInOPP)
571 Tdmsks = Tdmsks / 2;
572 if (VStartup * LineTime
573 < Tsetup + TWait + UrgentExtraLatency + Tdmbf + Tdmec + Tdmsks) {
574 MyError = true;
575 *VStartupRequiredWhenNotEnoughTimeForDynamicMetadata = (Tsetup + TWait
576 + UrgentExtraLatency + Tdmbf + Tdmec + Tdmsks) / LineTime;
577 } else
578 *VStartupRequiredWhenNotEnoughTimeForDynamicMetadata = 0.0;
579 } else
580 Tdm = 0;
581
582 if (GPUVMEnable) {
583 if (PageTableLevels == 4)
584 *Tno_bw = UrgentExtraLatency + UrgentLatencyPixelDataOnly;
585 else if (PageTableLevels == 3)
586 *Tno_bw = UrgentExtraLatency;
587 else
588 *Tno_bw = 0;
589 } else if (DCCEnable)
590 *Tno_bw = LineTime;
591 else
592 *Tno_bw = LineTime / 4;
593
594 dst_y_prefetch_equ = VStartup - dml_max(TCalc + TWait, XFCRemoteSurfaceFlipDelay) / LineTime
595 - (Tsetup + Tdm) / LineTime
596 - (*DSTYAfterScaler + *DSTXAfterScaler / HTotal);
597
598 Tsw_oto = dml_max(PrefetchSourceLinesY, PrefetchSourceLinesC) * LineTime;
599
600 prefetch_bw_oto = (MetaRowByte + PixelPTEBytesPerRow
601 + PrefetchSourceLinesY * SwathWidthY * dml_ceil(BytePerPixelDETY, 1)
602 + PrefetchSourceLinesC * SwathWidthY / 2 * dml_ceil(BytePerPixelDETC, 2))
603 / Tsw_oto;
604
605 if (GPUVMEnable == true) {
606 Tvm_oto =
607 dml_max(
608 *Tno_bw + PDEAndMetaPTEBytesFrame / prefetch_bw_oto,
609 dml_max(
610 UrgentExtraLatency
611 + UrgentLatencyPixelDataOnly
612 * (PageTableLevels
613 - 1),
614 LineTime / 4.0));
615 } else
616 Tvm_oto = LineTime / 4.0;
617
618 if ((GPUVMEnable == true || DCCEnable == true)) {
619 Tr0_oto = dml_max(
620 (MetaRowByte + PixelPTEBytesPerRow) / prefetch_bw_oto,
621 dml_max(UrgentLatencyPixelDataOnly, dml_max(LineTime - Tvm_oto, LineTime / 4)));
622 } else
623 Tr0_oto = LineTime - Tvm_oto;
624
625 Tpre_oto = Tvm_oto + Tr0_oto + Tsw_oto;
626
627 dst_y_prefetch_oto = Tpre_oto / LineTime;
628
629 if (dst_y_prefetch_oto < dst_y_prefetch_equ)
630 *DestinationLinesForPrefetch = dst_y_prefetch_oto;
631 else
632 *DestinationLinesForPrefetch = dst_y_prefetch_equ;
633
634 *DestinationLinesForPrefetch = dml_floor(4.0 * (*DestinationLinesForPrefetch + 0.125), 1)
635 / 4;
636
637 dml_print("DML: VStartup: %d\n", VStartup);
638 dml_print("DML: TCalc: %f\n", TCalc);
639 dml_print("DML: TWait: %f\n", TWait);
640 dml_print("DML: XFCRemoteSurfaceFlipDelay: %f\n", XFCRemoteSurfaceFlipDelay);
641 dml_print("DML: LineTime: %f\n", LineTime);
642 dml_print("DML: Tsetup: %f\n", Tsetup);
643 dml_print("DML: Tdm: %f\n", Tdm);
644 dml_print("DML: DSTYAfterScaler: %f\n", *DSTYAfterScaler);
645 dml_print("DML: DSTXAfterScaler: %f\n", *DSTXAfterScaler);
646 dml_print("DML: HTotal: %d\n", HTotal);
647
648 *PrefetchBandwidth = 0;
649 *DestinationLinesToRequestVMInVBlank = 0;
650 *DestinationLinesToRequestRowInVBlank = 0;
651 *VRatioPrefetchY = 0;
652 *VRatioPrefetchC = 0;
653 *RequiredPrefetchPixDataBW = 0;
654 if (*DestinationLinesForPrefetch > 1) {
655 *PrefetchBandwidth = (PDEAndMetaPTEBytesFrame + 2 * MetaRowByte
656 + 2 * PixelPTEBytesPerRow
657 + PrefetchSourceLinesY * SwathWidthY * dml_ceil(BytePerPixelDETY, 1)
658 + PrefetchSourceLinesC * SwathWidthY / 2
659 * dml_ceil(BytePerPixelDETC, 2))
660 / (*DestinationLinesForPrefetch * LineTime - *Tno_bw);
661 if (GPUVMEnable) {
662 TimeForFetchingMetaPTE =
663 dml_max(
664 *Tno_bw
665 + (double) PDEAndMetaPTEBytesFrame
666 / *PrefetchBandwidth,
667 dml_max(
668 UrgentExtraLatency
669 + UrgentLatencyPixelDataOnly
670 * (PageTableLevels
671 - 1),
672 LineTime / 4));
673 } else {
674 if (NumberOfCursors > 0 || XFCEnabled)
675 TimeForFetchingMetaPTE = LineTime / 4;
676 else
677 TimeForFetchingMetaPTE = 0.0;
678 }
679
680 if ((GPUVMEnable == true || DCCEnable == true)) {
681 TimeForFetchingRowInVBlank =
682 dml_max(
683 (MetaRowByte + PixelPTEBytesPerRow)
684 / *PrefetchBandwidth,
685 dml_max(
686 UrgentLatencyPixelDataOnly,
687 dml_max(
688 LineTime
689 - TimeForFetchingMetaPTE,
690 LineTime
691 / 4.0)));
692 } else {
693 if (NumberOfCursors > 0 || XFCEnabled)
694 TimeForFetchingRowInVBlank = LineTime - TimeForFetchingMetaPTE;
695 else
696 TimeForFetchingRowInVBlank = 0.0;
697 }
698
699 *DestinationLinesToRequestVMInVBlank = dml_floor(
700 4.0 * (TimeForFetchingMetaPTE / LineTime + 0.125),
701 1) / 4.0;
702
703 *DestinationLinesToRequestRowInVBlank = dml_floor(
704 4.0 * (TimeForFetchingRowInVBlank / LineTime + 0.125),
705 1) / 4.0;
706
707 LinesToRequestPrefetchPixelData =
708 *DestinationLinesForPrefetch
709 - ((NumberOfCursors > 0 || GPUVMEnable
710 || DCCEnable) ?
711 (*DestinationLinesToRequestVMInVBlank
712 + *DestinationLinesToRequestRowInVBlank) :
713 0.0);
714
715 if (LinesToRequestPrefetchPixelData > 0) {
716
717 *VRatioPrefetchY = (double) PrefetchSourceLinesY
718 / LinesToRequestPrefetchPixelData;
719 *VRatioPrefetchY = dml_max(*VRatioPrefetchY, 1.0);
720 if ((SwathHeightY > 4) && (VInitPreFillY > 3)) {
721 if (LinesToRequestPrefetchPixelData > (VInitPreFillY - 3.0) / 2.0) {
722 *VRatioPrefetchY =
723 dml_max(
724 (double) PrefetchSourceLinesY
725 / LinesToRequestPrefetchPixelData,
726 (double) MaxNumSwathY
727 * SwathHeightY
728 / (LinesToRequestPrefetchPixelData
729 - (VInitPreFillY
730 - 3.0)
731 / 2.0));
732 *VRatioPrefetchY = dml_max(*VRatioPrefetchY, 1.0);
733 } else {
734 MyError = true;
735 *VRatioPrefetchY = 0;
736 }
737 }
738
739 *VRatioPrefetchC = (double) PrefetchSourceLinesC
740 / LinesToRequestPrefetchPixelData;
741 *VRatioPrefetchC = dml_max(*VRatioPrefetchC, 1.0);
742
743 if ((SwathHeightC > 4)) {
744 if (LinesToRequestPrefetchPixelData > (VInitPreFillC - 3.0) / 2.0) {
745 *VRatioPrefetchC =
746 dml_max(
747 *VRatioPrefetchC,
748 (double) MaxNumSwathC
749 * SwathHeightC
750 / (LinesToRequestPrefetchPixelData
751 - (VInitPreFillC
752 - 3.0)
753 / 2.0));
754 *VRatioPrefetchC = dml_max(*VRatioPrefetchC, 1.0);
755 } else {
756 MyError = true;
757 *VRatioPrefetchC = 0;
758 }
759 }
760
761 *RequiredPrefetchPixDataBW =
762 DPPPerPlane
763 * ((double) PrefetchSourceLinesY
764 / LinesToRequestPrefetchPixelData
765 * dml_ceil(
766 BytePerPixelDETY,
767 1)
768 + (double) PrefetchSourceLinesC
769 / LinesToRequestPrefetchPixelData
770 * dml_ceil(
771 BytePerPixelDETC,
772 2)
773 / 2)
774 * SwathWidthY / LineTime;
775 } else {
776 MyError = true;
777 *VRatioPrefetchY = 0;
778 *VRatioPrefetchC = 0;
779 *RequiredPrefetchPixDataBW = 0;
780 }
781
782 } else {
783 MyError = true;
784 }
785
786 if (MyError) {
787 *PrefetchBandwidth = 0;
788 TimeForFetchingMetaPTE = 0;
789 TimeForFetchingRowInVBlank = 0;
790 *DestinationLinesToRequestVMInVBlank = 0;
791 *DestinationLinesToRequestRowInVBlank = 0;
792 *DestinationLinesForPrefetch = 0;
793 LinesToRequestPrefetchPixelData = 0;
794 *VRatioPrefetchY = 0;
795 *VRatioPrefetchC = 0;
796 *RequiredPrefetchPixDataBW = 0;
797 }
798
799 return MyError;
800 }
801
RoundToDFSGranularityUp(double Clock,double VCOSpeed)802 static double RoundToDFSGranularityUp(double Clock, double VCOSpeed)
803 {
804 return VCOSpeed * 4 / dml_floor(VCOSpeed * 4 / Clock, 1);
805 }
806
RoundToDFSGranularityDown(double Clock,double VCOSpeed)807 static double RoundToDFSGranularityDown(double Clock, double VCOSpeed)
808 {
809 return VCOSpeed * 4 / dml_ceil(VCOSpeed * 4 / Clock, 1);
810 }
811
CalculatePrefetchSourceLines(struct display_mode_lib * mode_lib,double VRatio,double vtaps,bool Interlace,bool ProgressiveToInterlaceUnitInOPP,unsigned int SwathHeight,unsigned int ViewportYStart,double * VInitPreFill,unsigned int * MaxNumSwath)812 static double CalculatePrefetchSourceLines(
813 struct display_mode_lib *mode_lib,
814 double VRatio,
815 double vtaps,
816 bool Interlace,
817 bool ProgressiveToInterlaceUnitInOPP,
818 unsigned int SwathHeight,
819 unsigned int ViewportYStart,
820 double *VInitPreFill,
821 unsigned int *MaxNumSwath)
822 {
823 unsigned int MaxPartialSwath;
824
825 if (ProgressiveToInterlaceUnitInOPP)
826 *VInitPreFill = dml_floor((VRatio + vtaps + 1) / 2.0, 1);
827 else
828 *VInitPreFill = dml_floor((VRatio + vtaps + 1 + Interlace * 0.5 * VRatio) / 2.0, 1);
829
830 if (!mode_lib->vba.IgnoreViewportPositioning) {
831
832 *MaxNumSwath = dml_ceil((*VInitPreFill - 1.0) / SwathHeight, 1) + 1.0;
833
834 if (*VInitPreFill > 1.0)
835 MaxPartialSwath = (unsigned int) (*VInitPreFill - 2) % SwathHeight;
836 else
837 MaxPartialSwath = (unsigned int) (*VInitPreFill + SwathHeight - 2)
838 % SwathHeight;
839 MaxPartialSwath = dml_max(1U, MaxPartialSwath);
840
841 } else {
842
843 if (ViewportYStart != 0)
844 dml_print(
845 "WARNING DML: using viewport y position of 0 even though actual viewport y position is non-zero in prefetch source lines calculation\n");
846
847 *MaxNumSwath = dml_ceil(*VInitPreFill / SwathHeight, 1);
848
849 if (*VInitPreFill > 1.0)
850 MaxPartialSwath = (unsigned int) (*VInitPreFill - 1) % SwathHeight;
851 else
852 MaxPartialSwath = (unsigned int) (*VInitPreFill + SwathHeight - 1)
853 % SwathHeight;
854 }
855
856 return *MaxNumSwath * SwathHeight + MaxPartialSwath;
857 }
858
CalculateVMAndRowBytes(struct display_mode_lib * mode_lib,bool DCCEnable,unsigned int BlockHeight256Bytes,unsigned int BlockWidth256Bytes,enum source_format_class SourcePixelFormat,unsigned int SurfaceTiling,unsigned int BytePerPixel,enum scan_direction_class ScanDirection,unsigned int ViewportWidth,unsigned int ViewportHeight,unsigned int SwathWidth,bool GPUVMEnable,unsigned int VMMPageSize,unsigned int PTEBufferSizeInRequestsLuma,unsigned int PDEProcessingBufIn64KBReqs,unsigned int Pitch,unsigned int DCCMetaPitch,unsigned int * MacroTileWidth,unsigned int * MetaRowByte,unsigned int * PixelPTEBytesPerRow,bool * PTEBufferSizeNotExceeded,unsigned int * dpte_row_height,unsigned int * meta_row_height)859 static unsigned int CalculateVMAndRowBytes(
860 struct display_mode_lib *mode_lib,
861 bool DCCEnable,
862 unsigned int BlockHeight256Bytes,
863 unsigned int BlockWidth256Bytes,
864 enum source_format_class SourcePixelFormat,
865 unsigned int SurfaceTiling,
866 unsigned int BytePerPixel,
867 enum scan_direction_class ScanDirection,
868 unsigned int ViewportWidth,
869 unsigned int ViewportHeight,
870 unsigned int SwathWidth,
871 bool GPUVMEnable,
872 unsigned int VMMPageSize,
873 unsigned int PTEBufferSizeInRequestsLuma,
874 unsigned int PDEProcessingBufIn64KBReqs,
875 unsigned int Pitch,
876 unsigned int DCCMetaPitch,
877 unsigned int *MacroTileWidth,
878 unsigned int *MetaRowByte,
879 unsigned int *PixelPTEBytesPerRow,
880 bool *PTEBufferSizeNotExceeded,
881 unsigned int *dpte_row_height,
882 unsigned int *meta_row_height)
883 {
884 unsigned int MetaRequestHeight;
885 unsigned int MetaRequestWidth;
886 unsigned int MetaSurfWidth;
887 unsigned int MetaSurfHeight;
888 unsigned int MPDEBytesFrame;
889 unsigned int MetaPTEBytesFrame;
890 unsigned int DCCMetaSurfaceBytes;
891
892 unsigned int MacroTileSizeBytes;
893 unsigned int MacroTileHeight;
894 unsigned int DPDE0BytesFrame;
895 unsigned int ExtraDPDEBytesFrame;
896 unsigned int PDEAndMetaPTEBytesFrame;
897
898 if (DCCEnable == true) {
899 MetaRequestHeight = 8 * BlockHeight256Bytes;
900 MetaRequestWidth = 8 * BlockWidth256Bytes;
901 if (ScanDirection == dm_horz) {
902 *meta_row_height = MetaRequestHeight;
903 MetaSurfWidth = dml_ceil((double) SwathWidth - 1, MetaRequestWidth)
904 + MetaRequestWidth;
905 *MetaRowByte = MetaSurfWidth * MetaRequestHeight * BytePerPixel / 256.0;
906 } else {
907 *meta_row_height = MetaRequestWidth;
908 MetaSurfHeight = dml_ceil((double) SwathWidth - 1, MetaRequestHeight)
909 + MetaRequestHeight;
910 *MetaRowByte = MetaSurfHeight * MetaRequestWidth * BytePerPixel / 256.0;
911 }
912 if (ScanDirection == dm_horz) {
913 DCCMetaSurfaceBytes = DCCMetaPitch
914 * (dml_ceil(ViewportHeight - 1, 64 * BlockHeight256Bytes)
915 + 64 * BlockHeight256Bytes) * BytePerPixel
916 / 256;
917 } else {
918 DCCMetaSurfaceBytes = DCCMetaPitch
919 * (dml_ceil(
920 (double) ViewportHeight - 1,
921 64 * BlockHeight256Bytes)
922 + 64 * BlockHeight256Bytes) * BytePerPixel
923 / 256;
924 }
925 if (GPUVMEnable == true) {
926 MetaPTEBytesFrame = (dml_ceil(
927 (double) (DCCMetaSurfaceBytes - VMMPageSize)
928 / (8 * VMMPageSize),
929 1) + 1) * 64;
930 MPDEBytesFrame = 128 * (mode_lib->vba.GPUVMMaxPageTableLevels - 1);
931 } else {
932 MetaPTEBytesFrame = 0;
933 MPDEBytesFrame = 0;
934 }
935 } else {
936 MetaPTEBytesFrame = 0;
937 MPDEBytesFrame = 0;
938 *MetaRowByte = 0;
939 }
940
941 if (SurfaceTiling == dm_sw_linear || SurfaceTiling == dm_sw_gfx7_2d_thin_gl || SurfaceTiling == dm_sw_gfx7_2d_thin_l_vp) {
942 MacroTileSizeBytes = 256;
943 MacroTileHeight = BlockHeight256Bytes;
944 } else if (SurfaceTiling == dm_sw_4kb_s || SurfaceTiling == dm_sw_4kb_s_x
945 || SurfaceTiling == dm_sw_4kb_d || SurfaceTiling == dm_sw_4kb_d_x) {
946 MacroTileSizeBytes = 4096;
947 MacroTileHeight = 4 * BlockHeight256Bytes;
948 } else if (SurfaceTiling == dm_sw_64kb_s || SurfaceTiling == dm_sw_64kb_s_t
949 || SurfaceTiling == dm_sw_64kb_s_x || SurfaceTiling == dm_sw_64kb_d
950 || SurfaceTiling == dm_sw_64kb_d_t || SurfaceTiling == dm_sw_64kb_d_x
951 || SurfaceTiling == dm_sw_64kb_r_x) {
952 MacroTileSizeBytes = 65536;
953 MacroTileHeight = 16 * BlockHeight256Bytes;
954 } else {
955 MacroTileSizeBytes = 262144;
956 MacroTileHeight = 32 * BlockHeight256Bytes;
957 }
958 *MacroTileWidth = MacroTileSizeBytes / BytePerPixel / MacroTileHeight;
959
960 if (GPUVMEnable == true && mode_lib->vba.GPUVMMaxPageTableLevels > 1) {
961 if (ScanDirection == dm_horz) {
962 DPDE0BytesFrame =
963 64
964 * (dml_ceil(
965 ((Pitch
966 * (dml_ceil(
967 ViewportHeight
968 - 1,
969 MacroTileHeight)
970 + MacroTileHeight)
971 * BytePerPixel)
972 - MacroTileSizeBytes)
973 / (8
974 * 2097152),
975 1) + 1);
976 } else {
977 DPDE0BytesFrame =
978 64
979 * (dml_ceil(
980 ((Pitch
981 * (dml_ceil(
982 (double) SwathWidth
983 - 1,
984 MacroTileHeight)
985 + MacroTileHeight)
986 * BytePerPixel)
987 - MacroTileSizeBytes)
988 / (8
989 * 2097152),
990 1) + 1);
991 }
992 ExtraDPDEBytesFrame = 128 * (mode_lib->vba.GPUVMMaxPageTableLevels - 2);
993 } else {
994 DPDE0BytesFrame = 0;
995 ExtraDPDEBytesFrame = 0;
996 }
997
998 PDEAndMetaPTEBytesFrame = MetaPTEBytesFrame + MPDEBytesFrame + DPDE0BytesFrame
999 + ExtraDPDEBytesFrame;
1000
1001 if (GPUVMEnable == true) {
1002 unsigned int PTERequestSize;
1003 unsigned int PixelPTEReqHeight;
1004 unsigned int PixelPTEReqWidth;
1005 double FractionOfPTEReturnDrop;
1006 unsigned int EffectivePDEProcessingBufIn64KBReqs;
1007
1008 if (SurfaceTiling == dm_sw_linear) {
1009 PixelPTEReqHeight = 1;
1010 PixelPTEReqWidth = 8.0 * VMMPageSize / BytePerPixel;
1011 PTERequestSize = 64;
1012 FractionOfPTEReturnDrop = 0;
1013 } else if (MacroTileSizeBytes == 4096) {
1014 PixelPTEReqHeight = MacroTileHeight;
1015 PixelPTEReqWidth = 8 * *MacroTileWidth;
1016 PTERequestSize = 64;
1017 if (ScanDirection == dm_horz)
1018 FractionOfPTEReturnDrop = 0;
1019 else
1020 FractionOfPTEReturnDrop = 7 / 8;
1021 } else if (VMMPageSize == 4096 && MacroTileSizeBytes > 4096) {
1022 PixelPTEReqHeight = 16 * BlockHeight256Bytes;
1023 PixelPTEReqWidth = 16 * BlockWidth256Bytes;
1024 PTERequestSize = 128;
1025 FractionOfPTEReturnDrop = 0;
1026 } else {
1027 PixelPTEReqHeight = MacroTileHeight;
1028 PixelPTEReqWidth = 8 * *MacroTileWidth;
1029 PTERequestSize = 64;
1030 FractionOfPTEReturnDrop = 0;
1031 }
1032
1033 if (SourcePixelFormat == dm_420_8 || SourcePixelFormat == dm_420_10)
1034 EffectivePDEProcessingBufIn64KBReqs = PDEProcessingBufIn64KBReqs / 2;
1035 else
1036 EffectivePDEProcessingBufIn64KBReqs = PDEProcessingBufIn64KBReqs;
1037
1038 if (SurfaceTiling == dm_sw_linear) {
1039 *dpte_row_height =
1040 dml_min(
1041 128,
1042 1
1043 << (unsigned int) dml_floor(
1044 dml_log2(
1045 dml_min(
1046 (double) PTEBufferSizeInRequestsLuma
1047 * PixelPTEReqWidth,
1048 EffectivePDEProcessingBufIn64KBReqs
1049 * 65536.0
1050 / BytePerPixel)
1051 / Pitch),
1052 1));
1053 *PixelPTEBytesPerRow = PTERequestSize
1054 * (dml_ceil(
1055 (double) (Pitch * *dpte_row_height - 1)
1056 / PixelPTEReqWidth,
1057 1) + 1);
1058 } else if (ScanDirection == dm_horz) {
1059 *dpte_row_height = PixelPTEReqHeight;
1060 *PixelPTEBytesPerRow = PTERequestSize
1061 * (dml_ceil(((double) SwathWidth - 1) / PixelPTEReqWidth, 1)
1062 + 1);
1063 } else {
1064 *dpte_row_height = dml_min(PixelPTEReqWidth, *MacroTileWidth);
1065 *PixelPTEBytesPerRow = PTERequestSize
1066 * (dml_ceil(
1067 ((double) SwathWidth - 1)
1068 / PixelPTEReqHeight,
1069 1) + 1);
1070 }
1071 if (*PixelPTEBytesPerRow * (1 - FractionOfPTEReturnDrop)
1072 <= 64 * PTEBufferSizeInRequestsLuma) {
1073 *PTEBufferSizeNotExceeded = true;
1074 } else {
1075 *PTEBufferSizeNotExceeded = false;
1076 }
1077 } else {
1078 *PixelPTEBytesPerRow = 0;
1079 *PTEBufferSizeNotExceeded = true;
1080 }
1081
1082 return PDEAndMetaPTEBytesFrame;
1083 }
1084
dml20_DISPCLKDPPCLKDCFCLKDeepSleepPrefetchParametersWatermarksAndPerformanceCalculation(struct display_mode_lib * mode_lib)1085 static void dml20_DISPCLKDPPCLKDCFCLKDeepSleepPrefetchParametersWatermarksAndPerformanceCalculation(
1086 struct display_mode_lib *mode_lib)
1087 {
1088 unsigned int j, k;
1089
1090 mode_lib->vba.WritebackDISPCLK = 0.0;
1091 mode_lib->vba.DISPCLKWithRamping = 0;
1092 mode_lib->vba.DISPCLKWithoutRamping = 0;
1093 mode_lib->vba.GlobalDPPCLK = 0.0;
1094
1095 // dml_ml->vba.DISPCLK and dml_ml->vba.DPPCLK Calculation
1096 //
1097 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1098 if (mode_lib->vba.WritebackEnable[k]) {
1099 mode_lib->vba.WritebackDISPCLK =
1100 dml_max(
1101 mode_lib->vba.WritebackDISPCLK,
1102 CalculateWriteBackDISPCLK(
1103 mode_lib->vba.WritebackPixelFormat[k],
1104 mode_lib->vba.PixelClock[k],
1105 mode_lib->vba.WritebackHRatio[k],
1106 mode_lib->vba.WritebackVRatio[k],
1107 mode_lib->vba.WritebackLumaHTaps[k],
1108 mode_lib->vba.WritebackLumaVTaps[k],
1109 mode_lib->vba.WritebackChromaHTaps[k],
1110 mode_lib->vba.WritebackChromaVTaps[k],
1111 mode_lib->vba.WritebackDestinationWidth[k],
1112 mode_lib->vba.HTotal[k],
1113 mode_lib->vba.WritebackChromaLineBufferWidth));
1114 }
1115 }
1116
1117 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1118 if (mode_lib->vba.HRatio[k] > 1) {
1119 mode_lib->vba.PSCL_THROUGHPUT_LUMA[k] = dml_min(
1120 mode_lib->vba.MaxDCHUBToPSCLThroughput,
1121 mode_lib->vba.MaxPSCLToLBThroughput
1122 * mode_lib->vba.HRatio[k]
1123 / dml_ceil(
1124 mode_lib->vba.htaps[k]
1125 / 6.0,
1126 1));
1127 } else {
1128 mode_lib->vba.PSCL_THROUGHPUT_LUMA[k] = dml_min(
1129 mode_lib->vba.MaxDCHUBToPSCLThroughput,
1130 mode_lib->vba.MaxPSCLToLBThroughput);
1131 }
1132
1133 mode_lib->vba.DPPCLKUsingSingleDPPLuma =
1134 mode_lib->vba.PixelClock[k]
1135 * dml_max(
1136 mode_lib->vba.vtaps[k] / 6.0
1137 * dml_min(
1138 1.0,
1139 mode_lib->vba.HRatio[k]),
1140 dml_max(
1141 mode_lib->vba.HRatio[k]
1142 * mode_lib->vba.VRatio[k]
1143 / mode_lib->vba.PSCL_THROUGHPUT_LUMA[k],
1144 1.0));
1145
1146 if ((mode_lib->vba.htaps[k] > 6 || mode_lib->vba.vtaps[k] > 6)
1147 && mode_lib->vba.DPPCLKUsingSingleDPPLuma
1148 < 2 * mode_lib->vba.PixelClock[k]) {
1149 mode_lib->vba.DPPCLKUsingSingleDPPLuma = 2 * mode_lib->vba.PixelClock[k];
1150 }
1151
1152 if ((mode_lib->vba.SourcePixelFormat[k] != dm_420_8
1153 && mode_lib->vba.SourcePixelFormat[k] != dm_420_10)) {
1154 mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k] = 0.0;
1155 mode_lib->vba.DPPCLKUsingSingleDPP[k] =
1156 mode_lib->vba.DPPCLKUsingSingleDPPLuma;
1157 } else {
1158 if (mode_lib->vba.HRatio[k] > 1) {
1159 mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k] =
1160 dml_min(
1161 mode_lib->vba.MaxDCHUBToPSCLThroughput,
1162 mode_lib->vba.MaxPSCLToLBThroughput
1163 * mode_lib->vba.HRatio[k]
1164 / 2
1165 / dml_ceil(
1166 mode_lib->vba.HTAPsChroma[k]
1167 / 6.0,
1168 1.0));
1169 } else {
1170 mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k] = dml_min(
1171 mode_lib->vba.MaxDCHUBToPSCLThroughput,
1172 mode_lib->vba.MaxPSCLToLBThroughput);
1173 }
1174 mode_lib->vba.DPPCLKUsingSingleDPPChroma =
1175 mode_lib->vba.PixelClock[k]
1176 * dml_max(
1177 mode_lib->vba.VTAPsChroma[k]
1178 / 6.0
1179 * dml_min(
1180 1.0,
1181 mode_lib->vba.HRatio[k]
1182 / 2),
1183 dml_max(
1184 mode_lib->vba.HRatio[k]
1185 * mode_lib->vba.VRatio[k]
1186 / 4
1187 / mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k],
1188 1.0));
1189
1190 if ((mode_lib->vba.HTAPsChroma[k] > 6 || mode_lib->vba.VTAPsChroma[k] > 6)
1191 && mode_lib->vba.DPPCLKUsingSingleDPPChroma
1192 < 2 * mode_lib->vba.PixelClock[k]) {
1193 mode_lib->vba.DPPCLKUsingSingleDPPChroma = 2
1194 * mode_lib->vba.PixelClock[k];
1195 }
1196
1197 mode_lib->vba.DPPCLKUsingSingleDPP[k] = dml_max(
1198 mode_lib->vba.DPPCLKUsingSingleDPPLuma,
1199 mode_lib->vba.DPPCLKUsingSingleDPPChroma);
1200 }
1201 }
1202
1203 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1204 if (mode_lib->vba.BlendingAndTiming[k] != k)
1205 continue;
1206 if (mode_lib->vba.ODMCombineEnabled[k]) {
1207 mode_lib->vba.DISPCLKWithRamping =
1208 dml_max(
1209 mode_lib->vba.DISPCLKWithRamping,
1210 mode_lib->vba.PixelClock[k] / 2
1211 * (1
1212 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1213 / 100)
1214 * (1
1215 + mode_lib->vba.DISPCLKRampingMargin
1216 / 100));
1217 mode_lib->vba.DISPCLKWithoutRamping =
1218 dml_max(
1219 mode_lib->vba.DISPCLKWithoutRamping,
1220 mode_lib->vba.PixelClock[k] / 2
1221 * (1
1222 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1223 / 100));
1224 } else if (!mode_lib->vba.ODMCombineEnabled[k]) {
1225 mode_lib->vba.DISPCLKWithRamping =
1226 dml_max(
1227 mode_lib->vba.DISPCLKWithRamping,
1228 mode_lib->vba.PixelClock[k]
1229 * (1
1230 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1231 / 100)
1232 * (1
1233 + mode_lib->vba.DISPCLKRampingMargin
1234 / 100));
1235 mode_lib->vba.DISPCLKWithoutRamping =
1236 dml_max(
1237 mode_lib->vba.DISPCLKWithoutRamping,
1238 mode_lib->vba.PixelClock[k]
1239 * (1
1240 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1241 / 100));
1242 }
1243 }
1244
1245 mode_lib->vba.DISPCLKWithRamping = dml_max(
1246 mode_lib->vba.DISPCLKWithRamping,
1247 mode_lib->vba.WritebackDISPCLK);
1248 mode_lib->vba.DISPCLKWithoutRamping = dml_max(
1249 mode_lib->vba.DISPCLKWithoutRamping,
1250 mode_lib->vba.WritebackDISPCLK);
1251
1252 ASSERT(mode_lib->vba.DISPCLKDPPCLKVCOSpeed != 0);
1253 mode_lib->vba.DISPCLKWithRampingRoundedToDFSGranularity = RoundToDFSGranularityUp(
1254 mode_lib->vba.DISPCLKWithRamping,
1255 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
1256 mode_lib->vba.DISPCLKWithoutRampingRoundedToDFSGranularity = RoundToDFSGranularityUp(
1257 mode_lib->vba.DISPCLKWithoutRamping,
1258 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
1259 mode_lib->vba.MaxDispclkRoundedToDFSGranularity = RoundToDFSGranularityDown(
1260 mode_lib->vba.soc.clock_limits[mode_lib->vba.soc.num_states].dispclk_mhz,
1261 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
1262 if (mode_lib->vba.DISPCLKWithoutRampingRoundedToDFSGranularity
1263 > mode_lib->vba.MaxDispclkRoundedToDFSGranularity) {
1264 mode_lib->vba.DISPCLK_calculated =
1265 mode_lib->vba.DISPCLKWithoutRampingRoundedToDFSGranularity;
1266 } else if (mode_lib->vba.DISPCLKWithRampingRoundedToDFSGranularity
1267 > mode_lib->vba.MaxDispclkRoundedToDFSGranularity) {
1268 mode_lib->vba.DISPCLK_calculated = mode_lib->vba.MaxDispclkRoundedToDFSGranularity;
1269 } else {
1270 mode_lib->vba.DISPCLK_calculated =
1271 mode_lib->vba.DISPCLKWithRampingRoundedToDFSGranularity;
1272 }
1273 DTRACE(" dispclk_mhz (calculated) = %f", mode_lib->vba.DISPCLK_calculated);
1274
1275 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1276 if (mode_lib->vba.DPPPerPlane[k] == 0) {
1277 mode_lib->vba.DPPCLK_calculated[k] = 0;
1278 } else {
1279 mode_lib->vba.DPPCLK_calculated[k] = mode_lib->vba.DPPCLKUsingSingleDPP[k]
1280 / mode_lib->vba.DPPPerPlane[k]
1281 * (1 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100);
1282 }
1283 mode_lib->vba.GlobalDPPCLK = dml_max(
1284 mode_lib->vba.GlobalDPPCLK,
1285 mode_lib->vba.DPPCLK_calculated[k]);
1286 }
1287 mode_lib->vba.GlobalDPPCLK = RoundToDFSGranularityUp(
1288 mode_lib->vba.GlobalDPPCLK,
1289 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
1290 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1291 mode_lib->vba.DPPCLK_calculated[k] = mode_lib->vba.GlobalDPPCLK / 255
1292 * dml_ceil(
1293 mode_lib->vba.DPPCLK_calculated[k] * 255
1294 / mode_lib->vba.GlobalDPPCLK,
1295 1);
1296 DTRACE(" dppclk_mhz[%i] (calculated) = %f", k, mode_lib->vba.DPPCLK_calculated[k]);
1297 }
1298
1299 // Urgent Watermark
1300 mode_lib->vba.DCCEnabledAnyPlane = false;
1301 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k)
1302 if (mode_lib->vba.DCCEnable[k])
1303 mode_lib->vba.DCCEnabledAnyPlane = true;
1304
1305 mode_lib->vba.ReturnBandwidthToDCN = dml_min(
1306 mode_lib->vba.ReturnBusWidth * mode_lib->vba.DCFCLK,
1307 mode_lib->vba.FabricAndDRAMBandwidth * 1000)
1308 * mode_lib->vba.PercentOfIdealDRAMFabricAndSDPPortBWReceivedAfterUrgLatencyPixelDataOnly / 100;
1309
1310 mode_lib->vba.ReturnBW = mode_lib->vba.ReturnBandwidthToDCN;
1311 mode_lib->vba.ReturnBW = adjust_ReturnBW(
1312 mode_lib,
1313 mode_lib->vba.ReturnBW,
1314 mode_lib->vba.DCCEnabledAnyPlane,
1315 mode_lib->vba.ReturnBandwidthToDCN);
1316
1317 // Let's do this calculation again??
1318 mode_lib->vba.ReturnBandwidthToDCN = dml_min(
1319 mode_lib->vba.ReturnBusWidth * mode_lib->vba.DCFCLK,
1320 mode_lib->vba.FabricAndDRAMBandwidth * 1000);
1321 mode_lib->vba.ReturnBW = adjust_ReturnBW(
1322 mode_lib,
1323 mode_lib->vba.ReturnBW,
1324 mode_lib->vba.DCCEnabledAnyPlane,
1325 mode_lib->vba.ReturnBandwidthToDCN);
1326
1327 DTRACE(" dcfclk_mhz = %f", mode_lib->vba.DCFCLK);
1328 DTRACE(" return_bw_to_dcn = %f", mode_lib->vba.ReturnBandwidthToDCN);
1329 DTRACE(" return_bus_bw = %f", mode_lib->vba.ReturnBW);
1330
1331 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1332 bool MainPlaneDoesODMCombine = false;
1333
1334 if (mode_lib->vba.SourceScan[k] == dm_horz)
1335 mode_lib->vba.SwathWidthSingleDPPY[k] = mode_lib->vba.ViewportWidth[k];
1336 else
1337 mode_lib->vba.SwathWidthSingleDPPY[k] = mode_lib->vba.ViewportHeight[k];
1338
1339 if (mode_lib->vba.ODMCombineEnabled[k] == dm_odm_combine_mode_2to1)
1340 MainPlaneDoesODMCombine = true;
1341 for (j = 0; j < mode_lib->vba.NumberOfActivePlanes; ++j)
1342 if (mode_lib->vba.BlendingAndTiming[k] == j
1343 && mode_lib->vba.ODMCombineEnabled[k] == dm_odm_combine_mode_2to1)
1344 MainPlaneDoesODMCombine = true;
1345
1346 if (MainPlaneDoesODMCombine == true)
1347 mode_lib->vba.SwathWidthY[k] = dml_min(
1348 (double) mode_lib->vba.SwathWidthSingleDPPY[k],
1349 dml_round(
1350 mode_lib->vba.HActive[k] / 2.0
1351 * mode_lib->vba.HRatio[k]));
1352 else {
1353 if (mode_lib->vba.DPPPerPlane[k] == 0) {
1354 mode_lib->vba.SwathWidthY[k] = 0;
1355 } else {
1356 mode_lib->vba.SwathWidthY[k] = mode_lib->vba.SwathWidthSingleDPPY[k]
1357 / mode_lib->vba.DPPPerPlane[k];
1358 }
1359 }
1360 }
1361
1362 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1363 if (mode_lib->vba.SourcePixelFormat[k] == dm_444_64) {
1364 mode_lib->vba.BytePerPixelDETY[k] = 8;
1365 mode_lib->vba.BytePerPixelDETC[k] = 0;
1366 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_32) {
1367 mode_lib->vba.BytePerPixelDETY[k] = 4;
1368 mode_lib->vba.BytePerPixelDETC[k] = 0;
1369 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_16) {
1370 mode_lib->vba.BytePerPixelDETY[k] = 2;
1371 mode_lib->vba.BytePerPixelDETC[k] = 0;
1372 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_8) {
1373 mode_lib->vba.BytePerPixelDETY[k] = 1;
1374 mode_lib->vba.BytePerPixelDETC[k] = 0;
1375 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8) {
1376 mode_lib->vba.BytePerPixelDETY[k] = 1;
1377 mode_lib->vba.BytePerPixelDETC[k] = 2;
1378 } else { // dm_420_10
1379 mode_lib->vba.BytePerPixelDETY[k] = 4.0 / 3.0;
1380 mode_lib->vba.BytePerPixelDETC[k] = 8.0 / 3.0;
1381 }
1382 }
1383
1384 mode_lib->vba.TotalDataReadBandwidth = 0.0;
1385 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1386 mode_lib->vba.ReadBandwidthPlaneLuma[k] = mode_lib->vba.SwathWidthSingleDPPY[k]
1387 * dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1)
1388 / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
1389 * mode_lib->vba.VRatio[k];
1390 mode_lib->vba.ReadBandwidthPlaneChroma[k] = mode_lib->vba.SwathWidthSingleDPPY[k]
1391 / 2 * dml_ceil(mode_lib->vba.BytePerPixelDETC[k], 2)
1392 / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
1393 * mode_lib->vba.VRatio[k] / 2;
1394 DTRACE(
1395 " read_bw[%i] = %fBps",
1396 k,
1397 mode_lib->vba.ReadBandwidthPlaneLuma[k]
1398 + mode_lib->vba.ReadBandwidthPlaneChroma[k]);
1399 mode_lib->vba.TotalDataReadBandwidth += mode_lib->vba.ReadBandwidthPlaneLuma[k]
1400 + mode_lib->vba.ReadBandwidthPlaneChroma[k];
1401 }
1402
1403 mode_lib->vba.TotalDCCActiveDPP = 0;
1404 mode_lib->vba.TotalActiveDPP = 0;
1405 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1406 mode_lib->vba.TotalActiveDPP = mode_lib->vba.TotalActiveDPP
1407 + mode_lib->vba.DPPPerPlane[k];
1408 if (mode_lib->vba.DCCEnable[k])
1409 mode_lib->vba.TotalDCCActiveDPP = mode_lib->vba.TotalDCCActiveDPP
1410 + mode_lib->vba.DPPPerPlane[k];
1411 }
1412
1413 mode_lib->vba.UrgentRoundTripAndOutOfOrderLatency =
1414 (mode_lib->vba.RoundTripPingLatencyCycles + 32) / mode_lib->vba.DCFCLK
1415 + mode_lib->vba.UrgentOutOfOrderReturnPerChannelPixelDataOnly
1416 * mode_lib->vba.NumberOfChannels
1417 / mode_lib->vba.ReturnBW;
1418
1419 mode_lib->vba.LastPixelOfLineExtraWatermark = 0;
1420 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1421 double DataFabricLineDeliveryTimeLuma, DataFabricLineDeliveryTimeChroma;
1422
1423 if (mode_lib->vba.VRatio[k] <= 1.0)
1424 mode_lib->vba.DisplayPipeLineDeliveryTimeLuma[k] =
1425 (double) mode_lib->vba.SwathWidthY[k]
1426 * mode_lib->vba.DPPPerPlane[k]
1427 / mode_lib->vba.HRatio[k]
1428 / mode_lib->vba.PixelClock[k];
1429 else
1430 mode_lib->vba.DisplayPipeLineDeliveryTimeLuma[k] =
1431 (double) mode_lib->vba.SwathWidthY[k]
1432 / mode_lib->vba.PSCL_THROUGHPUT_LUMA[k]
1433 / mode_lib->vba.DPPCLK[k];
1434
1435 DataFabricLineDeliveryTimeLuma = mode_lib->vba.SwathWidthSingleDPPY[k]
1436 * mode_lib->vba.SwathHeightY[k]
1437 * dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1)
1438 / (mode_lib->vba.ReturnBW * mode_lib->vba.ReadBandwidthPlaneLuma[k]
1439 / mode_lib->vba.TotalDataReadBandwidth);
1440 mode_lib->vba.LastPixelOfLineExtraWatermark = dml_max(
1441 mode_lib->vba.LastPixelOfLineExtraWatermark,
1442 DataFabricLineDeliveryTimeLuma
1443 - mode_lib->vba.DisplayPipeLineDeliveryTimeLuma[k]);
1444
1445 if (mode_lib->vba.BytePerPixelDETC[k] == 0)
1446 mode_lib->vba.DisplayPipeLineDeliveryTimeChroma[k] = 0.0;
1447 else if (mode_lib->vba.VRatio[k] / 2.0 <= 1.0)
1448 mode_lib->vba.DisplayPipeLineDeliveryTimeChroma[k] =
1449 mode_lib->vba.SwathWidthY[k] / 2.0
1450 * mode_lib->vba.DPPPerPlane[k]
1451 / (mode_lib->vba.HRatio[k] / 2.0)
1452 / mode_lib->vba.PixelClock[k];
1453 else
1454 mode_lib->vba.DisplayPipeLineDeliveryTimeChroma[k] =
1455 mode_lib->vba.SwathWidthY[k] / 2.0
1456 / mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k]
1457 / mode_lib->vba.DPPCLK[k];
1458
1459 DataFabricLineDeliveryTimeChroma = mode_lib->vba.SwathWidthSingleDPPY[k] / 2.0
1460 * mode_lib->vba.SwathHeightC[k]
1461 * dml_ceil(mode_lib->vba.BytePerPixelDETC[k], 2)
1462 / (mode_lib->vba.ReturnBW
1463 * mode_lib->vba.ReadBandwidthPlaneChroma[k]
1464 / mode_lib->vba.TotalDataReadBandwidth);
1465 mode_lib->vba.LastPixelOfLineExtraWatermark =
1466 dml_max(
1467 mode_lib->vba.LastPixelOfLineExtraWatermark,
1468 DataFabricLineDeliveryTimeChroma
1469 - mode_lib->vba.DisplayPipeLineDeliveryTimeChroma[k]);
1470 }
1471
1472 mode_lib->vba.UrgentExtraLatency = mode_lib->vba.UrgentRoundTripAndOutOfOrderLatency
1473 + (mode_lib->vba.TotalActiveDPP * mode_lib->vba.PixelChunkSizeInKByte
1474 + mode_lib->vba.TotalDCCActiveDPP
1475 * mode_lib->vba.MetaChunkSize) * 1024.0
1476 / mode_lib->vba.ReturnBW;
1477
1478 if (mode_lib->vba.GPUVMEnable)
1479 mode_lib->vba.UrgentExtraLatency += mode_lib->vba.TotalActiveDPP
1480 * mode_lib->vba.PTEGroupSize / mode_lib->vba.ReturnBW;
1481
1482 mode_lib->vba.UrgentWatermark = mode_lib->vba.UrgentLatencyPixelDataOnly
1483 + mode_lib->vba.LastPixelOfLineExtraWatermark
1484 + mode_lib->vba.UrgentExtraLatency;
1485
1486 DTRACE(" urgent_extra_latency = %fus", mode_lib->vba.UrgentExtraLatency);
1487 DTRACE(" wm_urgent = %fus", mode_lib->vba.UrgentWatermark);
1488
1489 mode_lib->vba.UrgentLatency = mode_lib->vba.UrgentLatencyPixelDataOnly;
1490
1491 mode_lib->vba.TotalActiveWriteback = 0;
1492 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1493 if (mode_lib->vba.WritebackEnable[k])
1494 mode_lib->vba.TotalActiveWriteback = mode_lib->vba.TotalActiveWriteback + mode_lib->vba.ActiveWritebacksPerPlane[k];
1495 }
1496
1497 if (mode_lib->vba.TotalActiveWriteback <= 1)
1498 mode_lib->vba.WritebackUrgentWatermark = mode_lib->vba.WritebackLatency;
1499 else
1500 mode_lib->vba.WritebackUrgentWatermark = mode_lib->vba.WritebackLatency
1501 + mode_lib->vba.WritebackChunkSize * 1024.0 / 32
1502 / mode_lib->vba.SOCCLK;
1503
1504 DTRACE(" wm_wb_urgent = %fus", mode_lib->vba.WritebackUrgentWatermark);
1505
1506 // NB P-State/DRAM Clock Change Watermark
1507 mode_lib->vba.DRAMClockChangeWatermark = mode_lib->vba.DRAMClockChangeLatency
1508 + mode_lib->vba.UrgentWatermark;
1509
1510 DTRACE(" wm_pstate_change = %fus", mode_lib->vba.DRAMClockChangeWatermark);
1511
1512 DTRACE(" calculating wb pstate watermark");
1513 DTRACE(" total wb outputs %d", mode_lib->vba.TotalActiveWriteback);
1514 DTRACE(" socclk frequency %f Mhz", mode_lib->vba.SOCCLK);
1515
1516 if (mode_lib->vba.TotalActiveWriteback <= 1)
1517 mode_lib->vba.WritebackDRAMClockChangeWatermark =
1518 mode_lib->vba.DRAMClockChangeLatency
1519 + mode_lib->vba.WritebackLatency;
1520 else
1521 mode_lib->vba.WritebackDRAMClockChangeWatermark =
1522 mode_lib->vba.DRAMClockChangeLatency
1523 + mode_lib->vba.WritebackLatency
1524 + mode_lib->vba.WritebackChunkSize * 1024.0 / 32
1525 / mode_lib->vba.SOCCLK;
1526
1527 DTRACE(" wm_wb_pstate %fus", mode_lib->vba.WritebackDRAMClockChangeWatermark);
1528
1529 // Stutter Efficiency
1530 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1531 mode_lib->vba.LinesInDETY[k] = mode_lib->vba.DETBufferSizeY[k]
1532 / mode_lib->vba.BytePerPixelDETY[k] / mode_lib->vba.SwathWidthY[k];
1533 mode_lib->vba.LinesInDETYRoundedDownToSwath[k] = dml_floor(
1534 mode_lib->vba.LinesInDETY[k],
1535 mode_lib->vba.SwathHeightY[k]);
1536 mode_lib->vba.FullDETBufferingTimeY[k] =
1537 mode_lib->vba.LinesInDETYRoundedDownToSwath[k]
1538 * (mode_lib->vba.HTotal[k]
1539 / mode_lib->vba.PixelClock[k])
1540 / mode_lib->vba.VRatio[k];
1541 if (mode_lib->vba.BytePerPixelDETC[k] > 0) {
1542 mode_lib->vba.LinesInDETC[k] = mode_lib->vba.DETBufferSizeC[k]
1543 / mode_lib->vba.BytePerPixelDETC[k]
1544 / (mode_lib->vba.SwathWidthY[k] / 2);
1545 mode_lib->vba.LinesInDETCRoundedDownToSwath[k] = dml_floor(
1546 mode_lib->vba.LinesInDETC[k],
1547 mode_lib->vba.SwathHeightC[k]);
1548 mode_lib->vba.FullDETBufferingTimeC[k] =
1549 mode_lib->vba.LinesInDETCRoundedDownToSwath[k]
1550 * (mode_lib->vba.HTotal[k]
1551 / mode_lib->vba.PixelClock[k])
1552 / (mode_lib->vba.VRatio[k] / 2);
1553 } else {
1554 mode_lib->vba.LinesInDETC[k] = 0;
1555 mode_lib->vba.LinesInDETCRoundedDownToSwath[k] = 0;
1556 mode_lib->vba.FullDETBufferingTimeC[k] = 999999;
1557 }
1558 }
1559
1560 mode_lib->vba.MinFullDETBufferingTime = 999999.0;
1561 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1562 if (mode_lib->vba.FullDETBufferingTimeY[k]
1563 < mode_lib->vba.MinFullDETBufferingTime) {
1564 mode_lib->vba.MinFullDETBufferingTime =
1565 mode_lib->vba.FullDETBufferingTimeY[k];
1566 mode_lib->vba.FrameTimeForMinFullDETBufferingTime =
1567 (double) mode_lib->vba.VTotal[k] * mode_lib->vba.HTotal[k]
1568 / mode_lib->vba.PixelClock[k];
1569 }
1570 if (mode_lib->vba.FullDETBufferingTimeC[k]
1571 < mode_lib->vba.MinFullDETBufferingTime) {
1572 mode_lib->vba.MinFullDETBufferingTime =
1573 mode_lib->vba.FullDETBufferingTimeC[k];
1574 mode_lib->vba.FrameTimeForMinFullDETBufferingTime =
1575 (double) mode_lib->vba.VTotal[k] * mode_lib->vba.HTotal[k]
1576 / mode_lib->vba.PixelClock[k];
1577 }
1578 }
1579
1580 mode_lib->vba.AverageReadBandwidthGBytePerSecond = 0.0;
1581 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1582 if (mode_lib->vba.DCCEnable[k]) {
1583 mode_lib->vba.AverageReadBandwidthGBytePerSecond =
1584 mode_lib->vba.AverageReadBandwidthGBytePerSecond
1585 + mode_lib->vba.ReadBandwidthPlaneLuma[k]
1586 / mode_lib->vba.DCCRate[k]
1587 / 1000
1588 + mode_lib->vba.ReadBandwidthPlaneChroma[k]
1589 / mode_lib->vba.DCCRate[k]
1590 / 1000;
1591 } else {
1592 mode_lib->vba.AverageReadBandwidthGBytePerSecond =
1593 mode_lib->vba.AverageReadBandwidthGBytePerSecond
1594 + mode_lib->vba.ReadBandwidthPlaneLuma[k]
1595 / 1000
1596 + mode_lib->vba.ReadBandwidthPlaneChroma[k]
1597 / 1000;
1598 }
1599 if (mode_lib->vba.DCCEnable[k]) {
1600 mode_lib->vba.AverageReadBandwidthGBytePerSecond =
1601 mode_lib->vba.AverageReadBandwidthGBytePerSecond
1602 + mode_lib->vba.ReadBandwidthPlaneLuma[k]
1603 / 1000 / 256
1604 + mode_lib->vba.ReadBandwidthPlaneChroma[k]
1605 / 1000 / 256;
1606 }
1607 if (mode_lib->vba.GPUVMEnable) {
1608 mode_lib->vba.AverageReadBandwidthGBytePerSecond =
1609 mode_lib->vba.AverageReadBandwidthGBytePerSecond
1610 + mode_lib->vba.ReadBandwidthPlaneLuma[k]
1611 / 1000 / 512
1612 + mode_lib->vba.ReadBandwidthPlaneChroma[k]
1613 / 1000 / 512;
1614 }
1615 }
1616
1617 mode_lib->vba.PartOfBurstThatFitsInROB =
1618 dml_min(
1619 mode_lib->vba.MinFullDETBufferingTime
1620 * mode_lib->vba.TotalDataReadBandwidth,
1621 mode_lib->vba.ROBBufferSizeInKByte * 1024
1622 * mode_lib->vba.TotalDataReadBandwidth
1623 / (mode_lib->vba.AverageReadBandwidthGBytePerSecond
1624 * 1000));
1625 mode_lib->vba.StutterBurstTime = mode_lib->vba.PartOfBurstThatFitsInROB
1626 * (mode_lib->vba.AverageReadBandwidthGBytePerSecond * 1000)
1627 / mode_lib->vba.TotalDataReadBandwidth / mode_lib->vba.ReturnBW
1628 + (mode_lib->vba.MinFullDETBufferingTime
1629 * mode_lib->vba.TotalDataReadBandwidth
1630 - mode_lib->vba.PartOfBurstThatFitsInROB)
1631 / (mode_lib->vba.DCFCLK * 64);
1632 if (mode_lib->vba.TotalActiveWriteback == 0) {
1633 mode_lib->vba.StutterEfficiencyNotIncludingVBlank = (1
1634 - (mode_lib->vba.SRExitTime + mode_lib->vba.StutterBurstTime)
1635 / mode_lib->vba.MinFullDETBufferingTime) * 100;
1636 } else {
1637 mode_lib->vba.StutterEfficiencyNotIncludingVBlank = 0;
1638 }
1639
1640 mode_lib->vba.SmallestVBlank = 999999;
1641 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1642 if (mode_lib->vba.SynchronizedVBlank || mode_lib->vba.NumberOfActivePlanes == 1) {
1643 mode_lib->vba.VBlankTime = (double) (mode_lib->vba.VTotal[k]
1644 - mode_lib->vba.VActive[k]) * mode_lib->vba.HTotal[k]
1645 / mode_lib->vba.PixelClock[k];
1646 } else {
1647 mode_lib->vba.VBlankTime = 0;
1648 }
1649 mode_lib->vba.SmallestVBlank = dml_min(
1650 mode_lib->vba.SmallestVBlank,
1651 mode_lib->vba.VBlankTime);
1652 }
1653
1654 mode_lib->vba.StutterEfficiency = (mode_lib->vba.StutterEfficiencyNotIncludingVBlank / 100
1655 * (mode_lib->vba.FrameTimeForMinFullDETBufferingTime
1656 - mode_lib->vba.SmallestVBlank)
1657 + mode_lib->vba.SmallestVBlank)
1658 / mode_lib->vba.FrameTimeForMinFullDETBufferingTime * 100;
1659
1660 // dml_ml->vba.DCFCLK Deep Sleep
1661 mode_lib->vba.DCFCLKDeepSleep = 8.0;
1662
1663 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; k++) {
1664 if (mode_lib->vba.BytePerPixelDETC[k] > 0) {
1665 mode_lib->vba.DCFCLKDeepSleepPerPlane[k] =
1666 dml_max(
1667 1.1 * mode_lib->vba.SwathWidthY[k]
1668 * dml_ceil(
1669 mode_lib->vba.BytePerPixelDETY[k],
1670 1) / 32
1671 / mode_lib->vba.DisplayPipeLineDeliveryTimeLuma[k],
1672 1.1 * mode_lib->vba.SwathWidthY[k] / 2.0
1673 * dml_ceil(
1674 mode_lib->vba.BytePerPixelDETC[k],
1675 2) / 32
1676 / mode_lib->vba.DisplayPipeLineDeliveryTimeChroma[k]);
1677 } else
1678 mode_lib->vba.DCFCLKDeepSleepPerPlane[k] = 1.1 * mode_lib->vba.SwathWidthY[k]
1679 * dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1) / 64.0
1680 / mode_lib->vba.DisplayPipeLineDeliveryTimeLuma[k];
1681 mode_lib->vba.DCFCLKDeepSleepPerPlane[k] = dml_max(
1682 mode_lib->vba.DCFCLKDeepSleepPerPlane[k],
1683 mode_lib->vba.PixelClock[k] / 16.0);
1684 mode_lib->vba.DCFCLKDeepSleep = dml_max(
1685 mode_lib->vba.DCFCLKDeepSleep,
1686 mode_lib->vba.DCFCLKDeepSleepPerPlane[k]);
1687
1688 DTRACE(
1689 " dcfclk_deepsleep_per_plane[%i] = %fMHz",
1690 k,
1691 mode_lib->vba.DCFCLKDeepSleepPerPlane[k]);
1692 }
1693
1694 DTRACE(" dcfclk_deepsleep_mhz = %fMHz", mode_lib->vba.DCFCLKDeepSleep);
1695
1696 // Stutter Watermark
1697 mode_lib->vba.StutterExitWatermark = mode_lib->vba.SRExitTime
1698 + mode_lib->vba.LastPixelOfLineExtraWatermark
1699 + mode_lib->vba.UrgentExtraLatency + 10 / mode_lib->vba.DCFCLKDeepSleep;
1700 mode_lib->vba.StutterEnterPlusExitWatermark = mode_lib->vba.SREnterPlusExitTime
1701 + mode_lib->vba.LastPixelOfLineExtraWatermark
1702 + mode_lib->vba.UrgentExtraLatency;
1703
1704 DTRACE(" wm_cstate_exit = %fus", mode_lib->vba.StutterExitWatermark);
1705 DTRACE(" wm_cstate_enter_exit = %fus", mode_lib->vba.StutterEnterPlusExitWatermark);
1706
1707 // Urgent Latency Supported
1708 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1709 mode_lib->vba.EffectiveDETPlusLBLinesLuma =
1710 dml_floor(
1711 mode_lib->vba.LinesInDETY[k]
1712 + dml_min(
1713 mode_lib->vba.LinesInDETY[k]
1714 * mode_lib->vba.DPPCLK[k]
1715 * mode_lib->vba.BytePerPixelDETY[k]
1716 * mode_lib->vba.PSCL_THROUGHPUT_LUMA[k]
1717 / (mode_lib->vba.ReturnBW
1718 / mode_lib->vba.DPPPerPlane[k]),
1719 (double) mode_lib->vba.EffectiveLBLatencyHidingSourceLinesLuma),
1720 mode_lib->vba.SwathHeightY[k]);
1721
1722 mode_lib->vba.UrgentLatencySupportUsLuma = mode_lib->vba.EffectiveDETPlusLBLinesLuma
1723 * (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
1724 / mode_lib->vba.VRatio[k]
1725 - mode_lib->vba.EffectiveDETPlusLBLinesLuma
1726 * mode_lib->vba.SwathWidthY[k]
1727 * mode_lib->vba.BytePerPixelDETY[k]
1728 / (mode_lib->vba.ReturnBW
1729 / mode_lib->vba.DPPPerPlane[k]);
1730
1731 if (mode_lib->vba.BytePerPixelDETC[k] > 0) {
1732 mode_lib->vba.EffectiveDETPlusLBLinesChroma =
1733 dml_floor(
1734 mode_lib->vba.LinesInDETC[k]
1735 + dml_min(
1736 mode_lib->vba.LinesInDETC[k]
1737 * mode_lib->vba.DPPCLK[k]
1738 * mode_lib->vba.BytePerPixelDETC[k]
1739 * mode_lib->vba.PSCL_THROUGHPUT_CHROMA[k]
1740 / (mode_lib->vba.ReturnBW
1741 / mode_lib->vba.DPPPerPlane[k]),
1742 (double) mode_lib->vba.EffectiveLBLatencyHidingSourceLinesChroma),
1743 mode_lib->vba.SwathHeightC[k]);
1744 mode_lib->vba.UrgentLatencySupportUsChroma =
1745 mode_lib->vba.EffectiveDETPlusLBLinesChroma
1746 * (mode_lib->vba.HTotal[k]
1747 / mode_lib->vba.PixelClock[k])
1748 / (mode_lib->vba.VRatio[k] / 2)
1749 - mode_lib->vba.EffectiveDETPlusLBLinesChroma
1750 * (mode_lib->vba.SwathWidthY[k]
1751 / 2)
1752 * mode_lib->vba.BytePerPixelDETC[k]
1753 / (mode_lib->vba.ReturnBW
1754 / mode_lib->vba.DPPPerPlane[k]);
1755 mode_lib->vba.UrgentLatencySupportUs[k] = dml_min(
1756 mode_lib->vba.UrgentLatencySupportUsLuma,
1757 mode_lib->vba.UrgentLatencySupportUsChroma);
1758 } else {
1759 mode_lib->vba.UrgentLatencySupportUs[k] =
1760 mode_lib->vba.UrgentLatencySupportUsLuma;
1761 }
1762 }
1763
1764 mode_lib->vba.MinUrgentLatencySupportUs = 999999;
1765 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1766 mode_lib->vba.MinUrgentLatencySupportUs = dml_min(
1767 mode_lib->vba.MinUrgentLatencySupportUs,
1768 mode_lib->vba.UrgentLatencySupportUs[k]);
1769 }
1770
1771 // Non-Urgent Latency Tolerance
1772 mode_lib->vba.NonUrgentLatencyTolerance = mode_lib->vba.MinUrgentLatencySupportUs
1773 - mode_lib->vba.UrgentWatermark;
1774
1775 // DSCCLK
1776 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1777 if ((mode_lib->vba.BlendingAndTiming[k] != k) || !mode_lib->vba.DSCEnabled[k]) {
1778 mode_lib->vba.DSCCLK_calculated[k] = 0.0;
1779 } else {
1780 if (mode_lib->vba.OutputFormat[k] == dm_420
1781 || mode_lib->vba.OutputFormat[k] == dm_n422)
1782 mode_lib->vba.DSCFormatFactor = 2;
1783 else
1784 mode_lib->vba.DSCFormatFactor = 1;
1785 if (mode_lib->vba.ODMCombineEnabled[k])
1786 mode_lib->vba.DSCCLK_calculated[k] =
1787 mode_lib->vba.PixelClockBackEnd[k] / 6
1788 / mode_lib->vba.DSCFormatFactor
1789 / (1
1790 - mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1791 / 100);
1792 else
1793 mode_lib->vba.DSCCLK_calculated[k] =
1794 mode_lib->vba.PixelClockBackEnd[k] / 3
1795 / mode_lib->vba.DSCFormatFactor
1796 / (1
1797 - mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading
1798 / 100);
1799 }
1800 }
1801
1802 // DSC Delay
1803 // TODO
1804 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1805 double bpp = mode_lib->vba.OutputBpp[k];
1806 unsigned int slices = mode_lib->vba.NumberOfDSCSlices[k];
1807
1808 if (mode_lib->vba.DSCEnabled[k] && bpp != 0) {
1809 if (!mode_lib->vba.ODMCombineEnabled[k]) {
1810 mode_lib->vba.DSCDelay[k] =
1811 dscceComputeDelay(
1812 mode_lib->vba.DSCInputBitPerComponent[k],
1813 bpp,
1814 dml_ceil(
1815 (double) mode_lib->vba.HActive[k]
1816 / mode_lib->vba.NumberOfDSCSlices[k],
1817 1),
1818 slices,
1819 mode_lib->vba.OutputFormat[k])
1820 + dscComputeDelay(
1821 mode_lib->vba.OutputFormat[k]);
1822 } else {
1823 mode_lib->vba.DSCDelay[k] =
1824 2
1825 * (dscceComputeDelay(
1826 mode_lib->vba.DSCInputBitPerComponent[k],
1827 bpp,
1828 dml_ceil(
1829 (double) mode_lib->vba.HActive[k]
1830 / mode_lib->vba.NumberOfDSCSlices[k],
1831 1),
1832 slices / 2.0,
1833 mode_lib->vba.OutputFormat[k])
1834 + dscComputeDelay(
1835 mode_lib->vba.OutputFormat[k]));
1836 }
1837 mode_lib->vba.DSCDelay[k] = mode_lib->vba.DSCDelay[k]
1838 * mode_lib->vba.PixelClock[k]
1839 / mode_lib->vba.PixelClockBackEnd[k];
1840 } else {
1841 mode_lib->vba.DSCDelay[k] = 0;
1842 }
1843 }
1844
1845 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k)
1846 for (j = 0; j < mode_lib->vba.NumberOfActivePlanes; ++j) // NumberOfPlanes
1847 if (j != k && mode_lib->vba.BlendingAndTiming[k] == j
1848 && mode_lib->vba.DSCEnabled[j])
1849 mode_lib->vba.DSCDelay[k] = mode_lib->vba.DSCDelay[j];
1850
1851 // Prefetch
1852 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1853 unsigned int PDEAndMetaPTEBytesFrameY;
1854 unsigned int PixelPTEBytesPerRowY;
1855 unsigned int MetaRowByteY;
1856 unsigned int MetaRowByteC;
1857 unsigned int PDEAndMetaPTEBytesFrameC;
1858 unsigned int PixelPTEBytesPerRowC;
1859
1860 Calculate256BBlockSizes(
1861 mode_lib->vba.SourcePixelFormat[k],
1862 mode_lib->vba.SurfaceTiling[k],
1863 dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1),
1864 dml_ceil(mode_lib->vba.BytePerPixelDETC[k], 2),
1865 &mode_lib->vba.BlockHeight256BytesY[k],
1866 &mode_lib->vba.BlockHeight256BytesC[k],
1867 &mode_lib->vba.BlockWidth256BytesY[k],
1868 &mode_lib->vba.BlockWidth256BytesC[k]);
1869 PDEAndMetaPTEBytesFrameY = CalculateVMAndRowBytes(
1870 mode_lib,
1871 mode_lib->vba.DCCEnable[k],
1872 mode_lib->vba.BlockHeight256BytesY[k],
1873 mode_lib->vba.BlockWidth256BytesY[k],
1874 mode_lib->vba.SourcePixelFormat[k],
1875 mode_lib->vba.SurfaceTiling[k],
1876 dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1),
1877 mode_lib->vba.SourceScan[k],
1878 mode_lib->vba.ViewportWidth[k],
1879 mode_lib->vba.ViewportHeight[k],
1880 mode_lib->vba.SwathWidthY[k],
1881 mode_lib->vba.GPUVMEnable,
1882 mode_lib->vba.VMMPageSize,
1883 mode_lib->vba.PTEBufferSizeInRequestsLuma,
1884 mode_lib->vba.PDEProcessingBufIn64KBReqs,
1885 mode_lib->vba.PitchY[k],
1886 mode_lib->vba.DCCMetaPitchY[k],
1887 &mode_lib->vba.MacroTileWidthY[k],
1888 &MetaRowByteY,
1889 &PixelPTEBytesPerRowY,
1890 &mode_lib->vba.PTEBufferSizeNotExceeded[mode_lib->vba.VoltageLevel][0],
1891 &mode_lib->vba.dpte_row_height[k],
1892 &mode_lib->vba.meta_row_height[k]);
1893 mode_lib->vba.PrefetchSourceLinesY[k] = CalculatePrefetchSourceLines(
1894 mode_lib,
1895 mode_lib->vba.VRatio[k],
1896 mode_lib->vba.vtaps[k],
1897 mode_lib->vba.Interlace[k],
1898 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
1899 mode_lib->vba.SwathHeightY[k],
1900 mode_lib->vba.ViewportYStartY[k],
1901 &mode_lib->vba.VInitPreFillY[k],
1902 &mode_lib->vba.MaxNumSwathY[k]);
1903
1904 if ((mode_lib->vba.SourcePixelFormat[k] != dm_444_64
1905 && mode_lib->vba.SourcePixelFormat[k] != dm_444_32
1906 && mode_lib->vba.SourcePixelFormat[k] != dm_444_16
1907 && mode_lib->vba.SourcePixelFormat[k] != dm_444_8)) {
1908 PDEAndMetaPTEBytesFrameC =
1909 CalculateVMAndRowBytes(
1910 mode_lib,
1911 mode_lib->vba.DCCEnable[k],
1912 mode_lib->vba.BlockHeight256BytesC[k],
1913 mode_lib->vba.BlockWidth256BytesC[k],
1914 mode_lib->vba.SourcePixelFormat[k],
1915 mode_lib->vba.SurfaceTiling[k],
1916 dml_ceil(
1917 mode_lib->vba.BytePerPixelDETC[k],
1918 2),
1919 mode_lib->vba.SourceScan[k],
1920 mode_lib->vba.ViewportWidth[k] / 2,
1921 mode_lib->vba.ViewportHeight[k] / 2,
1922 mode_lib->vba.SwathWidthY[k] / 2,
1923 mode_lib->vba.GPUVMEnable,
1924 mode_lib->vba.VMMPageSize,
1925 mode_lib->vba.PTEBufferSizeInRequestsLuma,
1926 mode_lib->vba.PDEProcessingBufIn64KBReqs,
1927 mode_lib->vba.PitchC[k],
1928 0,
1929 &mode_lib->vba.MacroTileWidthC[k],
1930 &MetaRowByteC,
1931 &PixelPTEBytesPerRowC,
1932 &mode_lib->vba.PTEBufferSizeNotExceeded[mode_lib->vba.VoltageLevel][0],
1933 &mode_lib->vba.dpte_row_height_chroma[k],
1934 &mode_lib->vba.meta_row_height_chroma[k]);
1935 mode_lib->vba.PrefetchSourceLinesC[k] = CalculatePrefetchSourceLines(
1936 mode_lib,
1937 mode_lib->vba.VRatio[k] / 2,
1938 mode_lib->vba.VTAPsChroma[k],
1939 mode_lib->vba.Interlace[k],
1940 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
1941 mode_lib->vba.SwathHeightC[k],
1942 mode_lib->vba.ViewportYStartC[k],
1943 &mode_lib->vba.VInitPreFillC[k],
1944 &mode_lib->vba.MaxNumSwathC[k]);
1945 } else {
1946 PixelPTEBytesPerRowC = 0;
1947 PDEAndMetaPTEBytesFrameC = 0;
1948 MetaRowByteC = 0;
1949 mode_lib->vba.MaxNumSwathC[k] = 0;
1950 mode_lib->vba.PrefetchSourceLinesC[k] = 0;
1951 }
1952
1953 mode_lib->vba.PixelPTEBytesPerRow[k] = PixelPTEBytesPerRowY + PixelPTEBytesPerRowC;
1954 mode_lib->vba.PDEAndMetaPTEBytesFrame[k] = PDEAndMetaPTEBytesFrameY
1955 + PDEAndMetaPTEBytesFrameC;
1956 mode_lib->vba.MetaRowByte[k] = MetaRowByteY + MetaRowByteC;
1957
1958 CalculateActiveRowBandwidth(
1959 mode_lib->vba.GPUVMEnable,
1960 mode_lib->vba.SourcePixelFormat[k],
1961 mode_lib->vba.VRatio[k],
1962 mode_lib->vba.DCCEnable[k],
1963 mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k],
1964 MetaRowByteY,
1965 MetaRowByteC,
1966 mode_lib->vba.meta_row_height[k],
1967 mode_lib->vba.meta_row_height_chroma[k],
1968 PixelPTEBytesPerRowY,
1969 PixelPTEBytesPerRowC,
1970 mode_lib->vba.dpte_row_height[k],
1971 mode_lib->vba.dpte_row_height_chroma[k],
1972 &mode_lib->vba.meta_row_bw[k],
1973 &mode_lib->vba.dpte_row_bw[k],
1974 &mode_lib->vba.qual_row_bw[k]);
1975 }
1976
1977 mode_lib->vba.TCalc = 24.0 / mode_lib->vba.DCFCLKDeepSleep;
1978
1979 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
1980 if (mode_lib->vba.BlendingAndTiming[k] == k) {
1981 if (mode_lib->vba.WritebackEnable[k] == true) {
1982 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k] =
1983 mode_lib->vba.WritebackLatency
1984 + CalculateWriteBackDelay(
1985 mode_lib->vba.WritebackPixelFormat[k],
1986 mode_lib->vba.WritebackHRatio[k],
1987 mode_lib->vba.WritebackVRatio[k],
1988 mode_lib->vba.WritebackLumaHTaps[k],
1989 mode_lib->vba.WritebackLumaVTaps[k],
1990 mode_lib->vba.WritebackChromaHTaps[k],
1991 mode_lib->vba.WritebackChromaVTaps[k],
1992 mode_lib->vba.WritebackDestinationWidth[k])
1993 / mode_lib->vba.DISPCLK;
1994 } else
1995 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k] = 0;
1996 for (j = 0; j < mode_lib->vba.NumberOfActivePlanes; ++j) {
1997 if (mode_lib->vba.BlendingAndTiming[j] == k
1998 && mode_lib->vba.WritebackEnable[j] == true) {
1999 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k] =
2000 dml_max(
2001 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k],
2002 mode_lib->vba.WritebackLatency
2003 + CalculateWriteBackDelay(
2004 mode_lib->vba.WritebackPixelFormat[j],
2005 mode_lib->vba.WritebackHRatio[j],
2006 mode_lib->vba.WritebackVRatio[j],
2007 mode_lib->vba.WritebackLumaHTaps[j],
2008 mode_lib->vba.WritebackLumaVTaps[j],
2009 mode_lib->vba.WritebackChromaHTaps[j],
2010 mode_lib->vba.WritebackChromaVTaps[j],
2011 mode_lib->vba.WritebackDestinationWidth[j])
2012 / mode_lib->vba.DISPCLK);
2013 }
2014 }
2015 }
2016 }
2017
2018 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k)
2019 for (j = 0; j < mode_lib->vba.NumberOfActivePlanes; ++j)
2020 if (mode_lib->vba.BlendingAndTiming[k] == j)
2021 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k] =
2022 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][j];
2023
2024 mode_lib->vba.VStartupLines = 13;
2025 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2026 mode_lib->vba.MaxVStartupLines[k] =
2027 mode_lib->vba.VTotal[k] - mode_lib->vba.VActive[k]
2028 - dml_max(
2029 1.0,
2030 dml_ceil(
2031 mode_lib->vba.WritebackDelay[mode_lib->vba.VoltageLevel][k]
2032 / (mode_lib->vba.HTotal[k]
2033 / mode_lib->vba.PixelClock[k]),
2034 1));
2035 }
2036
2037 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k)
2038 mode_lib->vba.MaximumMaxVStartupLines = dml_max(
2039 mode_lib->vba.MaximumMaxVStartupLines,
2040 mode_lib->vba.MaxVStartupLines[k]);
2041
2042 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2043 mode_lib->vba.cursor_bw[k] = 0.0;
2044 for (j = 0; j < mode_lib->vba.NumberOfCursors[k]; ++j)
2045 mode_lib->vba.cursor_bw[k] += mode_lib->vba.CursorWidth[k][j]
2046 * mode_lib->vba.CursorBPP[k][j] / 8.0
2047 / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
2048 * mode_lib->vba.VRatio[k];
2049 }
2050
2051 do {
2052 double MaxTotalRDBandwidth = 0;
2053 bool DestinationLineTimesForPrefetchLessThan2 = false;
2054 bool VRatioPrefetchMoreThan4 = false;
2055 bool prefetch_vm_bw_valid = true;
2056 bool prefetch_row_bw_valid = true;
2057 double TWait = CalculateTWait(
2058 mode_lib->vba.PrefetchMode[mode_lib->vba.VoltageLevel][mode_lib->vba.maxMpcComb],
2059 mode_lib->vba.DRAMClockChangeLatency,
2060 mode_lib->vba.UrgentLatencyPixelDataOnly,
2061 mode_lib->vba.SREnterPlusExitTime);
2062
2063 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2064 if (mode_lib->vba.XFCEnabled[k] == true) {
2065 mode_lib->vba.XFCRemoteSurfaceFlipDelay =
2066 CalculateRemoteSurfaceFlipDelay(
2067 mode_lib,
2068 mode_lib->vba.VRatio[k],
2069 mode_lib->vba.SwathWidthY[k],
2070 dml_ceil(
2071 mode_lib->vba.BytePerPixelDETY[k],
2072 1),
2073 mode_lib->vba.HTotal[k]
2074 / mode_lib->vba.PixelClock[k],
2075 mode_lib->vba.XFCTSlvVupdateOffset,
2076 mode_lib->vba.XFCTSlvVupdateWidth,
2077 mode_lib->vba.XFCTSlvVreadyOffset,
2078 mode_lib->vba.XFCXBUFLatencyTolerance,
2079 mode_lib->vba.XFCFillBWOverhead,
2080 mode_lib->vba.XFCSlvChunkSize,
2081 mode_lib->vba.XFCBusTransportTime,
2082 mode_lib->vba.TCalc,
2083 TWait,
2084 &mode_lib->vba.SrcActiveDrainRate,
2085 &mode_lib->vba.TInitXFill,
2086 &mode_lib->vba.TslvChk);
2087 } else {
2088 mode_lib->vba.XFCRemoteSurfaceFlipDelay = 0;
2089 }
2090 mode_lib->vba.ErrorResult[k] =
2091 CalculatePrefetchSchedule(
2092 mode_lib,
2093 mode_lib->vba.DPPCLK[k],
2094 mode_lib->vba.DISPCLK,
2095 mode_lib->vba.PixelClock[k],
2096 mode_lib->vba.DCFCLKDeepSleep,
2097 mode_lib->vba.DSCDelay[k],
2098 mode_lib->vba.DPPPerPlane[k],
2099 mode_lib->vba.ScalerEnabled[k],
2100 mode_lib->vba.NumberOfCursors[k],
2101 mode_lib->vba.DPPCLKDelaySubtotal,
2102 mode_lib->vba.DPPCLKDelaySCL,
2103 mode_lib->vba.DPPCLKDelaySCLLBOnly,
2104 mode_lib->vba.DPPCLKDelayCNVCFormater,
2105 mode_lib->vba.DPPCLKDelayCNVCCursor,
2106 mode_lib->vba.DISPCLKDelaySubtotal,
2107 (unsigned int) (mode_lib->vba.SwathWidthY[k]
2108 / mode_lib->vba.HRatio[k]),
2109 mode_lib->vba.OutputFormat[k],
2110 mode_lib->vba.VTotal[k]
2111 - mode_lib->vba.VActive[k],
2112 mode_lib->vba.HTotal[k],
2113 mode_lib->vba.MaxInterDCNTileRepeaters,
2114 dml_min(
2115 mode_lib->vba.VStartupLines,
2116 mode_lib->vba.MaxVStartupLines[k]),
2117 mode_lib->vba.GPUVMMaxPageTableLevels,
2118 mode_lib->vba.GPUVMEnable,
2119 mode_lib->vba.DynamicMetadataEnable[k],
2120 mode_lib->vba.DynamicMetadataLinesBeforeActiveRequired[k],
2121 mode_lib->vba.DynamicMetadataTransmittedBytes[k],
2122 mode_lib->vba.DCCEnable[k],
2123 mode_lib->vba.UrgentLatencyPixelDataOnly,
2124 mode_lib->vba.UrgentExtraLatency,
2125 mode_lib->vba.TCalc,
2126 mode_lib->vba.PDEAndMetaPTEBytesFrame[k],
2127 mode_lib->vba.MetaRowByte[k],
2128 mode_lib->vba.PixelPTEBytesPerRow[k],
2129 mode_lib->vba.PrefetchSourceLinesY[k],
2130 mode_lib->vba.SwathWidthY[k],
2131 mode_lib->vba.BytePerPixelDETY[k],
2132 mode_lib->vba.VInitPreFillY[k],
2133 mode_lib->vba.MaxNumSwathY[k],
2134 mode_lib->vba.PrefetchSourceLinesC[k],
2135 mode_lib->vba.BytePerPixelDETC[k],
2136 mode_lib->vba.VInitPreFillC[k],
2137 mode_lib->vba.MaxNumSwathC[k],
2138 mode_lib->vba.SwathHeightY[k],
2139 mode_lib->vba.SwathHeightC[k],
2140 TWait,
2141 mode_lib->vba.XFCEnabled[k],
2142 mode_lib->vba.XFCRemoteSurfaceFlipDelay,
2143 mode_lib->vba.Interlace[k],
2144 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
2145 &mode_lib->vba.DSTXAfterScaler[k],
2146 &mode_lib->vba.DSTYAfterScaler[k],
2147 &mode_lib->vba.DestinationLinesForPrefetch[k],
2148 &mode_lib->vba.PrefetchBandwidth[k],
2149 &mode_lib->vba.DestinationLinesToRequestVMInVBlank[k],
2150 &mode_lib->vba.DestinationLinesToRequestRowInVBlank[k],
2151 &mode_lib->vba.VRatioPrefetchY[k],
2152 &mode_lib->vba.VRatioPrefetchC[k],
2153 &mode_lib->vba.RequiredPrefetchPixDataBWLuma[k],
2154 &mode_lib->vba.VStartupRequiredWhenNotEnoughTimeForDynamicMetadata,
2155 &mode_lib->vba.Tno_bw[k],
2156 &mode_lib->vba.VUpdateOffsetPix[k],
2157 &mode_lib->vba.VUpdateWidthPix[k],
2158 &mode_lib->vba.VReadyOffsetPix[k]);
2159 if (mode_lib->vba.BlendingAndTiming[k] == k) {
2160 mode_lib->vba.VStartup[k] = dml_min(
2161 mode_lib->vba.VStartupLines,
2162 mode_lib->vba.MaxVStartupLines[k]);
2163 if (mode_lib->vba.VStartupRequiredWhenNotEnoughTimeForDynamicMetadata
2164 != 0) {
2165 mode_lib->vba.VStartup[k] =
2166 mode_lib->vba.VStartupRequiredWhenNotEnoughTimeForDynamicMetadata;
2167 }
2168 } else {
2169 mode_lib->vba.VStartup[k] =
2170 dml_min(
2171 mode_lib->vba.VStartupLines,
2172 mode_lib->vba.MaxVStartupLines[mode_lib->vba.BlendingAndTiming[k]]);
2173 }
2174 }
2175
2176 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2177
2178 if (mode_lib->vba.PDEAndMetaPTEBytesFrame[k] == 0)
2179 mode_lib->vba.prefetch_vm_bw[k] = 0;
2180 else if (mode_lib->vba.DestinationLinesToRequestVMInVBlank[k] > 0) {
2181 mode_lib->vba.prefetch_vm_bw[k] =
2182 (double) mode_lib->vba.PDEAndMetaPTEBytesFrame[k]
2183 / (mode_lib->vba.DestinationLinesToRequestVMInVBlank[k]
2184 * mode_lib->vba.HTotal[k]
2185 / mode_lib->vba.PixelClock[k]);
2186 } else {
2187 mode_lib->vba.prefetch_vm_bw[k] = 0;
2188 prefetch_vm_bw_valid = false;
2189 }
2190 if (mode_lib->vba.MetaRowByte[k] + mode_lib->vba.PixelPTEBytesPerRow[k]
2191 == 0)
2192 mode_lib->vba.prefetch_row_bw[k] = 0;
2193 else if (mode_lib->vba.DestinationLinesToRequestRowInVBlank[k] > 0) {
2194 mode_lib->vba.prefetch_row_bw[k] =
2195 (double) (mode_lib->vba.MetaRowByte[k]
2196 + mode_lib->vba.PixelPTEBytesPerRow[k])
2197 / (mode_lib->vba.DestinationLinesToRequestRowInVBlank[k]
2198 * mode_lib->vba.HTotal[k]
2199 / mode_lib->vba.PixelClock[k]);
2200 } else {
2201 mode_lib->vba.prefetch_row_bw[k] = 0;
2202 prefetch_row_bw_valid = false;
2203 }
2204
2205 MaxTotalRDBandwidth =
2206 MaxTotalRDBandwidth + mode_lib->vba.cursor_bw[k]
2207 + dml_max(
2208 mode_lib->vba.prefetch_vm_bw[k],
2209 dml_max(
2210 mode_lib->vba.prefetch_row_bw[k],
2211 dml_max(
2212 mode_lib->vba.ReadBandwidthPlaneLuma[k]
2213 + mode_lib->vba.ReadBandwidthPlaneChroma[k],
2214 mode_lib->vba.RequiredPrefetchPixDataBWLuma[k])
2215 + mode_lib->vba.meta_row_bw[k]
2216 + mode_lib->vba.dpte_row_bw[k]));
2217
2218 if (mode_lib->vba.DestinationLinesForPrefetch[k] < 2)
2219 DestinationLineTimesForPrefetchLessThan2 = true;
2220 if (mode_lib->vba.VRatioPrefetchY[k] > 4
2221 || mode_lib->vba.VRatioPrefetchC[k] > 4)
2222 VRatioPrefetchMoreThan4 = true;
2223 }
2224
2225 if (MaxTotalRDBandwidth <= mode_lib->vba.ReturnBW && prefetch_vm_bw_valid
2226 && prefetch_row_bw_valid && !VRatioPrefetchMoreThan4
2227 && !DestinationLineTimesForPrefetchLessThan2)
2228 mode_lib->vba.PrefetchModeSupported = true;
2229 else {
2230 mode_lib->vba.PrefetchModeSupported = false;
2231 dml_print(
2232 "DML: CalculatePrefetchSchedule ***failed***. Bandwidth violation. Results are NOT valid\n");
2233 }
2234
2235 if (mode_lib->vba.PrefetchModeSupported == true) {
2236 double final_flip_bw[DC__NUM_DPP__MAX];
2237 unsigned int ImmediateFlipBytes[DC__NUM_DPP__MAX];
2238 double total_dcn_read_bw_with_flip = 0;
2239
2240 mode_lib->vba.BandwidthAvailableForImmediateFlip = mode_lib->vba.ReturnBW;
2241 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2242 mode_lib->vba.BandwidthAvailableForImmediateFlip =
2243 mode_lib->vba.BandwidthAvailableForImmediateFlip
2244 - mode_lib->vba.cursor_bw[k]
2245 - dml_max(
2246 mode_lib->vba.ReadBandwidthPlaneLuma[k]
2247 + mode_lib->vba.ReadBandwidthPlaneChroma[k]
2248 + mode_lib->vba.qual_row_bw[k],
2249 mode_lib->vba.PrefetchBandwidth[k]);
2250 }
2251
2252 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2253 ImmediateFlipBytes[k] = 0;
2254 if ((mode_lib->vba.SourcePixelFormat[k] != dm_420_8
2255 && mode_lib->vba.SourcePixelFormat[k] != dm_420_10)) {
2256 ImmediateFlipBytes[k] =
2257 mode_lib->vba.PDEAndMetaPTEBytesFrame[k]
2258 + mode_lib->vba.MetaRowByte[k]
2259 + mode_lib->vba.PixelPTEBytesPerRow[k];
2260 }
2261 }
2262 mode_lib->vba.TotImmediateFlipBytes = 0;
2263 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2264 if ((mode_lib->vba.SourcePixelFormat[k] != dm_420_8
2265 && mode_lib->vba.SourcePixelFormat[k] != dm_420_10)) {
2266 mode_lib->vba.TotImmediateFlipBytes =
2267 mode_lib->vba.TotImmediateFlipBytes
2268 + ImmediateFlipBytes[k];
2269 }
2270 }
2271 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2272 CalculateFlipSchedule(
2273 mode_lib,
2274 mode_lib->vba.UrgentExtraLatency,
2275 mode_lib->vba.UrgentLatencyPixelDataOnly,
2276 mode_lib->vba.GPUVMMaxPageTableLevels,
2277 mode_lib->vba.GPUVMEnable,
2278 mode_lib->vba.BandwidthAvailableForImmediateFlip,
2279 mode_lib->vba.TotImmediateFlipBytes,
2280 mode_lib->vba.SourcePixelFormat[k],
2281 ImmediateFlipBytes[k],
2282 mode_lib->vba.HTotal[k]
2283 / mode_lib->vba.PixelClock[k],
2284 mode_lib->vba.VRatio[k],
2285 mode_lib->vba.Tno_bw[k],
2286 mode_lib->vba.PDEAndMetaPTEBytesFrame[k],
2287 mode_lib->vba.MetaRowByte[k],
2288 mode_lib->vba.PixelPTEBytesPerRow[k],
2289 mode_lib->vba.DCCEnable[k],
2290 mode_lib->vba.dpte_row_height[k],
2291 mode_lib->vba.meta_row_height[k],
2292 mode_lib->vba.qual_row_bw[k],
2293 &mode_lib->vba.DestinationLinesToRequestVMInImmediateFlip[k],
2294 &mode_lib->vba.DestinationLinesToRequestRowInImmediateFlip[k],
2295 &final_flip_bw[k],
2296 &mode_lib->vba.ImmediateFlipSupportedForPipe[k]);
2297 }
2298 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2299 total_dcn_read_bw_with_flip =
2300 total_dcn_read_bw_with_flip
2301 + mode_lib->vba.cursor_bw[k]
2302 + dml_max(
2303 mode_lib->vba.prefetch_vm_bw[k],
2304 dml_max(
2305 mode_lib->vba.prefetch_row_bw[k],
2306 final_flip_bw[k]
2307 + dml_max(
2308 mode_lib->vba.ReadBandwidthPlaneLuma[k]
2309 + mode_lib->vba.ReadBandwidthPlaneChroma[k],
2310 mode_lib->vba.RequiredPrefetchPixDataBWLuma[k])));
2311 }
2312 mode_lib->vba.ImmediateFlipSupported = true;
2313 if (total_dcn_read_bw_with_flip > mode_lib->vba.ReturnBW) {
2314 mode_lib->vba.ImmediateFlipSupported = false;
2315 }
2316 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2317 if (mode_lib->vba.ImmediateFlipSupportedForPipe[k] == false) {
2318 mode_lib->vba.ImmediateFlipSupported = false;
2319 }
2320 }
2321 } else {
2322 mode_lib->vba.ImmediateFlipSupported = false;
2323 }
2324
2325 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2326 if (mode_lib->vba.ErrorResult[k]) {
2327 mode_lib->vba.PrefetchModeSupported = false;
2328 dml_print(
2329 "DML: CalculatePrefetchSchedule ***failed***. Prefetch schedule violation. Results are NOT valid\n");
2330 }
2331 }
2332
2333 mode_lib->vba.VStartupLines = mode_lib->vba.VStartupLines + 1;
2334 } while (!((mode_lib->vba.PrefetchModeSupported
2335 && (!mode_lib->vba.ImmediateFlipSupport
2336 || mode_lib->vba.ImmediateFlipSupported))
2337 || mode_lib->vba.MaximumMaxVStartupLines < mode_lib->vba.VStartupLines));
2338
2339 //Display Pipeline Delivery Time in Prefetch
2340 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2341 if (mode_lib->vba.VRatioPrefetchY[k] <= 1) {
2342 mode_lib->vba.DisplayPipeLineDeliveryTimeLumaPrefetch[k] =
2343 mode_lib->vba.SwathWidthY[k] * mode_lib->vba.DPPPerPlane[k]
2344 / mode_lib->vba.HRatio[k]
2345 / mode_lib->vba.PixelClock[k];
2346 } else {
2347 mode_lib->vba.DisplayPipeLineDeliveryTimeLumaPrefetch[k] =
2348 mode_lib->vba.SwathWidthY[k]
2349 / mode_lib->vba.PSCL_THROUGHPUT_LUMA[k]
2350 / mode_lib->vba.DPPCLK[k];
2351 }
2352 if (mode_lib->vba.BytePerPixelDETC[k] == 0) {
2353 mode_lib->vba.DisplayPipeLineDeliveryTimeChromaPrefetch[k] = 0;
2354 } else {
2355 if (mode_lib->vba.VRatioPrefetchC[k] <= 1) {
2356 mode_lib->vba.DisplayPipeLineDeliveryTimeChromaPrefetch[k] =
2357 mode_lib->vba.SwathWidthY[k]
2358 * mode_lib->vba.DPPPerPlane[k]
2359 / mode_lib->vba.HRatio[k]
2360 / mode_lib->vba.PixelClock[k];
2361 } else {
2362 mode_lib->vba.DisplayPipeLineDeliveryTimeChromaPrefetch[k] =
2363 mode_lib->vba.SwathWidthY[k]
2364 / mode_lib->vba.PSCL_THROUGHPUT_LUMA[k]
2365 / mode_lib->vba.DPPCLK[k];
2366 }
2367 }
2368 }
2369
2370 // Min TTUVBlank
2371 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2372 if (mode_lib->vba.PrefetchMode[mode_lib->vba.VoltageLevel][mode_lib->vba.maxMpcComb] == 0) {
2373 mode_lib->vba.AllowDRAMClockChangeDuringVBlank[k] = true;
2374 mode_lib->vba.AllowDRAMSelfRefreshDuringVBlank[k] = true;
2375 mode_lib->vba.MinTTUVBlank[k] = dml_max(
2376 mode_lib->vba.DRAMClockChangeWatermark,
2377 dml_max(
2378 mode_lib->vba.StutterEnterPlusExitWatermark,
2379 mode_lib->vba.UrgentWatermark));
2380 } else if (mode_lib->vba.PrefetchMode[mode_lib->vba.VoltageLevel][mode_lib->vba.maxMpcComb] == 1) {
2381 mode_lib->vba.AllowDRAMClockChangeDuringVBlank[k] = false;
2382 mode_lib->vba.AllowDRAMSelfRefreshDuringVBlank[k] = true;
2383 mode_lib->vba.MinTTUVBlank[k] = dml_max(
2384 mode_lib->vba.StutterEnterPlusExitWatermark,
2385 mode_lib->vba.UrgentWatermark);
2386 } else {
2387 mode_lib->vba.AllowDRAMClockChangeDuringVBlank[k] = false;
2388 mode_lib->vba.AllowDRAMSelfRefreshDuringVBlank[k] = false;
2389 mode_lib->vba.MinTTUVBlank[k] = mode_lib->vba.UrgentWatermark;
2390 }
2391 if (!mode_lib->vba.DynamicMetadataEnable[k])
2392 mode_lib->vba.MinTTUVBlank[k] = mode_lib->vba.TCalc
2393 + mode_lib->vba.MinTTUVBlank[k];
2394 }
2395
2396 // DCC Configuration
2397 mode_lib->vba.ActiveDPPs = 0;
2398 // NB P-State/DRAM Clock Change Support
2399 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2400 mode_lib->vba.ActiveDPPs = mode_lib->vba.ActiveDPPs + mode_lib->vba.DPPPerPlane[k];
2401 }
2402
2403 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2404 double EffectiveLBLatencyHidingY;
2405 double EffectiveLBLatencyHidingC;
2406 double DPPOutputBufferLinesY;
2407 double DPPOutputBufferLinesC;
2408 double DPPOPPBufferingY;
2409 double MaxDETBufferingTimeY;
2410 double ActiveDRAMClockChangeLatencyMarginY;
2411
2412 mode_lib->vba.LBLatencyHidingSourceLinesY =
2413 dml_min(
2414 mode_lib->vba.MaxLineBufferLines,
2415 (unsigned int) dml_floor(
2416 (double) mode_lib->vba.LineBufferSize
2417 / mode_lib->vba.LBBitPerPixel[k]
2418 / (mode_lib->vba.SwathWidthY[k]
2419 / dml_max(
2420 mode_lib->vba.HRatio[k],
2421 1.0)),
2422 1)) - (mode_lib->vba.vtaps[k] - 1);
2423
2424 mode_lib->vba.LBLatencyHidingSourceLinesC =
2425 dml_min(
2426 mode_lib->vba.MaxLineBufferLines,
2427 (unsigned int) dml_floor(
2428 (double) mode_lib->vba.LineBufferSize
2429 / mode_lib->vba.LBBitPerPixel[k]
2430 / (mode_lib->vba.SwathWidthY[k]
2431 / 2.0
2432 / dml_max(
2433 mode_lib->vba.HRatio[k]
2434 / 2,
2435 1.0)),
2436 1))
2437 - (mode_lib->vba.VTAPsChroma[k] - 1);
2438
2439 EffectiveLBLatencyHidingY = mode_lib->vba.LBLatencyHidingSourceLinesY
2440 / mode_lib->vba.VRatio[k]
2441 * (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k]);
2442
2443 EffectiveLBLatencyHidingC = mode_lib->vba.LBLatencyHidingSourceLinesC
2444 / (mode_lib->vba.VRatio[k] / 2)
2445 * (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k]);
2446
2447 if (mode_lib->vba.SwathWidthY[k] > 2 * mode_lib->vba.DPPOutputBufferPixels) {
2448 DPPOutputBufferLinesY = mode_lib->vba.DPPOutputBufferPixels
2449 / mode_lib->vba.SwathWidthY[k];
2450 } else if (mode_lib->vba.SwathWidthY[k] > mode_lib->vba.DPPOutputBufferPixels) {
2451 DPPOutputBufferLinesY = 0.5;
2452 } else {
2453 DPPOutputBufferLinesY = 1;
2454 }
2455
2456 if (mode_lib->vba.SwathWidthY[k] / 2 > 2 * mode_lib->vba.DPPOutputBufferPixels) {
2457 DPPOutputBufferLinesC = mode_lib->vba.DPPOutputBufferPixels
2458 / (mode_lib->vba.SwathWidthY[k] / 2);
2459 } else if (mode_lib->vba.SwathWidthY[k] / 2 > mode_lib->vba.DPPOutputBufferPixels) {
2460 DPPOutputBufferLinesC = 0.5;
2461 } else {
2462 DPPOutputBufferLinesC = 1;
2463 }
2464
2465 DPPOPPBufferingY = (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
2466 * (DPPOutputBufferLinesY + mode_lib->vba.OPPOutputBufferLines);
2467 MaxDETBufferingTimeY = mode_lib->vba.FullDETBufferingTimeY[k]
2468 + (mode_lib->vba.LinesInDETY[k]
2469 - mode_lib->vba.LinesInDETYRoundedDownToSwath[k])
2470 / mode_lib->vba.SwathHeightY[k]
2471 * (mode_lib->vba.HTotal[k]
2472 / mode_lib->vba.PixelClock[k]);
2473
2474 ActiveDRAMClockChangeLatencyMarginY = DPPOPPBufferingY + EffectiveLBLatencyHidingY
2475 + MaxDETBufferingTimeY - mode_lib->vba.DRAMClockChangeWatermark;
2476
2477 if (mode_lib->vba.ActiveDPPs > 1) {
2478 ActiveDRAMClockChangeLatencyMarginY =
2479 ActiveDRAMClockChangeLatencyMarginY
2480 - (1 - 1 / (mode_lib->vba.ActiveDPPs - 1))
2481 * mode_lib->vba.SwathHeightY[k]
2482 * (mode_lib->vba.HTotal[k]
2483 / mode_lib->vba.PixelClock[k]);
2484 }
2485
2486 if (mode_lib->vba.BytePerPixelDETC[k] > 0) {
2487 double DPPOPPBufferingC = (mode_lib->vba.HTotal[k]
2488 / mode_lib->vba.PixelClock[k])
2489 * (DPPOutputBufferLinesC
2490 + mode_lib->vba.OPPOutputBufferLines);
2491 double MaxDETBufferingTimeC =
2492 mode_lib->vba.FullDETBufferingTimeC[k]
2493 + (mode_lib->vba.LinesInDETC[k]
2494 - mode_lib->vba.LinesInDETCRoundedDownToSwath[k])
2495 / mode_lib->vba.SwathHeightC[k]
2496 * (mode_lib->vba.HTotal[k]
2497 / mode_lib->vba.PixelClock[k]);
2498 double ActiveDRAMClockChangeLatencyMarginC = DPPOPPBufferingC
2499 + EffectiveLBLatencyHidingC + MaxDETBufferingTimeC
2500 - mode_lib->vba.DRAMClockChangeWatermark;
2501
2502 if (mode_lib->vba.ActiveDPPs > 1) {
2503 ActiveDRAMClockChangeLatencyMarginC =
2504 ActiveDRAMClockChangeLatencyMarginC
2505 - (1
2506 - 1
2507 / (mode_lib->vba.ActiveDPPs
2508 - 1))
2509 * mode_lib->vba.SwathHeightC[k]
2510 * (mode_lib->vba.HTotal[k]
2511 / mode_lib->vba.PixelClock[k]);
2512 }
2513 mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k] = dml_min(
2514 ActiveDRAMClockChangeLatencyMarginY,
2515 ActiveDRAMClockChangeLatencyMarginC);
2516 } else {
2517 mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k] =
2518 ActiveDRAMClockChangeLatencyMarginY;
2519 }
2520
2521 if (mode_lib->vba.WritebackEnable[k]) {
2522 double WritebackDRAMClockChangeLatencyMargin;
2523
2524 if (mode_lib->vba.WritebackPixelFormat[k] == dm_444_32) {
2525 WritebackDRAMClockChangeLatencyMargin =
2526 (double) (mode_lib->vba.WritebackInterfaceLumaBufferSize
2527 + mode_lib->vba.WritebackInterfaceChromaBufferSize)
2528 / (mode_lib->vba.WritebackDestinationWidth[k]
2529 * mode_lib->vba.WritebackDestinationHeight[k]
2530 / (mode_lib->vba.WritebackSourceHeight[k]
2531 * mode_lib->vba.HTotal[k]
2532 / mode_lib->vba.PixelClock[k])
2533 * 4)
2534 - mode_lib->vba.WritebackDRAMClockChangeWatermark;
2535 } else if (mode_lib->vba.WritebackPixelFormat[k] == dm_420_10) {
2536 WritebackDRAMClockChangeLatencyMargin =
2537 dml_min(
2538 (double) mode_lib->vba.WritebackInterfaceLumaBufferSize
2539 * 8.0 / 10,
2540 2.0
2541 * mode_lib->vba.WritebackInterfaceChromaBufferSize
2542 * 8 / 10)
2543 / (mode_lib->vba.WritebackDestinationWidth[k]
2544 * mode_lib->vba.WritebackDestinationHeight[k]
2545 / (mode_lib->vba.WritebackSourceHeight[k]
2546 * mode_lib->vba.HTotal[k]
2547 / mode_lib->vba.PixelClock[k]))
2548 - mode_lib->vba.WritebackDRAMClockChangeWatermark;
2549 } else {
2550 WritebackDRAMClockChangeLatencyMargin =
2551 dml_min(
2552 (double) mode_lib->vba.WritebackInterfaceLumaBufferSize,
2553 2.0
2554 * mode_lib->vba.WritebackInterfaceChromaBufferSize)
2555 / (mode_lib->vba.WritebackDestinationWidth[k]
2556 * mode_lib->vba.WritebackDestinationHeight[k]
2557 / (mode_lib->vba.WritebackSourceHeight[k]
2558 * mode_lib->vba.HTotal[k]
2559 / mode_lib->vba.PixelClock[k]))
2560 - mode_lib->vba.WritebackDRAMClockChangeWatermark;
2561 }
2562 mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k] = dml_min(
2563 mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k],
2564 WritebackDRAMClockChangeLatencyMargin);
2565 }
2566 }
2567
2568 mode_lib->vba.MinActiveDRAMClockChangeMargin = 999999;
2569 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2570 if (mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k]
2571 < mode_lib->vba.MinActiveDRAMClockChangeMargin) {
2572 mode_lib->vba.MinActiveDRAMClockChangeMargin =
2573 mode_lib->vba.ActiveDRAMClockChangeLatencyMargin[k];
2574 }
2575 }
2576
2577 mode_lib->vba.MinActiveDRAMClockChangeLatencySupported =
2578 mode_lib->vba.MinActiveDRAMClockChangeMargin
2579 + mode_lib->vba.DRAMClockChangeLatency;
2580
2581 if (mode_lib->vba.DRAMClockChangeSupportsVActive &&
2582 mode_lib->vba.MinActiveDRAMClockChangeMargin > 60) {
2583 mode_lib->vba.DRAMClockChangeWatermark += 25;
2584 mode_lib->vba.DRAMClockChangeSupport[0][0] = dm_dram_clock_change_vactive;
2585 } else {
2586 if (mode_lib->vba.SynchronizedVBlank || mode_lib->vba.NumberOfActivePlanes == 1) {
2587 mode_lib->vba.DRAMClockChangeSupport[0][0] = dm_dram_clock_change_vblank;
2588 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2589 if (!mode_lib->vba.AllowDRAMClockChangeDuringVBlank[k]) {
2590 mode_lib->vba.DRAMClockChangeSupport[0][0] =
2591 dm_dram_clock_change_unsupported;
2592 }
2593 }
2594 } else {
2595 mode_lib->vba.DRAMClockChangeSupport[0][0] = dm_dram_clock_change_unsupported;
2596 }
2597 }
2598 for (k = 0; k <= mode_lib->vba.soc.num_states; k++)
2599 for (j = 0; j < 2; j++)
2600 mode_lib->vba.DRAMClockChangeSupport[k][j] = mode_lib->vba.DRAMClockChangeSupport[0][0];
2601
2602 //XFC Parameters:
2603 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2604 if (mode_lib->vba.XFCEnabled[k] == true) {
2605 double TWait;
2606
2607 mode_lib->vba.XFCSlaveVUpdateOffset[k] = mode_lib->vba.XFCTSlvVupdateOffset;
2608 mode_lib->vba.XFCSlaveVupdateWidth[k] = mode_lib->vba.XFCTSlvVupdateWidth;
2609 mode_lib->vba.XFCSlaveVReadyOffset[k] = mode_lib->vba.XFCTSlvVreadyOffset;
2610 TWait = CalculateTWait(
2611 mode_lib->vba.PrefetchMode[mode_lib->vba.VoltageLevel][mode_lib->vba.maxMpcComb],
2612 mode_lib->vba.DRAMClockChangeLatency,
2613 mode_lib->vba.UrgentLatencyPixelDataOnly,
2614 mode_lib->vba.SREnterPlusExitTime);
2615 mode_lib->vba.XFCRemoteSurfaceFlipDelay = CalculateRemoteSurfaceFlipDelay(
2616 mode_lib,
2617 mode_lib->vba.VRatio[k],
2618 mode_lib->vba.SwathWidthY[k],
2619 dml_ceil(mode_lib->vba.BytePerPixelDETY[k], 1),
2620 mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k],
2621 mode_lib->vba.XFCTSlvVupdateOffset,
2622 mode_lib->vba.XFCTSlvVupdateWidth,
2623 mode_lib->vba.XFCTSlvVreadyOffset,
2624 mode_lib->vba.XFCXBUFLatencyTolerance,
2625 mode_lib->vba.XFCFillBWOverhead,
2626 mode_lib->vba.XFCSlvChunkSize,
2627 mode_lib->vba.XFCBusTransportTime,
2628 mode_lib->vba.TCalc,
2629 TWait,
2630 &mode_lib->vba.SrcActiveDrainRate,
2631 &mode_lib->vba.TInitXFill,
2632 &mode_lib->vba.TslvChk);
2633 mode_lib->vba.XFCRemoteSurfaceFlipLatency[k] =
2634 dml_floor(
2635 mode_lib->vba.XFCRemoteSurfaceFlipDelay
2636 / (mode_lib->vba.HTotal[k]
2637 / mode_lib->vba.PixelClock[k]),
2638 1);
2639 mode_lib->vba.XFCTransferDelay[k] =
2640 dml_ceil(
2641 mode_lib->vba.XFCBusTransportTime
2642 / (mode_lib->vba.HTotal[k]
2643 / mode_lib->vba.PixelClock[k]),
2644 1);
2645 mode_lib->vba.XFCPrechargeDelay[k] =
2646 dml_ceil(
2647 (mode_lib->vba.XFCBusTransportTime
2648 + mode_lib->vba.TInitXFill
2649 + mode_lib->vba.TslvChk)
2650 / (mode_lib->vba.HTotal[k]
2651 / mode_lib->vba.PixelClock[k]),
2652 1);
2653 mode_lib->vba.InitFillLevel = mode_lib->vba.XFCXBUFLatencyTolerance
2654 * mode_lib->vba.SrcActiveDrainRate;
2655 mode_lib->vba.FinalFillMargin =
2656 (mode_lib->vba.DestinationLinesToRequestVMInVBlank[k]
2657 + mode_lib->vba.DestinationLinesToRequestRowInVBlank[k])
2658 * mode_lib->vba.HTotal[k]
2659 / mode_lib->vba.PixelClock[k]
2660 * mode_lib->vba.SrcActiveDrainRate
2661 + mode_lib->vba.XFCFillConstant;
2662 mode_lib->vba.FinalFillLevel = mode_lib->vba.XFCRemoteSurfaceFlipDelay
2663 * mode_lib->vba.SrcActiveDrainRate
2664 + mode_lib->vba.FinalFillMargin;
2665 mode_lib->vba.RemainingFillLevel = dml_max(
2666 0.0,
2667 mode_lib->vba.FinalFillLevel - mode_lib->vba.InitFillLevel);
2668 mode_lib->vba.TFinalxFill = mode_lib->vba.RemainingFillLevel
2669 / (mode_lib->vba.SrcActiveDrainRate
2670 * mode_lib->vba.XFCFillBWOverhead / 100);
2671 mode_lib->vba.XFCPrefetchMargin[k] =
2672 mode_lib->vba.XFCRemoteSurfaceFlipDelay
2673 + mode_lib->vba.TFinalxFill
2674 + (mode_lib->vba.DestinationLinesToRequestVMInVBlank[k]
2675 + mode_lib->vba.DestinationLinesToRequestRowInVBlank[k])
2676 * mode_lib->vba.HTotal[k]
2677 / mode_lib->vba.PixelClock[k];
2678 } else {
2679 mode_lib->vba.XFCSlaveVUpdateOffset[k] = 0;
2680 mode_lib->vba.XFCSlaveVupdateWidth[k] = 0;
2681 mode_lib->vba.XFCSlaveVReadyOffset[k] = 0;
2682 mode_lib->vba.XFCRemoteSurfaceFlipLatency[k] = 0;
2683 mode_lib->vba.XFCPrechargeDelay[k] = 0;
2684 mode_lib->vba.XFCTransferDelay[k] = 0;
2685 mode_lib->vba.XFCPrefetchMargin[k] = 0;
2686 }
2687 }
2688 {
2689 unsigned int VStartupMargin = 0;
2690 bool FirstMainPlane = true;
2691
2692 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2693 if (mode_lib->vba.BlendingAndTiming[k] == k) {
2694 unsigned int Margin = (mode_lib->vba.MaxVStartupLines[k] - mode_lib->vba.VStartup[k])
2695 * mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k];
2696
2697 if (FirstMainPlane) {
2698 VStartupMargin = Margin;
2699 FirstMainPlane = false;
2700 } else
2701 VStartupMargin = dml_min(VStartupMargin, Margin);
2702 }
2703
2704 if (mode_lib->vba.UseMaximumVStartup) {
2705 if (mode_lib->vba.VTotal_Max[k] == mode_lib->vba.VTotal[k]) {
2706 //only use max vstart if it is not drr or lateflip.
2707 mode_lib->vba.VStartup[k] = mode_lib->vba.MaxVStartupLines[mode_lib->vba.BlendingAndTiming[k]];
2708 }
2709 }
2710 }
2711 }
2712 }
2713
dml20_DisplayPipeConfiguration(struct display_mode_lib * mode_lib)2714 static void dml20_DisplayPipeConfiguration(struct display_mode_lib *mode_lib)
2715 {
2716 double BytePerPixDETY;
2717 double BytePerPixDETC;
2718 double Read256BytesBlockHeightY;
2719 double Read256BytesBlockHeightC;
2720 double Read256BytesBlockWidthY;
2721 double Read256BytesBlockWidthC;
2722 double MaximumSwathHeightY;
2723 double MaximumSwathHeightC;
2724 double MinimumSwathHeightY;
2725 double MinimumSwathHeightC;
2726 double SwathWidth;
2727 double SwathWidthGranularityY;
2728 double SwathWidthGranularityC;
2729 double RoundedUpMaxSwathSizeBytesY;
2730 double RoundedUpMaxSwathSizeBytesC;
2731 unsigned int j, k;
2732
2733 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
2734 bool MainPlaneDoesODMCombine = false;
2735
2736 if (mode_lib->vba.SourcePixelFormat[k] == dm_444_64) {
2737 BytePerPixDETY = 8;
2738 BytePerPixDETC = 0;
2739 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_32) {
2740 BytePerPixDETY = 4;
2741 BytePerPixDETC = 0;
2742 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_16) {
2743 BytePerPixDETY = 2;
2744 BytePerPixDETC = 0;
2745 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_8) {
2746 BytePerPixDETY = 1;
2747 BytePerPixDETC = 0;
2748 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8) {
2749 BytePerPixDETY = 1;
2750 BytePerPixDETC = 2;
2751 } else {
2752 BytePerPixDETY = 4.0 / 3.0;
2753 BytePerPixDETC = 8.0 / 3.0;
2754 }
2755
2756 if ((mode_lib->vba.SourcePixelFormat[k] == dm_444_64
2757 || mode_lib->vba.SourcePixelFormat[k] == dm_444_32
2758 || mode_lib->vba.SourcePixelFormat[k] == dm_444_16
2759 || mode_lib->vba.SourcePixelFormat[k] == dm_444_8)) {
2760 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear) {
2761 Read256BytesBlockHeightY = 1;
2762 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_64) {
2763 Read256BytesBlockHeightY = 4;
2764 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_32
2765 || mode_lib->vba.SourcePixelFormat[k] == dm_444_16) {
2766 Read256BytesBlockHeightY = 8;
2767 } else {
2768 Read256BytesBlockHeightY = 16;
2769 }
2770 Read256BytesBlockWidthY = 256 / dml_ceil(BytePerPixDETY, 1)
2771 / Read256BytesBlockHeightY;
2772 Read256BytesBlockHeightC = 0;
2773 Read256BytesBlockWidthC = 0;
2774 } else {
2775 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear) {
2776 Read256BytesBlockHeightY = 1;
2777 Read256BytesBlockHeightC = 1;
2778 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8) {
2779 Read256BytesBlockHeightY = 16;
2780 Read256BytesBlockHeightC = 8;
2781 } else {
2782 Read256BytesBlockHeightY = 8;
2783 Read256BytesBlockHeightC = 8;
2784 }
2785 Read256BytesBlockWidthY = 256 / dml_ceil(BytePerPixDETY, 1)
2786 / Read256BytesBlockHeightY;
2787 Read256BytesBlockWidthC = 256 / dml_ceil(BytePerPixDETC, 2)
2788 / Read256BytesBlockHeightC;
2789 }
2790
2791 if (mode_lib->vba.SourceScan[k] == dm_horz) {
2792 MaximumSwathHeightY = Read256BytesBlockHeightY;
2793 MaximumSwathHeightC = Read256BytesBlockHeightC;
2794 } else {
2795 MaximumSwathHeightY = Read256BytesBlockWidthY;
2796 MaximumSwathHeightC = Read256BytesBlockWidthC;
2797 }
2798
2799 if ((mode_lib->vba.SourcePixelFormat[k] == dm_444_64
2800 || mode_lib->vba.SourcePixelFormat[k] == dm_444_32
2801 || mode_lib->vba.SourcePixelFormat[k] == dm_444_16
2802 || mode_lib->vba.SourcePixelFormat[k] == dm_444_8)) {
2803 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear
2804 || (mode_lib->vba.SourcePixelFormat[k] == dm_444_64
2805 && (mode_lib->vba.SurfaceTiling[k]
2806 == dm_sw_4kb_s
2807 || mode_lib->vba.SurfaceTiling[k]
2808 == dm_sw_4kb_s_x
2809 || mode_lib->vba.SurfaceTiling[k]
2810 == dm_sw_64kb_s
2811 || mode_lib->vba.SurfaceTiling[k]
2812 == dm_sw_64kb_s_t
2813 || mode_lib->vba.SurfaceTiling[k]
2814 == dm_sw_64kb_s_x
2815 || mode_lib->vba.SurfaceTiling[k]
2816 == dm_sw_var_s
2817 || mode_lib->vba.SurfaceTiling[k]
2818 == dm_sw_var_s_x)
2819 && mode_lib->vba.SourceScan[k] == dm_horz)) {
2820 MinimumSwathHeightY = MaximumSwathHeightY;
2821 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_8
2822 && mode_lib->vba.SourceScan[k] != dm_horz) {
2823 MinimumSwathHeightY = MaximumSwathHeightY;
2824 } else {
2825 MinimumSwathHeightY = MaximumSwathHeightY / 2.0;
2826 }
2827 MinimumSwathHeightC = MaximumSwathHeightC;
2828 } else {
2829 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear) {
2830 MinimumSwathHeightY = MaximumSwathHeightY;
2831 MinimumSwathHeightC = MaximumSwathHeightC;
2832 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8
2833 && mode_lib->vba.SourceScan[k] == dm_horz) {
2834 MinimumSwathHeightY = MaximumSwathHeightY / 2.0;
2835 MinimumSwathHeightC = MaximumSwathHeightC;
2836 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_10
2837 && mode_lib->vba.SourceScan[k] == dm_horz) {
2838 MinimumSwathHeightC = MaximumSwathHeightC / 2.0;
2839 MinimumSwathHeightY = MaximumSwathHeightY;
2840 } else {
2841 MinimumSwathHeightY = MaximumSwathHeightY;
2842 MinimumSwathHeightC = MaximumSwathHeightC;
2843 }
2844 }
2845
2846 if (mode_lib->vba.SourceScan[k] == dm_horz) {
2847 SwathWidth = mode_lib->vba.ViewportWidth[k];
2848 } else {
2849 SwathWidth = mode_lib->vba.ViewportHeight[k];
2850 }
2851
2852 if (mode_lib->vba.ODMCombineEnabled[k] == dm_odm_combine_mode_2to1) {
2853 MainPlaneDoesODMCombine = true;
2854 }
2855 for (j = 0; j < mode_lib->vba.NumberOfActivePlanes; ++j) {
2856 if (mode_lib->vba.BlendingAndTiming[k] == j
2857 && mode_lib->vba.ODMCombineEnabled[k] == dm_odm_combine_mode_2to1) {
2858 MainPlaneDoesODMCombine = true;
2859 }
2860 }
2861
2862 if (MainPlaneDoesODMCombine == true) {
2863 SwathWidth = dml_min(
2864 SwathWidth,
2865 mode_lib->vba.HActive[k] / 2.0 * mode_lib->vba.HRatio[k]);
2866 } else {
2867 if (mode_lib->vba.DPPPerPlane[k] == 0)
2868 SwathWidth = 0;
2869 else
2870 SwathWidth = SwathWidth / mode_lib->vba.DPPPerPlane[k];
2871 }
2872
2873 SwathWidthGranularityY = 256 / dml_ceil(BytePerPixDETY, 1) / MaximumSwathHeightY;
2874 RoundedUpMaxSwathSizeBytesY = (dml_ceil(
2875 (double) (SwathWidth - 1),
2876 SwathWidthGranularityY) + SwathWidthGranularityY) * BytePerPixDETY
2877 * MaximumSwathHeightY;
2878 if (mode_lib->vba.SourcePixelFormat[k] == dm_420_10) {
2879 RoundedUpMaxSwathSizeBytesY = dml_ceil(RoundedUpMaxSwathSizeBytesY, 256)
2880 + 256;
2881 }
2882 if (MaximumSwathHeightC > 0) {
2883 SwathWidthGranularityC = 256.0 / dml_ceil(BytePerPixDETC, 2)
2884 / MaximumSwathHeightC;
2885 RoundedUpMaxSwathSizeBytesC = (dml_ceil(
2886 (double) (SwathWidth / 2.0 - 1),
2887 SwathWidthGranularityC) + SwathWidthGranularityC)
2888 * BytePerPixDETC * MaximumSwathHeightC;
2889 if (mode_lib->vba.SourcePixelFormat[k] == dm_420_10) {
2890 RoundedUpMaxSwathSizeBytesC = dml_ceil(
2891 RoundedUpMaxSwathSizeBytesC,
2892 256) + 256;
2893 }
2894 } else
2895 RoundedUpMaxSwathSizeBytesC = 0.0;
2896
2897 if (RoundedUpMaxSwathSizeBytesY + RoundedUpMaxSwathSizeBytesC
2898 <= mode_lib->vba.DETBufferSizeInKByte[0] * 1024.0 / 2.0) {
2899 mode_lib->vba.SwathHeightY[k] = MaximumSwathHeightY;
2900 mode_lib->vba.SwathHeightC[k] = MaximumSwathHeightC;
2901 } else {
2902 mode_lib->vba.SwathHeightY[k] = MinimumSwathHeightY;
2903 mode_lib->vba.SwathHeightC[k] = MinimumSwathHeightC;
2904 }
2905
2906 if (mode_lib->vba.SwathHeightC[k] == 0) {
2907 mode_lib->vba.DETBufferSizeY[k] = mode_lib->vba.DETBufferSizeInKByte[0] * 1024;
2908 mode_lib->vba.DETBufferSizeC[k] = 0;
2909 } else if (mode_lib->vba.SwathHeightY[k] <= mode_lib->vba.SwathHeightC[k]) {
2910 mode_lib->vba.DETBufferSizeY[k] = mode_lib->vba.DETBufferSizeInKByte[0]
2911 * 1024.0 / 2;
2912 mode_lib->vba.DETBufferSizeC[k] = mode_lib->vba.DETBufferSizeInKByte[0]
2913 * 1024.0 / 2;
2914 } else {
2915 mode_lib->vba.DETBufferSizeY[k] = mode_lib->vba.DETBufferSizeInKByte[0]
2916 * 1024.0 * 2 / 3;
2917 mode_lib->vba.DETBufferSizeC[k] = mode_lib->vba.DETBufferSizeInKByte[0]
2918 * 1024.0 / 3;
2919 }
2920 }
2921 }
2922
CalculateTWait(unsigned int PrefetchMode,double DRAMClockChangeLatency,double UrgentLatencyPixelDataOnly,double SREnterPlusExitTime)2923 static double CalculateTWait(
2924 unsigned int PrefetchMode,
2925 double DRAMClockChangeLatency,
2926 double UrgentLatencyPixelDataOnly,
2927 double SREnterPlusExitTime)
2928 {
2929 if (PrefetchMode == 0) {
2930 return dml_max(
2931 DRAMClockChangeLatency + UrgentLatencyPixelDataOnly,
2932 dml_max(SREnterPlusExitTime, UrgentLatencyPixelDataOnly));
2933 } else if (PrefetchMode == 1) {
2934 return dml_max(SREnterPlusExitTime, UrgentLatencyPixelDataOnly);
2935 } else {
2936 return UrgentLatencyPixelDataOnly;
2937 }
2938 }
2939
CalculateRemoteSurfaceFlipDelay(struct display_mode_lib * mode_lib,double VRatio,double SwathWidth,double Bpp,double LineTime,double XFCTSlvVupdateOffset,double XFCTSlvVupdateWidth,double XFCTSlvVreadyOffset,double XFCXBUFLatencyTolerance,double XFCFillBWOverhead,double XFCSlvChunkSize,double XFCBusTransportTime,double TCalc,double TWait,double * SrcActiveDrainRate,double * TInitXFill,double * TslvChk)2940 static double CalculateRemoteSurfaceFlipDelay(
2941 struct display_mode_lib *mode_lib,
2942 double VRatio,
2943 double SwathWidth,
2944 double Bpp,
2945 double LineTime,
2946 double XFCTSlvVupdateOffset,
2947 double XFCTSlvVupdateWidth,
2948 double XFCTSlvVreadyOffset,
2949 double XFCXBUFLatencyTolerance,
2950 double XFCFillBWOverhead,
2951 double XFCSlvChunkSize,
2952 double XFCBusTransportTime,
2953 double TCalc,
2954 double TWait,
2955 double *SrcActiveDrainRate,
2956 double *TInitXFill,
2957 double *TslvChk)
2958 {
2959 double TSlvSetup, AvgfillRate, result;
2960
2961 *SrcActiveDrainRate = VRatio * SwathWidth * Bpp / LineTime;
2962 TSlvSetup = XFCTSlvVupdateOffset + XFCTSlvVupdateWidth + XFCTSlvVreadyOffset;
2963 *TInitXFill = XFCXBUFLatencyTolerance / (1 + XFCFillBWOverhead / 100);
2964 AvgfillRate = *SrcActiveDrainRate * (1 + XFCFillBWOverhead / 100);
2965 *TslvChk = XFCSlvChunkSize / AvgfillRate;
2966 dml_print(
2967 "DML::CalculateRemoteSurfaceFlipDelay: SrcActiveDrainRate: %f\n",
2968 *SrcActiveDrainRate);
2969 dml_print("DML::CalculateRemoteSurfaceFlipDelay: TSlvSetup: %f\n", TSlvSetup);
2970 dml_print("DML::CalculateRemoteSurfaceFlipDelay: TInitXFill: %f\n", *TInitXFill);
2971 dml_print("DML::CalculateRemoteSurfaceFlipDelay: AvgfillRate: %f\n", AvgfillRate);
2972 dml_print("DML::CalculateRemoteSurfaceFlipDelay: TslvChk: %f\n", *TslvChk);
2973 result = 2 * XFCBusTransportTime + TSlvSetup + TCalc + TWait + *TslvChk + *TInitXFill; // TODO: This doesn't seem to match programming guide
2974 dml_print("DML::CalculateRemoteSurfaceFlipDelay: RemoteSurfaceFlipDelay: %f\n", result);
2975 return result;
2976 }
2977
CalculateWriteBackDelay(enum source_format_class WritebackPixelFormat,double WritebackHRatio,double WritebackVRatio,unsigned int WritebackLumaHTaps,unsigned int WritebackLumaVTaps,unsigned int WritebackChromaHTaps,unsigned int WritebackChromaVTaps,unsigned int WritebackDestinationWidth)2978 static double CalculateWriteBackDelay(
2979 enum source_format_class WritebackPixelFormat,
2980 double WritebackHRatio,
2981 double WritebackVRatio,
2982 unsigned int WritebackLumaHTaps,
2983 unsigned int WritebackLumaVTaps,
2984 unsigned int WritebackChromaHTaps,
2985 unsigned int WritebackChromaVTaps,
2986 unsigned int WritebackDestinationWidth)
2987 {
2988 double CalculateWriteBackDelay =
2989 dml_max(
2990 dml_ceil(WritebackLumaHTaps / 4.0, 1) / WritebackHRatio,
2991 WritebackLumaVTaps * dml_ceil(1.0 / WritebackVRatio, 1)
2992 * dml_ceil(
2993 WritebackDestinationWidth
2994 / 4.0,
2995 1)
2996 + dml_ceil(1.0 / WritebackVRatio, 1)
2997 * (dml_ceil(
2998 WritebackLumaVTaps
2999 / 4.0,
3000 1) + 4));
3001
3002 if (WritebackPixelFormat != dm_444_32) {
3003 CalculateWriteBackDelay =
3004 dml_max(
3005 CalculateWriteBackDelay,
3006 dml_max(
3007 dml_ceil(
3008 WritebackChromaHTaps
3009 / 2.0,
3010 1)
3011 / (2
3012 * WritebackHRatio),
3013 WritebackChromaVTaps
3014 * dml_ceil(
3015 1
3016 / (2
3017 * WritebackVRatio),
3018 1)
3019 * dml_ceil(
3020 WritebackDestinationWidth
3021 / 2.0
3022 / 2.0,
3023 1)
3024 + dml_ceil(
3025 1
3026 / (2
3027 * WritebackVRatio),
3028 1)
3029 * (dml_ceil(
3030 WritebackChromaVTaps
3031 / 4.0,
3032 1)
3033 + 4)));
3034 }
3035 return CalculateWriteBackDelay;
3036 }
3037
CalculateActiveRowBandwidth(bool GPUVMEnable,enum source_format_class SourcePixelFormat,double VRatio,bool DCCEnable,double LineTime,unsigned int MetaRowByteLuma,unsigned int MetaRowByteChroma,unsigned int meta_row_height_luma,unsigned int meta_row_height_chroma,unsigned int PixelPTEBytesPerRowLuma,unsigned int PixelPTEBytesPerRowChroma,unsigned int dpte_row_height_luma,unsigned int dpte_row_height_chroma,double * meta_row_bw,double * dpte_row_bw,double * qual_row_bw)3038 static void CalculateActiveRowBandwidth(
3039 bool GPUVMEnable,
3040 enum source_format_class SourcePixelFormat,
3041 double VRatio,
3042 bool DCCEnable,
3043 double LineTime,
3044 unsigned int MetaRowByteLuma,
3045 unsigned int MetaRowByteChroma,
3046 unsigned int meta_row_height_luma,
3047 unsigned int meta_row_height_chroma,
3048 unsigned int PixelPTEBytesPerRowLuma,
3049 unsigned int PixelPTEBytesPerRowChroma,
3050 unsigned int dpte_row_height_luma,
3051 unsigned int dpte_row_height_chroma,
3052 double *meta_row_bw,
3053 double *dpte_row_bw,
3054 double *qual_row_bw)
3055 {
3056 if (DCCEnable != true) {
3057 *meta_row_bw = 0;
3058 } else if (SourcePixelFormat == dm_420_8 || SourcePixelFormat == dm_420_10) {
3059 *meta_row_bw = VRatio * MetaRowByteLuma / (meta_row_height_luma * LineTime)
3060 + VRatio / 2 * MetaRowByteChroma
3061 / (meta_row_height_chroma * LineTime);
3062 } else {
3063 *meta_row_bw = VRatio * MetaRowByteLuma / (meta_row_height_luma * LineTime);
3064 }
3065
3066 if (GPUVMEnable != true) {
3067 *dpte_row_bw = 0;
3068 } else if (SourcePixelFormat == dm_420_8 || SourcePixelFormat == dm_420_10) {
3069 *dpte_row_bw = VRatio * PixelPTEBytesPerRowLuma / (dpte_row_height_luma * LineTime)
3070 + VRatio / 2 * PixelPTEBytesPerRowChroma
3071 / (dpte_row_height_chroma * LineTime);
3072 } else {
3073 *dpte_row_bw = VRatio * PixelPTEBytesPerRowLuma / (dpte_row_height_luma * LineTime);
3074 }
3075
3076 if ((SourcePixelFormat == dm_420_8 || SourcePixelFormat == dm_420_10)) {
3077 *qual_row_bw = *meta_row_bw + *dpte_row_bw;
3078 } else {
3079 *qual_row_bw = 0;
3080 }
3081 }
3082
CalculateFlipSchedule(struct display_mode_lib * mode_lib,double UrgentExtraLatency,double UrgentLatencyPixelDataOnly,unsigned int GPUVMMaxPageTableLevels,bool GPUVMEnable,double BandwidthAvailableForImmediateFlip,unsigned int TotImmediateFlipBytes,enum source_format_class SourcePixelFormat,unsigned int ImmediateFlipBytes,double LineTime,double VRatio,double Tno_bw,double PDEAndMetaPTEBytesFrame,unsigned int MetaRowByte,unsigned int PixelPTEBytesPerRow,bool DCCEnable,unsigned int dpte_row_height,unsigned int meta_row_height,double qual_row_bw,double * DestinationLinesToRequestVMInImmediateFlip,double * DestinationLinesToRequestRowInImmediateFlip,double * final_flip_bw,bool * ImmediateFlipSupportedForPipe)3083 static void CalculateFlipSchedule(
3084 struct display_mode_lib *mode_lib,
3085 double UrgentExtraLatency,
3086 double UrgentLatencyPixelDataOnly,
3087 unsigned int GPUVMMaxPageTableLevels,
3088 bool GPUVMEnable,
3089 double BandwidthAvailableForImmediateFlip,
3090 unsigned int TotImmediateFlipBytes,
3091 enum source_format_class SourcePixelFormat,
3092 unsigned int ImmediateFlipBytes,
3093 double LineTime,
3094 double VRatio,
3095 double Tno_bw,
3096 double PDEAndMetaPTEBytesFrame,
3097 unsigned int MetaRowByte,
3098 unsigned int PixelPTEBytesPerRow,
3099 bool DCCEnable,
3100 unsigned int dpte_row_height,
3101 unsigned int meta_row_height,
3102 double qual_row_bw,
3103 double *DestinationLinesToRequestVMInImmediateFlip,
3104 double *DestinationLinesToRequestRowInImmediateFlip,
3105 double *final_flip_bw,
3106 bool *ImmediateFlipSupportedForPipe)
3107 {
3108 double min_row_time = 0.0;
3109
3110 if (SourcePixelFormat == dm_420_8 || SourcePixelFormat == dm_420_10) {
3111 *DestinationLinesToRequestVMInImmediateFlip = 0.0;
3112 *DestinationLinesToRequestRowInImmediateFlip = 0.0;
3113 *final_flip_bw = qual_row_bw;
3114 *ImmediateFlipSupportedForPipe = true;
3115 } else {
3116 double TimeForFetchingMetaPTEImmediateFlip;
3117 double TimeForFetchingRowInVBlankImmediateFlip;
3118
3119 if (GPUVMEnable == true) {
3120 mode_lib->vba.ImmediateFlipBW[0] = BandwidthAvailableForImmediateFlip
3121 * ImmediateFlipBytes / TotImmediateFlipBytes;
3122 TimeForFetchingMetaPTEImmediateFlip =
3123 dml_max(
3124 Tno_bw
3125 + PDEAndMetaPTEBytesFrame
3126 / mode_lib->vba.ImmediateFlipBW[0],
3127 dml_max(
3128 UrgentExtraLatency
3129 + UrgentLatencyPixelDataOnly
3130 * (GPUVMMaxPageTableLevels
3131 - 1),
3132 LineTime / 4.0));
3133 } else {
3134 TimeForFetchingMetaPTEImmediateFlip = 0;
3135 }
3136
3137 *DestinationLinesToRequestVMInImmediateFlip = dml_floor(
3138 4.0 * (TimeForFetchingMetaPTEImmediateFlip / LineTime + 0.125),
3139 1) / 4.0;
3140
3141 if ((GPUVMEnable || DCCEnable)) {
3142 mode_lib->vba.ImmediateFlipBW[0] = BandwidthAvailableForImmediateFlip
3143 * ImmediateFlipBytes / TotImmediateFlipBytes;
3144 TimeForFetchingRowInVBlankImmediateFlip = dml_max(
3145 (MetaRowByte + PixelPTEBytesPerRow)
3146 / mode_lib->vba.ImmediateFlipBW[0],
3147 dml_max(UrgentLatencyPixelDataOnly, LineTime / 4.0));
3148 } else {
3149 TimeForFetchingRowInVBlankImmediateFlip = 0;
3150 }
3151
3152 *DestinationLinesToRequestRowInImmediateFlip = dml_floor(
3153 4.0 * (TimeForFetchingRowInVBlankImmediateFlip / LineTime + 0.125),
3154 1) / 4.0;
3155
3156 if (GPUVMEnable == true) {
3157 *final_flip_bw =
3158 dml_max(
3159 PDEAndMetaPTEBytesFrame
3160 / (*DestinationLinesToRequestVMInImmediateFlip
3161 * LineTime),
3162 (MetaRowByte + PixelPTEBytesPerRow)
3163 / (TimeForFetchingRowInVBlankImmediateFlip
3164 * LineTime));
3165 } else if (MetaRowByte + PixelPTEBytesPerRow > 0) {
3166 *final_flip_bw = (MetaRowByte + PixelPTEBytesPerRow)
3167 / (TimeForFetchingRowInVBlankImmediateFlip * LineTime);
3168 } else {
3169 *final_flip_bw = 0;
3170 }
3171
3172 if (GPUVMEnable && !DCCEnable)
3173 min_row_time = dpte_row_height * LineTime / VRatio;
3174 else if (!GPUVMEnable && DCCEnable)
3175 min_row_time = meta_row_height * LineTime / VRatio;
3176 else
3177 min_row_time = dml_min(dpte_row_height, meta_row_height) * LineTime
3178 / VRatio;
3179
3180 if (*DestinationLinesToRequestVMInImmediateFlip >= 8
3181 || *DestinationLinesToRequestRowInImmediateFlip >= 16
3182 || TimeForFetchingMetaPTEImmediateFlip
3183 + 2 * TimeForFetchingRowInVBlankImmediateFlip
3184 > min_row_time)
3185 *ImmediateFlipSupportedForPipe = false;
3186 else
3187 *ImmediateFlipSupportedForPipe = true;
3188 }
3189 }
3190
TruncToValidBPP(double DecimalBPP,bool DSCEnabled,enum output_encoder_class Output,enum output_format_class Format,unsigned int DSCInputBitPerComponent)3191 static unsigned int TruncToValidBPP(
3192 double DecimalBPP,
3193 bool DSCEnabled,
3194 enum output_encoder_class Output,
3195 enum output_format_class Format,
3196 unsigned int DSCInputBitPerComponent)
3197 {
3198 if (Output == dm_hdmi) {
3199 if (Format == dm_420) {
3200 if (DecimalBPP >= 18)
3201 return 18;
3202 else if (DecimalBPP >= 15)
3203 return 15;
3204 else if (DecimalBPP >= 12)
3205 return 12;
3206 else
3207 return BPP_INVALID;
3208 } else if (Format == dm_444) {
3209 if (DecimalBPP >= 36)
3210 return 36;
3211 else if (DecimalBPP >= 30)
3212 return 30;
3213 else if (DecimalBPP >= 24)
3214 return 24;
3215 else if (DecimalBPP >= 18)
3216 return 18;
3217 else
3218 return BPP_INVALID;
3219 } else {
3220 if (DecimalBPP / 1.5 >= 24)
3221 return 24;
3222 else if (DecimalBPP / 1.5 >= 20)
3223 return 20;
3224 else if (DecimalBPP / 1.5 >= 16)
3225 return 16;
3226 else
3227 return BPP_INVALID;
3228 }
3229 } else {
3230 if (DSCEnabled) {
3231 if (Format == dm_420) {
3232 if (DecimalBPP < 6)
3233 return BPP_INVALID;
3234 else if (DecimalBPP >= 1.5 * DSCInputBitPerComponent - 1 / 16)
3235 return 1.5 * DSCInputBitPerComponent - 1 / 16;
3236 else
3237 return dml_floor(16 * DecimalBPP, 1) / 16;
3238 } else if (Format == dm_n422) {
3239 if (DecimalBPP < 7)
3240 return BPP_INVALID;
3241 else if (DecimalBPP >= 2 * DSCInputBitPerComponent - 1 / 16)
3242 return 2 * DSCInputBitPerComponent - 1 / 16;
3243 else
3244 return dml_floor(16 * DecimalBPP, 1) / 16;
3245 } else {
3246 if (DecimalBPP < 8)
3247 return BPP_INVALID;
3248 else if (DecimalBPP >= 3 * DSCInputBitPerComponent - 1 / 16)
3249 return 3 * DSCInputBitPerComponent - 1 / 16;
3250 else
3251 return dml_floor(16 * DecimalBPP, 1) / 16;
3252 }
3253 } else if (Format == dm_420) {
3254 if (DecimalBPP >= 18)
3255 return 18;
3256 else if (DecimalBPP >= 15)
3257 return 15;
3258 else if (DecimalBPP >= 12)
3259 return 12;
3260 else
3261 return BPP_INVALID;
3262 } else if (Format == dm_s422 || Format == dm_n422) {
3263 if (DecimalBPP >= 24)
3264 return 24;
3265 else if (DecimalBPP >= 20)
3266 return 20;
3267 else if (DecimalBPP >= 16)
3268 return 16;
3269 else
3270 return BPP_INVALID;
3271 } else {
3272 if (DecimalBPP >= 36)
3273 return 36;
3274 else if (DecimalBPP >= 30)
3275 return 30;
3276 else if (DecimalBPP >= 24)
3277 return 24;
3278 else if (DecimalBPP >= 18)
3279 return 18;
3280 else
3281 return BPP_INVALID;
3282 }
3283 }
3284 }
3285
dml20_ModeSupportAndSystemConfigurationFull(struct display_mode_lib * mode_lib)3286 void dml20_ModeSupportAndSystemConfigurationFull(struct display_mode_lib *mode_lib)
3287 {
3288 struct vba_vars_st *locals = &mode_lib->vba;
3289
3290 int i;
3291 unsigned int j, k, m;
3292
3293 /*MODE SUPPORT, VOLTAGE STATE AND SOC CONFIGURATION*/
3294
3295 /*Scale Ratio, taps Support Check*/
3296
3297 mode_lib->vba.ScaleRatioAndTapsSupport = true;
3298 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3299 if (mode_lib->vba.ScalerEnabled[k] == false
3300 && ((mode_lib->vba.SourcePixelFormat[k] != dm_444_64
3301 && mode_lib->vba.SourcePixelFormat[k] != dm_444_32
3302 && mode_lib->vba.SourcePixelFormat[k] != dm_444_16
3303 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_16
3304 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_8)
3305 || mode_lib->vba.HRatio[k] != 1.0
3306 || mode_lib->vba.htaps[k] != 1.0
3307 || mode_lib->vba.VRatio[k] != 1.0
3308 || mode_lib->vba.vtaps[k] != 1.0)) {
3309 mode_lib->vba.ScaleRatioAndTapsSupport = false;
3310 } else if (mode_lib->vba.vtaps[k] < 1.0 || mode_lib->vba.vtaps[k] > 8.0
3311 || mode_lib->vba.htaps[k] < 1.0 || mode_lib->vba.htaps[k] > 8.0
3312 || (mode_lib->vba.htaps[k] > 1.0
3313 && (mode_lib->vba.htaps[k] % 2) == 1)
3314 || mode_lib->vba.HRatio[k] > mode_lib->vba.MaxHSCLRatio
3315 || mode_lib->vba.VRatio[k] > mode_lib->vba.MaxVSCLRatio
3316 || mode_lib->vba.HRatio[k] > mode_lib->vba.htaps[k]
3317 || mode_lib->vba.VRatio[k] > mode_lib->vba.vtaps[k]
3318 || (mode_lib->vba.SourcePixelFormat[k] != dm_444_64
3319 && mode_lib->vba.SourcePixelFormat[k] != dm_444_32
3320 && mode_lib->vba.SourcePixelFormat[k] != dm_444_16
3321 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_16
3322 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_8
3323 && (mode_lib->vba.HRatio[k] / 2.0
3324 > mode_lib->vba.HTAPsChroma[k]
3325 || mode_lib->vba.VRatio[k] / 2.0
3326 > mode_lib->vba.VTAPsChroma[k]))) {
3327 mode_lib->vba.ScaleRatioAndTapsSupport = false;
3328 }
3329 }
3330 /*Source Format, Pixel Format and Scan Support Check*/
3331
3332 mode_lib->vba.SourceFormatPixelAndScanSupport = true;
3333 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3334 if ((mode_lib->vba.SurfaceTiling[k] == dm_sw_linear
3335 && mode_lib->vba.SourceScan[k] != dm_horz)
3336 || ((mode_lib->vba.SurfaceTiling[k] == dm_sw_4kb_d
3337 || mode_lib->vba.SurfaceTiling[k] == dm_sw_4kb_d_x
3338 || mode_lib->vba.SurfaceTiling[k] == dm_sw_64kb_d
3339 || mode_lib->vba.SurfaceTiling[k] == dm_sw_64kb_d_t
3340 || mode_lib->vba.SurfaceTiling[k] == dm_sw_64kb_d_x
3341 || mode_lib->vba.SurfaceTiling[k] == dm_sw_var_d
3342 || mode_lib->vba.SurfaceTiling[k] == dm_sw_var_d_x)
3343 && mode_lib->vba.SourcePixelFormat[k] != dm_444_64)
3344 || (mode_lib->vba.SurfaceTiling[k] == dm_sw_64kb_r_x
3345 && (mode_lib->vba.SourcePixelFormat[k] == dm_mono_8
3346 || mode_lib->vba.SourcePixelFormat[k]
3347 == dm_420_8
3348 || mode_lib->vba.SourcePixelFormat[k]
3349 == dm_420_10))
3350 || (((mode_lib->vba.SurfaceTiling[k] == dm_sw_gfx7_2d_thin_gl
3351 || mode_lib->vba.SurfaceTiling[k]
3352 == dm_sw_gfx7_2d_thin_l_vp)
3353 && !((mode_lib->vba.SourcePixelFormat[k]
3354 == dm_444_64
3355 || mode_lib->vba.SourcePixelFormat[k]
3356 == dm_444_32)
3357 && mode_lib->vba.SourceScan[k]
3358 == dm_horz
3359 && mode_lib->vba.SupportGFX7CompatibleTilingIn32bppAnd64bpp
3360 == true
3361 && mode_lib->vba.DCCEnable[k]
3362 == false))
3363 || (mode_lib->vba.DCCEnable[k] == true
3364 && (mode_lib->vba.SurfaceTiling[k]
3365 == dm_sw_linear
3366 || mode_lib->vba.SourcePixelFormat[k]
3367 == dm_420_8
3368 || mode_lib->vba.SourcePixelFormat[k]
3369 == dm_420_10)))) {
3370 mode_lib->vba.SourceFormatPixelAndScanSupport = false;
3371 }
3372 }
3373 /*Bandwidth Support Check*/
3374
3375 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3376 if (mode_lib->vba.SourcePixelFormat[k] == dm_444_64) {
3377 locals->BytePerPixelInDETY[k] = 8.0;
3378 locals->BytePerPixelInDETC[k] = 0.0;
3379 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_32) {
3380 locals->BytePerPixelInDETY[k] = 4.0;
3381 locals->BytePerPixelInDETC[k] = 0.0;
3382 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_444_16
3383 || mode_lib->vba.SourcePixelFormat[k] == dm_mono_16) {
3384 locals->BytePerPixelInDETY[k] = 2.0;
3385 locals->BytePerPixelInDETC[k] = 0.0;
3386 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_mono_8) {
3387 locals->BytePerPixelInDETY[k] = 1.0;
3388 locals->BytePerPixelInDETC[k] = 0.0;
3389 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8) {
3390 locals->BytePerPixelInDETY[k] = 1.0;
3391 locals->BytePerPixelInDETC[k] = 2.0;
3392 } else {
3393 locals->BytePerPixelInDETY[k] = 4.0 / 3;
3394 locals->BytePerPixelInDETC[k] = 8.0 / 3;
3395 }
3396 if (mode_lib->vba.SourceScan[k] == dm_horz) {
3397 locals->SwathWidthYSingleDPP[k] = mode_lib->vba.ViewportWidth[k];
3398 } else {
3399 locals->SwathWidthYSingleDPP[k] = mode_lib->vba.ViewportHeight[k];
3400 }
3401 }
3402 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3403 locals->ReadBandwidthLuma[k] = locals->SwathWidthYSingleDPP[k] * dml_ceil(locals->BytePerPixelInDETY[k], 1.0)
3404 / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k]) * mode_lib->vba.VRatio[k];
3405 locals->ReadBandwidthChroma[k] = locals->SwathWidthYSingleDPP[k] / 2 * dml_ceil(locals->BytePerPixelInDETC[k], 2.0)
3406 / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k]) * mode_lib->vba.VRatio[k] / 2.0;
3407 locals->ReadBandwidth[k] = locals->ReadBandwidthLuma[k] + locals->ReadBandwidthChroma[k];
3408 }
3409 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3410 if (mode_lib->vba.WritebackEnable[k] == true
3411 && mode_lib->vba.WritebackPixelFormat[k] == dm_444_32) {
3412 locals->WriteBandwidth[k] = mode_lib->vba.WritebackDestinationWidth[k]
3413 * mode_lib->vba.WritebackDestinationHeight[k]
3414 / (mode_lib->vba.WritebackSourceHeight[k]
3415 * mode_lib->vba.HTotal[k]
3416 / mode_lib->vba.PixelClock[k]) * 4.0;
3417 } else if (mode_lib->vba.WritebackEnable[k] == true
3418 && mode_lib->vba.WritebackPixelFormat[k] == dm_420_10) {
3419 locals->WriteBandwidth[k] = mode_lib->vba.WritebackDestinationWidth[k]
3420 * mode_lib->vba.WritebackDestinationHeight[k]
3421 / (mode_lib->vba.WritebackSourceHeight[k]
3422 * mode_lib->vba.HTotal[k]
3423 / mode_lib->vba.PixelClock[k]) * 3.0;
3424 } else if (mode_lib->vba.WritebackEnable[k] == true) {
3425 locals->WriteBandwidth[k] = mode_lib->vba.WritebackDestinationWidth[k]
3426 * mode_lib->vba.WritebackDestinationHeight[k]
3427 / (mode_lib->vba.WritebackSourceHeight[k]
3428 * mode_lib->vba.HTotal[k]
3429 / mode_lib->vba.PixelClock[k]) * 1.5;
3430 } else {
3431 locals->WriteBandwidth[k] = 0.0;
3432 }
3433 }
3434 mode_lib->vba.DCCEnabledInAnyPlane = false;
3435 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3436 if (mode_lib->vba.DCCEnable[k] == true) {
3437 mode_lib->vba.DCCEnabledInAnyPlane = true;
3438 }
3439 }
3440 mode_lib->vba.UrgentLatency = mode_lib->vba.UrgentLatencyPixelDataOnly;
3441 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
3442 locals->FabricAndDRAMBandwidthPerState[i] = dml_min(
3443 mode_lib->vba.DRAMSpeedPerState[i] * mode_lib->vba.NumberOfChannels
3444 * mode_lib->vba.DRAMChannelWidth,
3445 mode_lib->vba.FabricClockPerState[i]
3446 * mode_lib->vba.FabricDatapathToDCNDataReturn) / 1000;
3447 locals->ReturnBWToDCNPerState = dml_min(locals->ReturnBusWidth * locals->DCFCLKPerState[i],
3448 locals->FabricAndDRAMBandwidthPerState[i] * 1000)
3449 * locals->PercentOfIdealDRAMFabricAndSDPPortBWReceivedAfterUrgLatencyPixelDataOnly / 100;
3450
3451 locals->ReturnBWPerState[i][0] = locals->ReturnBWToDCNPerState;
3452
3453 if (locals->DCCEnabledInAnyPlane == true && locals->ReturnBWToDCNPerState > locals->DCFCLKPerState[i] * locals->ReturnBusWidth / 4) {
3454 locals->ReturnBWPerState[i][0] = dml_min(locals->ReturnBWPerState[i][0],
3455 locals->ReturnBWToDCNPerState * 4 * (1 - locals->UrgentLatency /
3456 ((locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024
3457 / (locals->ReturnBWToDCNPerState - locals->DCFCLKPerState[i]
3458 * locals->ReturnBusWidth / 4) + locals->UrgentLatency)));
3459 }
3460 locals->CriticalPoint = 2 * locals->ReturnBusWidth * locals->DCFCLKPerState[i] *
3461 locals->UrgentLatency / (locals->ReturnBWToDCNPerState * locals->UrgentLatency
3462 + (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024);
3463
3464 if (locals->DCCEnabledInAnyPlane && locals->CriticalPoint > 1 && locals->CriticalPoint < 4) {
3465 locals->ReturnBWPerState[i][0] = dml_min(locals->ReturnBWPerState[i][0],
3466 4 * locals->ReturnBWToDCNPerState *
3467 (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024
3468 * locals->ReturnBusWidth * locals->DCFCLKPerState[i] * locals->UrgentLatency /
3469 dml_pow((locals->ReturnBWToDCNPerState * locals->UrgentLatency
3470 + (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024), 2));
3471 }
3472
3473 locals->ReturnBWToDCNPerState = dml_min(locals->ReturnBusWidth *
3474 locals->DCFCLKPerState[i], locals->FabricAndDRAMBandwidthPerState[i] * 1000);
3475
3476 if (locals->DCCEnabledInAnyPlane == true && locals->ReturnBWToDCNPerState > locals->DCFCLKPerState[i] * locals->ReturnBusWidth / 4) {
3477 locals->ReturnBWPerState[i][0] = dml_min(locals->ReturnBWPerState[i][0],
3478 locals->ReturnBWToDCNPerState * 4 * (1 - locals->UrgentLatency /
3479 ((locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024
3480 / (locals->ReturnBWToDCNPerState - locals->DCFCLKPerState[i]
3481 * locals->ReturnBusWidth / 4) + locals->UrgentLatency)));
3482 }
3483 locals->CriticalPoint = 2 * locals->ReturnBusWidth * locals->DCFCLKPerState[i] *
3484 locals->UrgentLatency / (locals->ReturnBWToDCNPerState * locals->UrgentLatency
3485 + (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024);
3486
3487 if (locals->DCCEnabledInAnyPlane && locals->CriticalPoint > 1 && locals->CriticalPoint < 4) {
3488 locals->ReturnBWPerState[i][0] = dml_min(locals->ReturnBWPerState[i][0],
3489 4 * locals->ReturnBWToDCNPerState *
3490 (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024
3491 * locals->ReturnBusWidth * locals->DCFCLKPerState[i] * locals->UrgentLatency /
3492 dml_pow((locals->ReturnBWToDCNPerState * locals->UrgentLatency
3493 + (locals->ROBBufferSizeInKByte - locals->PixelChunkSizeInKByte) * 1024), 2));
3494 }
3495 }
3496 /*Writeback Latency support check*/
3497
3498 mode_lib->vba.WritebackLatencySupport = true;
3499 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3500 if (mode_lib->vba.WritebackEnable[k] == true) {
3501 if (mode_lib->vba.WritebackPixelFormat[k] == dm_444_32) {
3502 if (locals->WriteBandwidth[k]
3503 > (mode_lib->vba.WritebackInterfaceLumaBufferSize
3504 + mode_lib->vba.WritebackInterfaceChromaBufferSize)
3505 / mode_lib->vba.WritebackLatency) {
3506 mode_lib->vba.WritebackLatencySupport = false;
3507 }
3508 } else {
3509 if (locals->WriteBandwidth[k]
3510 > 1.5
3511 * dml_min(
3512 mode_lib->vba.WritebackInterfaceLumaBufferSize,
3513 2.0
3514 * mode_lib->vba.WritebackInterfaceChromaBufferSize)
3515 / mode_lib->vba.WritebackLatency) {
3516 mode_lib->vba.WritebackLatencySupport = false;
3517 }
3518 }
3519 }
3520 }
3521 /*Re-ordering Buffer Support Check*/
3522
3523 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
3524 locals->UrgentRoundTripAndOutOfOrderLatencyPerState[i] =
3525 (mode_lib->vba.RoundTripPingLatencyCycles + 32.0) / mode_lib->vba.DCFCLKPerState[i]
3526 + locals->UrgentOutOfOrderReturnPerChannel * mode_lib->vba.NumberOfChannels / locals->ReturnBWPerState[i][0];
3527 if ((mode_lib->vba.ROBBufferSizeInKByte - mode_lib->vba.PixelChunkSizeInKByte) * 1024.0 / locals->ReturnBWPerState[i][0]
3528 > locals->UrgentRoundTripAndOutOfOrderLatencyPerState[i]) {
3529 locals->ROBSupport[i][0] = true;
3530 } else {
3531 locals->ROBSupport[i][0] = false;
3532 }
3533 }
3534 /*Writeback Mode Support Check*/
3535
3536 mode_lib->vba.TotalNumberOfActiveWriteback = 0;
3537 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3538 if (mode_lib->vba.WritebackEnable[k] == true) {
3539 if (mode_lib->vba.ActiveWritebacksPerPlane[k] == 0)
3540 mode_lib->vba.ActiveWritebacksPerPlane[k] = 1;
3541 mode_lib->vba.TotalNumberOfActiveWriteback =
3542 mode_lib->vba.TotalNumberOfActiveWriteback
3543 + mode_lib->vba.ActiveWritebacksPerPlane[k];
3544 }
3545 }
3546 mode_lib->vba.WritebackModeSupport = true;
3547 if (mode_lib->vba.TotalNumberOfActiveWriteback > mode_lib->vba.MaxNumWriteback) {
3548 mode_lib->vba.WritebackModeSupport = false;
3549 }
3550 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3551 if (mode_lib->vba.WritebackEnable[k] == true
3552 && mode_lib->vba.Writeback10bpc420Supported != true
3553 && mode_lib->vba.WritebackPixelFormat[k] == dm_420_10) {
3554 mode_lib->vba.WritebackModeSupport = false;
3555 }
3556 }
3557 /*Writeback Scale Ratio and Taps Support Check*/
3558
3559 mode_lib->vba.WritebackScaleRatioAndTapsSupport = true;
3560 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3561 if (mode_lib->vba.WritebackEnable[k] == true) {
3562 if (mode_lib->vba.WritebackLumaAndChromaScalingSupported == false
3563 && (mode_lib->vba.WritebackHRatio[k] != 1.0
3564 || mode_lib->vba.WritebackVRatio[k] != 1.0)) {
3565 mode_lib->vba.WritebackScaleRatioAndTapsSupport = false;
3566 }
3567 if (mode_lib->vba.WritebackHRatio[k] > mode_lib->vba.WritebackMaxHSCLRatio
3568 || mode_lib->vba.WritebackVRatio[k]
3569 > mode_lib->vba.WritebackMaxVSCLRatio
3570 || mode_lib->vba.WritebackHRatio[k]
3571 < mode_lib->vba.WritebackMinHSCLRatio
3572 || mode_lib->vba.WritebackVRatio[k]
3573 < mode_lib->vba.WritebackMinVSCLRatio
3574 || mode_lib->vba.WritebackLumaHTaps[k]
3575 > mode_lib->vba.WritebackMaxHSCLTaps
3576 || mode_lib->vba.WritebackLumaVTaps[k]
3577 > mode_lib->vba.WritebackMaxVSCLTaps
3578 || mode_lib->vba.WritebackHRatio[k]
3579 > mode_lib->vba.WritebackLumaHTaps[k]
3580 || mode_lib->vba.WritebackVRatio[k]
3581 > mode_lib->vba.WritebackLumaVTaps[k]
3582 || (mode_lib->vba.WritebackLumaHTaps[k] > 2.0
3583 && ((mode_lib->vba.WritebackLumaHTaps[k] % 2)
3584 == 1))
3585 || (mode_lib->vba.WritebackPixelFormat[k] != dm_444_32
3586 && (mode_lib->vba.WritebackChromaHTaps[k]
3587 > mode_lib->vba.WritebackMaxHSCLTaps
3588 || mode_lib->vba.WritebackChromaVTaps[k]
3589 > mode_lib->vba.WritebackMaxVSCLTaps
3590 || 2.0
3591 * mode_lib->vba.WritebackHRatio[k]
3592 > mode_lib->vba.WritebackChromaHTaps[k]
3593 || 2.0
3594 * mode_lib->vba.WritebackVRatio[k]
3595 > mode_lib->vba.WritebackChromaVTaps[k]
3596 || (mode_lib->vba.WritebackChromaHTaps[k] > 2.0
3597 && ((mode_lib->vba.WritebackChromaHTaps[k] % 2) == 1))))) {
3598 mode_lib->vba.WritebackScaleRatioAndTapsSupport = false;
3599 }
3600 if (mode_lib->vba.WritebackVRatio[k] < 1.0) {
3601 mode_lib->vba.WritebackLumaVExtra =
3602 dml_max(1.0 - 2.0 / dml_ceil(1.0 / mode_lib->vba.WritebackVRatio[k], 1.0), 0.0);
3603 } else {
3604 mode_lib->vba.WritebackLumaVExtra = -1;
3605 }
3606 if ((mode_lib->vba.WritebackPixelFormat[k] == dm_444_32
3607 && mode_lib->vba.WritebackLumaVTaps[k]
3608 > (mode_lib->vba.WritebackLineBufferLumaBufferSize
3609 + mode_lib->vba.WritebackLineBufferChromaBufferSize)
3610 / 3.0
3611 / mode_lib->vba.WritebackDestinationWidth[k]
3612 - mode_lib->vba.WritebackLumaVExtra)
3613 || (mode_lib->vba.WritebackPixelFormat[k] == dm_420_8
3614 && mode_lib->vba.WritebackLumaVTaps[k]
3615 > mode_lib->vba.WritebackLineBufferLumaBufferSize
3616 * 8.0 / 10.0 / mode_lib->vba.WritebackDestinationWidth[k]
3617 - mode_lib->vba.WritebackLumaVExtra)
3618 || (mode_lib->vba.WritebackPixelFormat[k] == dm_420_10
3619 && mode_lib->vba.WritebackLumaVTaps[k]
3620 > mode_lib->vba.WritebackLineBufferLumaBufferSize
3621 * 8.0 / 10.0
3622 / mode_lib->vba.WritebackDestinationWidth[k]
3623 - mode_lib->vba.WritebackLumaVExtra)) {
3624 mode_lib->vba.WritebackScaleRatioAndTapsSupport = false;
3625 }
3626 if (2.0 * mode_lib->vba.WritebackVRatio[k] < 1) {
3627 mode_lib->vba.WritebackChromaVExtra = 0.0;
3628 } else {
3629 mode_lib->vba.WritebackChromaVExtra = -1;
3630 }
3631 if ((mode_lib->vba.WritebackPixelFormat[k] == dm_420_8
3632 && mode_lib->vba.WritebackChromaVTaps[k]
3633 > mode_lib->vba.WritebackLineBufferChromaBufferSize
3634 * 8.0 / 10.0 / mode_lib->vba.WritebackDestinationWidth[k]
3635 - mode_lib->vba.WritebackChromaVExtra)
3636 || (mode_lib->vba.WritebackPixelFormat[k] == dm_420_10
3637 && mode_lib->vba.WritebackChromaVTaps[k]
3638 > mode_lib->vba.WritebackLineBufferChromaBufferSize
3639 * 8.0 / 10.0
3640 / mode_lib->vba.WritebackDestinationWidth[k]
3641 - mode_lib->vba.WritebackChromaVExtra)) {
3642 mode_lib->vba.WritebackScaleRatioAndTapsSupport = false;
3643 }
3644 }
3645 }
3646 /*Maximum DISPCLK/DPPCLK Support check*/
3647
3648 mode_lib->vba.WritebackRequiredDISPCLK = 0.0;
3649 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3650 if (mode_lib->vba.WritebackEnable[k] == true) {
3651 mode_lib->vba.WritebackRequiredDISPCLK =
3652 dml_max(
3653 mode_lib->vba.WritebackRequiredDISPCLK,
3654 CalculateWriteBackDISPCLK(
3655 mode_lib->vba.WritebackPixelFormat[k],
3656 mode_lib->vba.PixelClock[k],
3657 mode_lib->vba.WritebackHRatio[k],
3658 mode_lib->vba.WritebackVRatio[k],
3659 mode_lib->vba.WritebackLumaHTaps[k],
3660 mode_lib->vba.WritebackLumaVTaps[k],
3661 mode_lib->vba.WritebackChromaHTaps[k],
3662 mode_lib->vba.WritebackChromaVTaps[k],
3663 mode_lib->vba.WritebackDestinationWidth[k],
3664 mode_lib->vba.HTotal[k],
3665 mode_lib->vba.WritebackChromaLineBufferWidth));
3666 }
3667 }
3668 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3669 if (mode_lib->vba.HRatio[k] > 1.0) {
3670 locals->PSCL_FACTOR[k] = dml_min(
3671 mode_lib->vba.MaxDCHUBToPSCLThroughput,
3672 mode_lib->vba.MaxPSCLToLBThroughput
3673 * mode_lib->vba.HRatio[k]
3674 / dml_ceil(
3675 mode_lib->vba.htaps[k]
3676 / 6.0,
3677 1.0));
3678 } else {
3679 locals->PSCL_FACTOR[k] = dml_min(
3680 mode_lib->vba.MaxDCHUBToPSCLThroughput,
3681 mode_lib->vba.MaxPSCLToLBThroughput);
3682 }
3683 if (locals->BytePerPixelInDETC[k] == 0.0) {
3684 locals->PSCL_FACTOR_CHROMA[k] = 0.0;
3685 locals->MinDPPCLKUsingSingleDPP[k] =
3686 mode_lib->vba.PixelClock[k]
3687 * dml_max3(
3688 mode_lib->vba.vtaps[k] / 6.0
3689 * dml_min(
3690 1.0,
3691 mode_lib->vba.HRatio[k]),
3692 mode_lib->vba.HRatio[k]
3693 * mode_lib->vba.VRatio[k]
3694 / locals->PSCL_FACTOR[k],
3695 1.0);
3696 if ((mode_lib->vba.htaps[k] > 6.0 || mode_lib->vba.vtaps[k] > 6.0)
3697 && locals->MinDPPCLKUsingSingleDPP[k]
3698 < 2.0 * mode_lib->vba.PixelClock[k]) {
3699 locals->MinDPPCLKUsingSingleDPP[k] = 2.0
3700 * mode_lib->vba.PixelClock[k];
3701 }
3702 } else {
3703 if (mode_lib->vba.HRatio[k] / 2.0 > 1.0) {
3704 locals->PSCL_FACTOR_CHROMA[k] =
3705 dml_min(
3706 mode_lib->vba.MaxDCHUBToPSCLThroughput,
3707 mode_lib->vba.MaxPSCLToLBThroughput
3708 * mode_lib->vba.HRatio[k]
3709 / 2.0
3710 / dml_ceil(
3711 mode_lib->vba.HTAPsChroma[k]
3712 / 6.0,
3713 1.0));
3714 } else {
3715 locals->PSCL_FACTOR_CHROMA[k] = dml_min(
3716 mode_lib->vba.MaxDCHUBToPSCLThroughput,
3717 mode_lib->vba.MaxPSCLToLBThroughput);
3718 }
3719 locals->MinDPPCLKUsingSingleDPP[k] =
3720 mode_lib->vba.PixelClock[k]
3721 * dml_max5(
3722 mode_lib->vba.vtaps[k] / 6.0
3723 * dml_min(
3724 1.0,
3725 mode_lib->vba.HRatio[k]),
3726 mode_lib->vba.HRatio[k]
3727 * mode_lib->vba.VRatio[k]
3728 / locals->PSCL_FACTOR[k],
3729 mode_lib->vba.VTAPsChroma[k]
3730 / 6.0
3731 * dml_min(
3732 1.0,
3733 mode_lib->vba.HRatio[k]
3734 / 2.0),
3735 mode_lib->vba.HRatio[k]
3736 * mode_lib->vba.VRatio[k]
3737 / 4.0
3738 / locals->PSCL_FACTOR_CHROMA[k],
3739 1.0);
3740 if ((mode_lib->vba.htaps[k] > 6.0 || mode_lib->vba.vtaps[k] > 6.0
3741 || mode_lib->vba.HTAPsChroma[k] > 6.0
3742 || mode_lib->vba.VTAPsChroma[k] > 6.0)
3743 && locals->MinDPPCLKUsingSingleDPP[k]
3744 < 2.0 * mode_lib->vba.PixelClock[k]) {
3745 locals->MinDPPCLKUsingSingleDPP[k] = 2.0
3746 * mode_lib->vba.PixelClock[k];
3747 }
3748 }
3749 }
3750 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3751 Calculate256BBlockSizes(
3752 mode_lib->vba.SourcePixelFormat[k],
3753 mode_lib->vba.SurfaceTiling[k],
3754 dml_ceil(locals->BytePerPixelInDETY[k], 1.0),
3755 dml_ceil(locals->BytePerPixelInDETC[k], 2.0),
3756 &locals->Read256BlockHeightY[k],
3757 &locals->Read256BlockHeightC[k],
3758 &locals->Read256BlockWidthY[k],
3759 &locals->Read256BlockWidthC[k]);
3760 if (mode_lib->vba.SourceScan[k] == dm_horz) {
3761 locals->MaxSwathHeightY[k] = locals->Read256BlockHeightY[k];
3762 locals->MaxSwathHeightC[k] = locals->Read256BlockHeightC[k];
3763 } else {
3764 locals->MaxSwathHeightY[k] = locals->Read256BlockWidthY[k];
3765 locals->MaxSwathHeightC[k] = locals->Read256BlockWidthC[k];
3766 }
3767 if ((mode_lib->vba.SourcePixelFormat[k] == dm_444_64
3768 || mode_lib->vba.SourcePixelFormat[k] == dm_444_32
3769 || mode_lib->vba.SourcePixelFormat[k] == dm_444_16
3770 || mode_lib->vba.SourcePixelFormat[k] == dm_mono_16
3771 || mode_lib->vba.SourcePixelFormat[k] == dm_mono_8)) {
3772 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear
3773 || (mode_lib->vba.SourcePixelFormat[k] == dm_444_64
3774 && (mode_lib->vba.SurfaceTiling[k]
3775 == dm_sw_4kb_s
3776 || mode_lib->vba.SurfaceTiling[k]
3777 == dm_sw_4kb_s_x
3778 || mode_lib->vba.SurfaceTiling[k]
3779 == dm_sw_64kb_s
3780 || mode_lib->vba.SurfaceTiling[k]
3781 == dm_sw_64kb_s_t
3782 || mode_lib->vba.SurfaceTiling[k]
3783 == dm_sw_64kb_s_x
3784 || mode_lib->vba.SurfaceTiling[k]
3785 == dm_sw_var_s
3786 || mode_lib->vba.SurfaceTiling[k]
3787 == dm_sw_var_s_x)
3788 && mode_lib->vba.SourceScan[k] == dm_horz)) {
3789 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k];
3790 } else {
3791 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k]
3792 / 2.0;
3793 }
3794 locals->MinSwathHeightC[k] = locals->MaxSwathHeightC[k];
3795 } else {
3796 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear) {
3797 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k];
3798 locals->MinSwathHeightC[k] = locals->MaxSwathHeightC[k];
3799 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_8
3800 && mode_lib->vba.SourceScan[k] == dm_horz) {
3801 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k]
3802 / 2.0;
3803 locals->MinSwathHeightC[k] = locals->MaxSwathHeightC[k];
3804 } else if (mode_lib->vba.SourcePixelFormat[k] == dm_420_10
3805 && mode_lib->vba.SourceScan[k] == dm_horz) {
3806 locals->MinSwathHeightC[k] = locals->MaxSwathHeightC[k]
3807 / 2.0;
3808 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k];
3809 } else {
3810 locals->MinSwathHeightY[k] = locals->MaxSwathHeightY[k];
3811 locals->MinSwathHeightC[k] = locals->MaxSwathHeightC[k];
3812 }
3813 }
3814 if (mode_lib->vba.SurfaceTiling[k] == dm_sw_linear) {
3815 mode_lib->vba.MaximumSwathWidthSupport = 8192.0;
3816 } else {
3817 mode_lib->vba.MaximumSwathWidthSupport = 5120.0;
3818 }
3819 mode_lib->vba.MaximumSwathWidthInDETBuffer =
3820 dml_min(
3821 mode_lib->vba.MaximumSwathWidthSupport,
3822 mode_lib->vba.DETBufferSizeInKByte[0] * 1024.0 / 2.0
3823 / (locals->BytePerPixelInDETY[k]
3824 * locals->MinSwathHeightY[k]
3825 + locals->BytePerPixelInDETC[k]
3826 / 2.0
3827 * locals->MinSwathHeightC[k]));
3828 if (locals->BytePerPixelInDETC[k] == 0.0) {
3829 mode_lib->vba.MaximumSwathWidthInLineBuffer =
3830 mode_lib->vba.LineBufferSize
3831 * dml_max(mode_lib->vba.HRatio[k], 1.0)
3832 / mode_lib->vba.LBBitPerPixel[k]
3833 / (mode_lib->vba.vtaps[k]
3834 + dml_max(
3835 dml_ceil(
3836 mode_lib->vba.VRatio[k],
3837 1.0)
3838 - 2,
3839 0.0));
3840 } else {
3841 mode_lib->vba.MaximumSwathWidthInLineBuffer =
3842 dml_min(
3843 mode_lib->vba.LineBufferSize
3844 * dml_max(
3845 mode_lib->vba.HRatio[k],
3846 1.0)
3847 / mode_lib->vba.LBBitPerPixel[k]
3848 / (mode_lib->vba.vtaps[k]
3849 + dml_max(
3850 dml_ceil(
3851 mode_lib->vba.VRatio[k],
3852 1.0)
3853 - 2,
3854 0.0)),
3855 2.0 * mode_lib->vba.LineBufferSize
3856 * dml_max(
3857 mode_lib->vba.HRatio[k]
3858 / 2.0,
3859 1.0)
3860 / mode_lib->vba.LBBitPerPixel[k]
3861 / (mode_lib->vba.VTAPsChroma[k]
3862 + dml_max(
3863 dml_ceil(
3864 mode_lib->vba.VRatio[k]
3865 / 2.0,
3866 1.0)
3867 - 2,
3868 0.0)));
3869 }
3870 locals->MaximumSwathWidth[k] = dml_min(
3871 mode_lib->vba.MaximumSwathWidthInDETBuffer,
3872 mode_lib->vba.MaximumSwathWidthInLineBuffer);
3873 }
3874 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
3875 for (j = 0; j < 2; j++) {
3876 mode_lib->vba.MaxDispclkRoundedDownToDFSGranularity = RoundToDFSGranularityDown(
3877 mode_lib->vba.MaxDispclk[i],
3878 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
3879 mode_lib->vba.MaxDppclkRoundedDownToDFSGranularity = RoundToDFSGranularityDown(
3880 mode_lib->vba.MaxDppclk[i],
3881 mode_lib->vba.DISPCLKDPPCLKVCOSpeed);
3882 locals->RequiredDISPCLK[i][j] = 0.0;
3883 locals->DISPCLK_DPPCLK_Support[i][j] = true;
3884 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3885 mode_lib->vba.PlaneRequiredDISPCLKWithoutODMCombine =
3886 mode_lib->vba.PixelClock[k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0)
3887 * (1.0 + mode_lib->vba.DISPCLKRampingMargin / 100.0);
3888 if (mode_lib->vba.PlaneRequiredDISPCLKWithoutODMCombine >= mode_lib->vba.MaxDispclk[i]
3889 && i == mode_lib->vba.soc.num_states)
3890 mode_lib->vba.PlaneRequiredDISPCLKWithoutODMCombine = mode_lib->vba.PixelClock[k]
3891 * (1 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0);
3892
3893 mode_lib->vba.PlaneRequiredDISPCLKWithODMCombine = mode_lib->vba.PixelClock[k] / 2
3894 * (1 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) * (1 + mode_lib->vba.DISPCLKRampingMargin / 100.0);
3895 if (mode_lib->vba.PlaneRequiredDISPCLKWithODMCombine >= mode_lib->vba.MaxDispclk[i]
3896 && i == mode_lib->vba.soc.num_states)
3897 mode_lib->vba.PlaneRequiredDISPCLKWithODMCombine = mode_lib->vba.PixelClock[k] / 2
3898 * (1 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0);
3899
3900 locals->ODMCombineEnablePerState[i][k] = dm_odm_combine_mode_disabled;
3901 mode_lib->vba.PlaneRequiredDISPCLK = mode_lib->vba.PlaneRequiredDISPCLKWithoutODMCombine;
3902 if (mode_lib->vba.ODMCapability) {
3903 if (locals->PlaneRequiredDISPCLKWithoutODMCombine > mode_lib->vba.MaxDispclkRoundedDownToDFSGranularity) {
3904 locals->ODMCombineEnablePerState[i][k] = dm_odm_combine_mode_2to1;
3905 mode_lib->vba.PlaneRequiredDISPCLK = mode_lib->vba.PlaneRequiredDISPCLKWithODMCombine;
3906 } else if (locals->HActive[k] > DCN20_MAX_420_IMAGE_WIDTH && locals->OutputFormat[k] == dm_420) {
3907 locals->ODMCombineEnablePerState[i][k] = dm_odm_combine_mode_2to1;
3908 mode_lib->vba.PlaneRequiredDISPCLK = mode_lib->vba.PlaneRequiredDISPCLKWithODMCombine;
3909 }
3910 }
3911
3912 if (locals->MinDPPCLKUsingSingleDPP[k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) <= mode_lib->vba.MaxDppclkRoundedDownToDFSGranularity
3913 && locals->SwathWidthYSingleDPP[k] <= locals->MaximumSwathWidth[k]
3914 && locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_disabled) {
3915 locals->NoOfDPP[i][j][k] = 1;
3916 locals->RequiredDPPCLK[i][j][k] =
3917 locals->MinDPPCLKUsingSingleDPP[k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0);
3918 } else {
3919 locals->NoOfDPP[i][j][k] = 2;
3920 locals->RequiredDPPCLK[i][j][k] =
3921 locals->MinDPPCLKUsingSingleDPP[k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) / 2.0;
3922 }
3923 locals->RequiredDISPCLK[i][j] = dml_max(
3924 locals->RequiredDISPCLK[i][j],
3925 mode_lib->vba.PlaneRequiredDISPCLK);
3926 if ((locals->MinDPPCLKUsingSingleDPP[k] / locals->NoOfDPP[i][j][k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0)
3927 > mode_lib->vba.MaxDppclkRoundedDownToDFSGranularity)
3928 || (mode_lib->vba.PlaneRequiredDISPCLK > mode_lib->vba.MaxDispclkRoundedDownToDFSGranularity)) {
3929 locals->DISPCLK_DPPCLK_Support[i][j] = false;
3930 }
3931 }
3932 locals->TotalNumberOfActiveDPP[i][j] = 0.0;
3933 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++)
3934 locals->TotalNumberOfActiveDPP[i][j] = locals->TotalNumberOfActiveDPP[i][j] + locals->NoOfDPP[i][j][k];
3935 if (j == 1) {
3936 while (locals->TotalNumberOfActiveDPP[i][j] < mode_lib->vba.MaxNumDPP
3937 && locals->TotalNumberOfActiveDPP[i][j] < 2 * mode_lib->vba.NumberOfActivePlanes) {
3938 double BWOfNonSplitPlaneOfMaximumBandwidth;
3939 unsigned int NumberOfNonSplitPlaneOfMaximumBandwidth;
3940
3941 BWOfNonSplitPlaneOfMaximumBandwidth = 0;
3942 NumberOfNonSplitPlaneOfMaximumBandwidth = 0;
3943 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; ++k) {
3944 if (locals->ReadBandwidth[k] > BWOfNonSplitPlaneOfMaximumBandwidth && locals->NoOfDPP[i][j][k] == 1) {
3945 BWOfNonSplitPlaneOfMaximumBandwidth = locals->ReadBandwidth[k];
3946 NumberOfNonSplitPlaneOfMaximumBandwidth = k;
3947 }
3948 }
3949 locals->NoOfDPP[i][j][NumberOfNonSplitPlaneOfMaximumBandwidth] = 2;
3950 locals->RequiredDPPCLK[i][j][NumberOfNonSplitPlaneOfMaximumBandwidth] =
3951 locals->MinDPPCLKUsingSingleDPP[NumberOfNonSplitPlaneOfMaximumBandwidth]
3952 * (1 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100) / 2;
3953 locals->TotalNumberOfActiveDPP[i][j] = locals->TotalNumberOfActiveDPP[i][j] + 1;
3954 }
3955 }
3956 if (locals->TotalNumberOfActiveDPP[i][j] > mode_lib->vba.MaxNumDPP) {
3957 locals->RequiredDISPCLK[i][j] = 0.0;
3958 locals->DISPCLK_DPPCLK_Support[i][j] = true;
3959 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
3960 locals->ODMCombineEnablePerState[i][k] = dm_odm_combine_mode_disabled;
3961 if (locals->SwathWidthYSingleDPP[k] <= locals->MaximumSwathWidth[k]) {
3962 locals->NoOfDPP[i][j][k] = 1;
3963 locals->RequiredDPPCLK[i][j][k] = locals->MinDPPCLKUsingSingleDPP[k]
3964 * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0);
3965 } else {
3966 locals->NoOfDPP[i][j][k] = 2;
3967 locals->RequiredDPPCLK[i][j][k] = locals->MinDPPCLKUsingSingleDPP[k]
3968 * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) / 2.0;
3969 }
3970 if (i != mode_lib->vba.soc.num_states) {
3971 mode_lib->vba.PlaneRequiredDISPCLK =
3972 mode_lib->vba.PixelClock[k]
3973 * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0)
3974 * (1.0 + mode_lib->vba.DISPCLKRampingMargin / 100.0);
3975 } else {
3976 mode_lib->vba.PlaneRequiredDISPCLK = mode_lib->vba.PixelClock[k]
3977 * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0);
3978 }
3979 locals->RequiredDISPCLK[i][j] = dml_max(
3980 locals->RequiredDISPCLK[i][j],
3981 mode_lib->vba.PlaneRequiredDISPCLK);
3982 if (locals->MinDPPCLKUsingSingleDPP[k] / locals->NoOfDPP[i][j][k] * (1.0 + mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0)
3983 > mode_lib->vba.MaxDppclkRoundedDownToDFSGranularity
3984 || mode_lib->vba.PlaneRequiredDISPCLK > mode_lib->vba.MaxDispclkRoundedDownToDFSGranularity)
3985 locals->DISPCLK_DPPCLK_Support[i][j] = false;
3986 }
3987 locals->TotalNumberOfActiveDPP[i][j] = 0.0;
3988 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++)
3989 locals->TotalNumberOfActiveDPP[i][j] = locals->TotalNumberOfActiveDPP[i][j] + locals->NoOfDPP[i][j][k];
3990 }
3991 locals->RequiredDISPCLK[i][j] = dml_max(
3992 locals->RequiredDISPCLK[i][j],
3993 mode_lib->vba.WritebackRequiredDISPCLK);
3994 if (mode_lib->vba.MaxDispclkRoundedDownToDFSGranularity
3995 < mode_lib->vba.WritebackRequiredDISPCLK) {
3996 locals->DISPCLK_DPPCLK_Support[i][j] = false;
3997 }
3998 }
3999 }
4000 /*Viewport Size Check*/
4001
4002 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4003 locals->ViewportSizeSupport[i][0] = true;
4004 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4005 if (locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_2to1) {
4006 if (dml_min(locals->SwathWidthYSingleDPP[k], dml_round(mode_lib->vba.HActive[k] / 2.0 * mode_lib->vba.HRatio[k]))
4007 > locals->MaximumSwathWidth[k]) {
4008 locals->ViewportSizeSupport[i][0] = false;
4009 }
4010 } else {
4011 if (locals->SwathWidthYSingleDPP[k] / 2.0 > locals->MaximumSwathWidth[k]) {
4012 locals->ViewportSizeSupport[i][0] = false;
4013 }
4014 }
4015 }
4016 }
4017 /*Total Available Pipes Support Check*/
4018
4019 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4020 for (j = 0; j < 2; j++) {
4021 if (locals->TotalNumberOfActiveDPP[i][j] <= mode_lib->vba.MaxNumDPP)
4022 locals->TotalAvailablePipesSupport[i][j] = true;
4023 else
4024 locals->TotalAvailablePipesSupport[i][j] = false;
4025 }
4026 }
4027 /*Total Available OTG Support Check*/
4028
4029 mode_lib->vba.TotalNumberOfActiveOTG = 0.0;
4030 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4031 if (mode_lib->vba.BlendingAndTiming[k] == k) {
4032 mode_lib->vba.TotalNumberOfActiveOTG = mode_lib->vba.TotalNumberOfActiveOTG
4033 + 1.0;
4034 }
4035 }
4036 if (mode_lib->vba.TotalNumberOfActiveOTG <= mode_lib->vba.MaxNumOTG) {
4037 mode_lib->vba.NumberOfOTGSupport = true;
4038 } else {
4039 mode_lib->vba.NumberOfOTGSupport = false;
4040 }
4041 /*Display IO and DSC Support Check*/
4042
4043 mode_lib->vba.NonsupportedDSCInputBPC = false;
4044 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4045 if (!(mode_lib->vba.DSCInputBitPerComponent[k] == 12.0
4046 || mode_lib->vba.DSCInputBitPerComponent[k] == 10.0
4047 || mode_lib->vba.DSCInputBitPerComponent[k] == 8.0)) {
4048 mode_lib->vba.NonsupportedDSCInputBPC = true;
4049 }
4050 }
4051 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4052 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4053 locals->RequiresDSC[i][k] = 0;
4054 locals->RequiresFEC[i][k] = 0;
4055 if (mode_lib->vba.BlendingAndTiming[k] == k) {
4056 if (mode_lib->vba.Output[k] == dm_hdmi) {
4057 locals->RequiresDSC[i][k] = 0;
4058 locals->RequiresFEC[i][k] = 0;
4059 locals->OutputBppPerState[i][k] = TruncToValidBPP(
4060 dml_min(600.0, mode_lib->vba.PHYCLKPerState[i]) / mode_lib->vba.PixelClockBackEnd[k] * 24,
4061 false,
4062 mode_lib->vba.Output[k],
4063 mode_lib->vba.OutputFormat[k],
4064 mode_lib->vba.DSCInputBitPerComponent[k]);
4065 } else if (mode_lib->vba.Output[k] == dm_dp
4066 || mode_lib->vba.Output[k] == dm_edp) {
4067 if (mode_lib->vba.Output[k] == dm_edp) {
4068 mode_lib->vba.EffectiveFECOverhead = 0.0;
4069 } else {
4070 mode_lib->vba.EffectiveFECOverhead =
4071 mode_lib->vba.FECOverhead;
4072 }
4073 if (mode_lib->vba.PHYCLKPerState[i] >= 270.0) {
4074 mode_lib->vba.Outbpp = TruncToValidBPP(
4075 (1.0 - mode_lib->vba.Downspreading / 100.0) * 270.0
4076 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4077 false,
4078 mode_lib->vba.Output[k],
4079 mode_lib->vba.OutputFormat[k],
4080 mode_lib->vba.DSCInputBitPerComponent[k]);
4081 mode_lib->vba.OutbppDSC = TruncToValidBPP(
4082 (1.0 - mode_lib->vba.Downspreading / 100.0) * (1.0 - mode_lib->vba.EffectiveFECOverhead / 100.0) * 270.0
4083 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4084 true,
4085 mode_lib->vba.Output[k],
4086 mode_lib->vba.OutputFormat[k],
4087 mode_lib->vba.DSCInputBitPerComponent[k]);
4088 if (mode_lib->vba.DSCEnabled[k] == true) {
4089 locals->RequiresDSC[i][k] = true;
4090 if (mode_lib->vba.Output[k] == dm_dp) {
4091 locals->RequiresFEC[i][k] = true;
4092 } else {
4093 locals->RequiresFEC[i][k] = false;
4094 }
4095 mode_lib->vba.Outbpp = mode_lib->vba.OutbppDSC;
4096 } else {
4097 locals->RequiresDSC[i][k] = false;
4098 locals->RequiresFEC[i][k] = false;
4099 }
4100 locals->OutputBppPerState[i][k] = mode_lib->vba.Outbpp;
4101 }
4102 if (mode_lib->vba.Outbpp == BPP_INVALID && mode_lib->vba.PHYCLKPerState[i] >= 540.0) {
4103 mode_lib->vba.Outbpp = TruncToValidBPP(
4104 (1.0 - mode_lib->vba.Downspreading / 100.0) * 540.0
4105 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4106 false,
4107 mode_lib->vba.Output[k],
4108 mode_lib->vba.OutputFormat[k],
4109 mode_lib->vba.DSCInputBitPerComponent[k]);
4110 mode_lib->vba.OutbppDSC = TruncToValidBPP(
4111 (1.0 - mode_lib->vba.Downspreading / 100.0) * (1.0 - mode_lib->vba.EffectiveFECOverhead / 100.0) * 540.0
4112 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4113 true,
4114 mode_lib->vba.Output[k],
4115 mode_lib->vba.OutputFormat[k],
4116 mode_lib->vba.DSCInputBitPerComponent[k]);
4117 if (mode_lib->vba.DSCEnabled[k] == true) {
4118 locals->RequiresDSC[i][k] = true;
4119 if (mode_lib->vba.Output[k] == dm_dp) {
4120 locals->RequiresFEC[i][k] = true;
4121 } else {
4122 locals->RequiresFEC[i][k] = false;
4123 }
4124 mode_lib->vba.Outbpp = mode_lib->vba.OutbppDSC;
4125 } else {
4126 locals->RequiresDSC[i][k] = false;
4127 locals->RequiresFEC[i][k] = false;
4128 }
4129 locals->OutputBppPerState[i][k] = mode_lib->vba.Outbpp;
4130 }
4131 if (mode_lib->vba.Outbpp == BPP_INVALID
4132 && mode_lib->vba.PHYCLKPerState[i]
4133 >= 810.0) {
4134 mode_lib->vba.Outbpp = TruncToValidBPP(
4135 (1.0 - mode_lib->vba.Downspreading / 100.0) * 810.0
4136 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4137 false,
4138 mode_lib->vba.Output[k],
4139 mode_lib->vba.OutputFormat[k],
4140 mode_lib->vba.DSCInputBitPerComponent[k]);
4141 mode_lib->vba.OutbppDSC = TruncToValidBPP(
4142 (1.0 - mode_lib->vba.Downspreading / 100.0) * (1.0 - mode_lib->vba.EffectiveFECOverhead / 100.0) * 810.0
4143 * mode_lib->vba.OutputLinkDPLanes[k] / mode_lib->vba.PixelClockBackEnd[k] * 8.0,
4144 true,
4145 mode_lib->vba.Output[k],
4146 mode_lib->vba.OutputFormat[k],
4147 mode_lib->vba.DSCInputBitPerComponent[k]);
4148 if (mode_lib->vba.DSCEnabled[k] == true || mode_lib->vba.Outbpp == BPP_INVALID) {
4149 locals->RequiresDSC[i][k] = true;
4150 if (mode_lib->vba.Output[k] == dm_dp) {
4151 locals->RequiresFEC[i][k] = true;
4152 } else {
4153 locals->RequiresFEC[i][k] = false;
4154 }
4155 mode_lib->vba.Outbpp = mode_lib->vba.OutbppDSC;
4156 } else {
4157 locals->RequiresDSC[i][k] = false;
4158 locals->RequiresFEC[i][k] = false;
4159 }
4160 locals->OutputBppPerState[i][k] =
4161 mode_lib->vba.Outbpp;
4162 }
4163 }
4164 } else {
4165 locals->OutputBppPerState[i][k] = BPP_BLENDED_PIPE;
4166 }
4167 }
4168 }
4169 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4170 locals->DIOSupport[i] = true;
4171 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4172 if (!mode_lib->vba.skip_dio_check[k]
4173 && (locals->OutputBppPerState[i][k] == BPP_INVALID
4174 || (mode_lib->vba.OutputFormat[k] == dm_420
4175 && mode_lib->vba.Interlace[k] == true
4176 && mode_lib->vba.ProgressiveToInterlaceUnitInOPP == true))) {
4177 locals->DIOSupport[i] = false;
4178 }
4179 }
4180 }
4181 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4182 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4183 locals->DSCCLKRequiredMoreThanSupported[i] = false;
4184 if (mode_lib->vba.BlendingAndTiming[k] == k) {
4185 if ((mode_lib->vba.Output[k] == dm_dp
4186 || mode_lib->vba.Output[k] == dm_edp)) {
4187 if (mode_lib->vba.OutputFormat[k] == dm_420
4188 || mode_lib->vba.OutputFormat[k]
4189 == dm_n422) {
4190 mode_lib->vba.DSCFormatFactor = 2;
4191 } else {
4192 mode_lib->vba.DSCFormatFactor = 1;
4193 }
4194 if (locals->RequiresDSC[i][k] == true) {
4195 if (locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_2to1) {
4196 if (mode_lib->vba.PixelClockBackEnd[k] / 6.0 / mode_lib->vba.DSCFormatFactor
4197 > (1.0 - mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) * mode_lib->vba.MaxDSCCLK[i]) {
4198 locals->DSCCLKRequiredMoreThanSupported[i] =
4199 true;
4200 }
4201 } else {
4202 if (mode_lib->vba.PixelClockBackEnd[k] / 3.0 / mode_lib->vba.DSCFormatFactor
4203 > (1.0 - mode_lib->vba.DISPCLKDPPCLKDSCCLKDownSpreading / 100.0) * mode_lib->vba.MaxDSCCLK[i]) {
4204 locals->DSCCLKRequiredMoreThanSupported[i] =
4205 true;
4206 }
4207 }
4208 }
4209 }
4210 }
4211 }
4212 }
4213 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4214 locals->NotEnoughDSCUnits[i] = false;
4215 mode_lib->vba.TotalDSCUnitsRequired = 0.0;
4216 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4217 if (locals->RequiresDSC[i][k] == true) {
4218 if (locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_2to1) {
4219 mode_lib->vba.TotalDSCUnitsRequired =
4220 mode_lib->vba.TotalDSCUnitsRequired + 2.0;
4221 } else {
4222 mode_lib->vba.TotalDSCUnitsRequired =
4223 mode_lib->vba.TotalDSCUnitsRequired + 1.0;
4224 }
4225 }
4226 }
4227 if (mode_lib->vba.TotalDSCUnitsRequired > mode_lib->vba.NumberOfDSC) {
4228 locals->NotEnoughDSCUnits[i] = true;
4229 }
4230 }
4231 /*DSC Delay per state*/
4232
4233 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4234 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4235 if (mode_lib->vba.BlendingAndTiming[k] != k) {
4236 mode_lib->vba.slices = 0;
4237 } else if (locals->RequiresDSC[i][k] == 0
4238 || locals->RequiresDSC[i][k] == false) {
4239 mode_lib->vba.slices = 0;
4240 } else if (mode_lib->vba.PixelClockBackEnd[k] > 3200.0) {
4241 mode_lib->vba.slices = dml_ceil(
4242 mode_lib->vba.PixelClockBackEnd[k] / 400.0,
4243 4.0);
4244 } else if (mode_lib->vba.PixelClockBackEnd[k] > 1360.0) {
4245 mode_lib->vba.slices = 8.0;
4246 } else if (mode_lib->vba.PixelClockBackEnd[k] > 680.0) {
4247 mode_lib->vba.slices = 4.0;
4248 } else if (mode_lib->vba.PixelClockBackEnd[k] > 340.0) {
4249 mode_lib->vba.slices = 2.0;
4250 } else {
4251 mode_lib->vba.slices = 1.0;
4252 }
4253 if (locals->OutputBppPerState[i][k] == BPP_BLENDED_PIPE
4254 || locals->OutputBppPerState[i][k] == BPP_INVALID) {
4255 mode_lib->vba.bpp = 0.0;
4256 } else {
4257 mode_lib->vba.bpp = locals->OutputBppPerState[i][k];
4258 }
4259 if (locals->RequiresDSC[i][k] == true && mode_lib->vba.bpp != 0.0) {
4260 if (locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_disabled) {
4261 locals->DSCDelayPerState[i][k] =
4262 dscceComputeDelay(
4263 mode_lib->vba.DSCInputBitPerComponent[k],
4264 mode_lib->vba.bpp,
4265 dml_ceil(
4266 mode_lib->vba.HActive[k]
4267 / mode_lib->vba.slices,
4268 1.0),
4269 mode_lib->vba.slices,
4270 mode_lib->vba.OutputFormat[k])
4271 + dscComputeDelay(
4272 mode_lib->vba.OutputFormat[k]);
4273 } else {
4274 locals->DSCDelayPerState[i][k] =
4275 2.0 * (dscceComputeDelay(
4276 mode_lib->vba.DSCInputBitPerComponent[k],
4277 mode_lib->vba.bpp,
4278 dml_ceil(mode_lib->vba.HActive[k] / mode_lib->vba.slices, 1.0),
4279 mode_lib->vba.slices / 2,
4280 mode_lib->vba.OutputFormat[k])
4281 + dscComputeDelay(mode_lib->vba.OutputFormat[k]));
4282 }
4283 locals->DSCDelayPerState[i][k] =
4284 locals->DSCDelayPerState[i][k] * mode_lib->vba.PixelClock[k] / mode_lib->vba.PixelClockBackEnd[k];
4285 } else {
4286 locals->DSCDelayPerState[i][k] = 0.0;
4287 }
4288 }
4289 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4290 for (m = 0; m <= mode_lib->vba.NumberOfActivePlanes - 1; m++) {
4291 for (j = 0; j <= mode_lib->vba.NumberOfActivePlanes - 1; j++) {
4292 if (mode_lib->vba.BlendingAndTiming[k] == m && locals->RequiresDSC[i][m] == true)
4293 locals->DSCDelayPerState[i][k] = locals->DSCDelayPerState[i][m];
4294 }
4295 }
4296 }
4297 }
4298
4299 //Prefetch Check
4300 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4301 for (j = 0; j < 2; j++) {
4302 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4303 if (locals->ODMCombineEnablePerState[i][k] == dm_odm_combine_mode_2to1)
4304 locals->SwathWidthYPerState[i][j][k] = dml_min(locals->SwathWidthYSingleDPP[k], dml_round(locals->HActive[k] / 2 * locals->HRatio[k]));
4305 else
4306 locals->SwathWidthYPerState[i][j][k] = locals->SwathWidthYSingleDPP[k] / locals->NoOfDPP[i][j][k];
4307 locals->SwathWidthGranularityY = 256 / dml_ceil(locals->BytePerPixelInDETY[k], 1) / locals->MaxSwathHeightY[k];
4308 locals->RoundedUpMaxSwathSizeBytesY = (dml_ceil(locals->SwathWidthYPerState[i][j][k] - 1, locals->SwathWidthGranularityY)
4309 + locals->SwathWidthGranularityY) * locals->BytePerPixelInDETY[k] * locals->MaxSwathHeightY[k];
4310 if (locals->SourcePixelFormat[k] == dm_420_10) {
4311 locals->RoundedUpMaxSwathSizeBytesY = dml_ceil(locals->RoundedUpMaxSwathSizeBytesY, 256) + 256;
4312 }
4313 if (locals->MaxSwathHeightC[k] > 0) {
4314 locals->SwathWidthGranularityC = 256 / dml_ceil(locals->BytePerPixelInDETC[k], 2) / locals->MaxSwathHeightC[k];
4315
4316 locals->RoundedUpMaxSwathSizeBytesC = (dml_ceil(locals->SwathWidthYPerState[i][j][k] / 2 - 1, locals->SwathWidthGranularityC)
4317 + locals->SwathWidthGranularityC) * locals->BytePerPixelInDETC[k] * locals->MaxSwathHeightC[k];
4318 }
4319 if (locals->SourcePixelFormat[k] == dm_420_10) {
4320 locals->RoundedUpMaxSwathSizeBytesC = dml_ceil(locals->RoundedUpMaxSwathSizeBytesC, 256) + 256;
4321 } else {
4322 locals->RoundedUpMaxSwathSizeBytesC = 0;
4323 }
4324
4325 if (locals->RoundedUpMaxSwathSizeBytesY + locals->RoundedUpMaxSwathSizeBytesC <= locals->DETBufferSizeInKByte[0] * 1024 / 2) {
4326 locals->SwathHeightYPerState[i][j][k] = locals->MaxSwathHeightY[k];
4327 locals->SwathHeightCPerState[i][j][k] = locals->MaxSwathHeightC[k];
4328 } else {
4329 locals->SwathHeightYPerState[i][j][k] = locals->MinSwathHeightY[k];
4330 locals->SwathHeightCPerState[i][j][k] = locals->MinSwathHeightC[k];
4331 }
4332
4333 if (locals->BytePerPixelInDETC[k] == 0) {
4334 locals->LinesInDETLuma = locals->DETBufferSizeInKByte[0] * 1024 / locals->BytePerPixelInDETY[k] / locals->SwathWidthYPerState[i][j][k];
4335 locals->LinesInDETChroma = 0;
4336 } else if (locals->SwathHeightYPerState[i][j][k] <= locals->SwathHeightCPerState[i][j][k]) {
4337 locals->LinesInDETLuma = locals->DETBufferSizeInKByte[0] * 1024 / 2 / locals->BytePerPixelInDETY[k] /
4338 locals->SwathWidthYPerState[i][j][k];
4339 locals->LinesInDETChroma = locals->DETBufferSizeInKByte[0] * 1024 / 2 / locals->BytePerPixelInDETC[k] / (locals->SwathWidthYPerState[i][j][k] / 2);
4340 } else {
4341 locals->LinesInDETLuma = locals->DETBufferSizeInKByte[0] * 1024 * 2 / 3 / locals->BytePerPixelInDETY[k] / locals->SwathWidthYPerState[i][j][k];
4342 locals->LinesInDETChroma = locals->DETBufferSizeInKByte[0] * 1024 / 3 / locals->BytePerPixelInDETY[k] / (locals->SwathWidthYPerState[i][j][k] / 2);
4343 }
4344
4345 locals->EffectiveLBLatencyHidingSourceLinesLuma = dml_min(locals->MaxLineBufferLines,
4346 dml_floor(locals->LineBufferSize / locals->LBBitPerPixel[k] / (locals->SwathWidthYPerState[i][j][k]
4347 / dml_max(locals->HRatio[k], 1)), 1)) - (locals->vtaps[k] - 1);
4348
4349 locals->EffectiveLBLatencyHidingSourceLinesChroma = dml_min(locals->MaxLineBufferLines,
4350 dml_floor(locals->LineBufferSize / locals->LBBitPerPixel[k]
4351 / (locals->SwathWidthYPerState[i][j][k] / 2
4352 / dml_max(locals->HRatio[k] / 2, 1)), 1)) - (locals->VTAPsChroma[k] - 1);
4353
4354 locals->EffectiveDETLBLinesLuma = dml_floor(locals->LinesInDETLuma + dml_min(
4355 locals->LinesInDETLuma * locals->RequiredDISPCLK[i][j] * locals->BytePerPixelInDETY[k] *
4356 locals->PSCL_FACTOR[k] / locals->ReturnBWPerState[i][0],
4357 locals->EffectiveLBLatencyHidingSourceLinesLuma),
4358 locals->SwathHeightYPerState[i][j][k]);
4359 if (locals->LinesInDETChroma) {
4360 locals->EffectiveDETLBLinesChroma = dml_floor(locals->LinesInDETChroma +
4361 dml_min(locals->LinesInDETChroma * locals->RequiredDISPCLK[i][j] *
4362 locals->BytePerPixelInDETC[k] *
4363 locals->PSCL_FACTOR_CHROMA[k] / locals->ReturnBWPerState[i][0],
4364 locals->EffectiveLBLatencyHidingSourceLinesChroma),
4365 locals->SwathHeightCPerState[i][j][k]);
4366 } else {
4367 locals->EffectiveDETLBLinesChroma = 0;
4368 }
4369
4370 if (locals->BytePerPixelInDETC[k] == 0) {
4371 locals->UrgentLatencySupportUsPerState[i][j][k] = locals->EffectiveDETLBLinesLuma * (locals->HTotal[k] / locals->PixelClock[k])
4372 / locals->VRatio[k] - locals->EffectiveDETLBLinesLuma * locals->SwathWidthYPerState[i][j][k] *
4373 dml_ceil(locals->BytePerPixelInDETY[k], 1) / (locals->ReturnBWPerState[i][0] / locals->NoOfDPP[i][j][k]);
4374 } else {
4375 locals->UrgentLatencySupportUsPerState[i][j][k] = dml_min(
4376 locals->EffectiveDETLBLinesLuma * (locals->HTotal[k] / locals->PixelClock[k])
4377 / locals->VRatio[k] - locals->EffectiveDETLBLinesLuma * locals->SwathWidthYPerState[i][j][k] *
4378 dml_ceil(locals->BytePerPixelInDETY[k], 1) / (locals->ReturnBWPerState[i][0] / locals->NoOfDPP[i][j][k]),
4379 locals->EffectiveDETLBLinesChroma * (locals->HTotal[k] / locals->PixelClock[k]) / (locals->VRatio[k] / 2) -
4380 locals->EffectiveDETLBLinesChroma * locals->SwathWidthYPerState[i][j][k] / 2 *
4381 dml_ceil(locals->BytePerPixelInDETC[k], 2) / (locals->ReturnBWPerState[i][0] / locals->NoOfDPP[i][j][k]));
4382 }
4383 }
4384 }
4385 }
4386
4387 for (i = 0; i <= locals->soc.num_states; i++) {
4388 for (j = 0; j < 2; j++) {
4389 locals->UrgentLatencySupport[i][j] = true;
4390 for (k = 0; k < locals->NumberOfActivePlanes; k++) {
4391 if (locals->UrgentLatencySupportUsPerState[i][j][k] < locals->UrgentLatency)
4392 locals->UrgentLatencySupport[i][j] = false;
4393 }
4394 }
4395 }
4396
4397
4398 /*Prefetch Check*/
4399 for (i = 0; i <= locals->soc.num_states; i++) {
4400 for (j = 0; j < 2; j++) {
4401 locals->TotalNumberOfDCCActiveDPP[i][j] = 0;
4402 for (k = 0; k < locals->NumberOfActivePlanes; k++) {
4403 if (locals->DCCEnable[k] == true) {
4404 locals->TotalNumberOfDCCActiveDPP[i][j] =
4405 locals->TotalNumberOfDCCActiveDPP[i][j] + locals->NoOfDPP[i][j][k];
4406 }
4407 }
4408 }
4409 }
4410
4411 CalculateMinAndMaxPrefetchMode(locals->AllowDRAMSelfRefreshOrDRAMClockChangeInVblank, &locals->MinPrefetchMode, &locals->MaxPrefetchMode);
4412
4413 for (i = 0; i <= locals->soc.num_states; i++) {
4414 for (j = 0; j < 2; j++) {
4415 for (k = 0; k < locals->NumberOfActivePlanes; k++) {
4416 locals->NoOfDPPThisState[k] = locals->NoOfDPP[i][j][k];
4417 locals->RequiredDPPCLKThisState[k] = locals->RequiredDPPCLK[i][j][k];
4418 locals->SwathHeightYThisState[k] = locals->SwathHeightYPerState[i][j][k];
4419 locals->SwathHeightCThisState[k] = locals->SwathHeightCPerState[i][j][k];
4420 locals->SwathWidthYThisState[k] = locals->SwathWidthYPerState[i][j][k];
4421 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] = dml_max(
4422 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4423 mode_lib->vba.PixelClock[k] / 16.0);
4424 if (mode_lib->vba.BytePerPixelInDETC[k] == 0.0) {
4425 if (mode_lib->vba.VRatio[k] <= 1.0) {
4426 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4427 dml_max(
4428 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4429 1.1
4430 * dml_ceil(
4431 mode_lib->vba.BytePerPixelInDETY[k],
4432 1.0)
4433 / 64.0
4434 * mode_lib->vba.HRatio[k]
4435 * mode_lib->vba.PixelClock[k]
4436 / mode_lib->vba.NoOfDPP[i][j][k]);
4437 } else {
4438 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4439 dml_max(
4440 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4441 1.1
4442 * dml_ceil(
4443 mode_lib->vba.BytePerPixelInDETY[k],
4444 1.0)
4445 / 64.0
4446 * mode_lib->vba.PSCL_FACTOR[k]
4447 * mode_lib->vba.RequiredDPPCLK[i][j][k]);
4448 }
4449 } else {
4450 if (mode_lib->vba.VRatio[k] <= 1.0) {
4451 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4452 dml_max(
4453 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4454 1.1
4455 * dml_ceil(
4456 mode_lib->vba.BytePerPixelInDETY[k],
4457 1.0)
4458 / 32.0
4459 * mode_lib->vba.HRatio[k]
4460 * mode_lib->vba.PixelClock[k]
4461 / mode_lib->vba.NoOfDPP[i][j][k]);
4462 } else {
4463 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4464 dml_max(
4465 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4466 1.1
4467 * dml_ceil(
4468 mode_lib->vba.BytePerPixelInDETY[k],
4469 1.0)
4470 / 32.0
4471 * mode_lib->vba.PSCL_FACTOR[k]
4472 * mode_lib->vba.RequiredDPPCLK[i][j][k]);
4473 }
4474 if (mode_lib->vba.VRatio[k] / 2.0 <= 1.0) {
4475 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4476 dml_max(
4477 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4478 1.1
4479 * dml_ceil(
4480 mode_lib->vba.BytePerPixelInDETC[k],
4481 2.0)
4482 / 32.0
4483 * mode_lib->vba.HRatio[k]
4484 / 2.0
4485 * mode_lib->vba.PixelClock[k]
4486 / mode_lib->vba.NoOfDPP[i][j][k]);
4487 } else {
4488 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0] =
4489 dml_max(
4490 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4491 1.1
4492 * dml_ceil(
4493 mode_lib->vba.BytePerPixelInDETC[k],
4494 2.0)
4495 / 32.0
4496 * mode_lib->vba.PSCL_FACTOR_CHROMA[k]
4497 * mode_lib->vba.RequiredDPPCLK[i][j][k]);
4498 }
4499 }
4500 }
4501 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4502 mode_lib->vba.PDEAndMetaPTEBytesPerFrameY = CalculateVMAndRowBytes(
4503 mode_lib,
4504 mode_lib->vba.DCCEnable[k],
4505 mode_lib->vba.Read256BlockHeightY[k],
4506 mode_lib->vba.Read256BlockWidthY[k],
4507 mode_lib->vba.SourcePixelFormat[k],
4508 mode_lib->vba.SurfaceTiling[k],
4509 dml_ceil(mode_lib->vba.BytePerPixelInDETY[k], 1.0),
4510 mode_lib->vba.SourceScan[k],
4511 mode_lib->vba.ViewportWidth[k],
4512 mode_lib->vba.ViewportHeight[k],
4513 mode_lib->vba.SwathWidthYPerState[i][j][k],
4514 mode_lib->vba.GPUVMEnable,
4515 mode_lib->vba.VMMPageSize,
4516 mode_lib->vba.PTEBufferSizeInRequestsLuma,
4517 mode_lib->vba.PDEProcessingBufIn64KBReqs,
4518 mode_lib->vba.PitchY[k],
4519 mode_lib->vba.DCCMetaPitchY[k],
4520 &mode_lib->vba.MacroTileWidthY[k],
4521 &mode_lib->vba.MetaRowBytesY,
4522 &mode_lib->vba.DPTEBytesPerRowY,
4523 &mode_lib->vba.PTEBufferSizeNotExceededY[i][j][k],
4524 &mode_lib->vba.dpte_row_height[k],
4525 &mode_lib->vba.meta_row_height[k]);
4526 mode_lib->vba.PrefetchLinesY[0][0][k] = CalculatePrefetchSourceLines(
4527 mode_lib,
4528 mode_lib->vba.VRatio[k],
4529 mode_lib->vba.vtaps[k],
4530 mode_lib->vba.Interlace[k],
4531 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
4532 mode_lib->vba.SwathHeightYPerState[i][j][k],
4533 mode_lib->vba.ViewportYStartY[k],
4534 &mode_lib->vba.PrefillY[k],
4535 &mode_lib->vba.MaxNumSwY[k]);
4536 if ((mode_lib->vba.SourcePixelFormat[k] != dm_444_64
4537 && mode_lib->vba.SourcePixelFormat[k] != dm_444_32
4538 && mode_lib->vba.SourcePixelFormat[k] != dm_444_16
4539 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_16
4540 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_8)) {
4541 mode_lib->vba.PDEAndMetaPTEBytesPerFrameC = CalculateVMAndRowBytes(
4542 mode_lib,
4543 mode_lib->vba.DCCEnable[k],
4544 mode_lib->vba.Read256BlockHeightY[k],
4545 mode_lib->vba.Read256BlockWidthY[k],
4546 mode_lib->vba.SourcePixelFormat[k],
4547 mode_lib->vba.SurfaceTiling[k],
4548 dml_ceil(mode_lib->vba.BytePerPixelInDETC[k], 2.0),
4549 mode_lib->vba.SourceScan[k],
4550 mode_lib->vba.ViewportWidth[k] / 2.0,
4551 mode_lib->vba.ViewportHeight[k] / 2.0,
4552 mode_lib->vba.SwathWidthYPerState[i][j][k] / 2.0,
4553 mode_lib->vba.GPUVMEnable,
4554 mode_lib->vba.VMMPageSize,
4555 mode_lib->vba.PTEBufferSizeInRequestsLuma,
4556 mode_lib->vba.PDEProcessingBufIn64KBReqs,
4557 mode_lib->vba.PitchC[k],
4558 0.0,
4559 &mode_lib->vba.MacroTileWidthC[k],
4560 &mode_lib->vba.MetaRowBytesC,
4561 &mode_lib->vba.DPTEBytesPerRowC,
4562 &mode_lib->vba.PTEBufferSizeNotExceededC[i][j][k],
4563 &mode_lib->vba.dpte_row_height_chroma[k],
4564 &mode_lib->vba.meta_row_height_chroma[k]);
4565 mode_lib->vba.PrefetchLinesC[0][0][k] = CalculatePrefetchSourceLines(
4566 mode_lib,
4567 mode_lib->vba.VRatio[k] / 2.0,
4568 mode_lib->vba.VTAPsChroma[k],
4569 mode_lib->vba.Interlace[k],
4570 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
4571 mode_lib->vba.SwathHeightCPerState[i][j][k],
4572 mode_lib->vba.ViewportYStartC[k],
4573 &mode_lib->vba.PrefillC[k],
4574 &mode_lib->vba.MaxNumSwC[k]);
4575 } else {
4576 mode_lib->vba.PDEAndMetaPTEBytesPerFrameC = 0.0;
4577 mode_lib->vba.MetaRowBytesC = 0.0;
4578 mode_lib->vba.DPTEBytesPerRowC = 0.0;
4579 locals->PrefetchLinesC[0][0][k] = 0.0;
4580 locals->PTEBufferSizeNotExceededC[i][j][k] = true;
4581 locals->PTEBufferSizeInRequestsForLuma = mode_lib->vba.PTEBufferSizeInRequestsLuma + mode_lib->vba.PTEBufferSizeInRequestsChroma;
4582 }
4583 locals->PDEAndMetaPTEBytesPerFrame[0][0][k] =
4584 mode_lib->vba.PDEAndMetaPTEBytesPerFrameY + mode_lib->vba.PDEAndMetaPTEBytesPerFrameC;
4585 locals->MetaRowBytes[0][0][k] = mode_lib->vba.MetaRowBytesY + mode_lib->vba.MetaRowBytesC;
4586 locals->DPTEBytesPerRow[0][0][k] = mode_lib->vba.DPTEBytesPerRowY + mode_lib->vba.DPTEBytesPerRowC;
4587
4588 CalculateActiveRowBandwidth(
4589 mode_lib->vba.GPUVMEnable,
4590 mode_lib->vba.SourcePixelFormat[k],
4591 mode_lib->vba.VRatio[k],
4592 mode_lib->vba.DCCEnable[k],
4593 mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k],
4594 mode_lib->vba.MetaRowBytesY,
4595 mode_lib->vba.MetaRowBytesC,
4596 mode_lib->vba.meta_row_height[k],
4597 mode_lib->vba.meta_row_height_chroma[k],
4598 mode_lib->vba.DPTEBytesPerRowY,
4599 mode_lib->vba.DPTEBytesPerRowC,
4600 mode_lib->vba.dpte_row_height[k],
4601 mode_lib->vba.dpte_row_height_chroma[k],
4602 &mode_lib->vba.meta_row_bw[k],
4603 &mode_lib->vba.dpte_row_bw[k],
4604 &mode_lib->vba.qual_row_bw[k]);
4605 }
4606 mode_lib->vba.ExtraLatency =
4607 mode_lib->vba.UrgentRoundTripAndOutOfOrderLatencyPerState[i]
4608 + (mode_lib->vba.TotalNumberOfActiveDPP[i][j]
4609 * mode_lib->vba.PixelChunkSizeInKByte
4610 + mode_lib->vba.TotalNumberOfDCCActiveDPP[i][j]
4611 * mode_lib->vba.MetaChunkSize)
4612 * 1024.0
4613 / mode_lib->vba.ReturnBWPerState[i][0];
4614 if (mode_lib->vba.GPUVMEnable == true) {
4615 mode_lib->vba.ExtraLatency = mode_lib->vba.ExtraLatency
4616 + mode_lib->vba.TotalNumberOfActiveDPP[i][j]
4617 * mode_lib->vba.PTEGroupSize
4618 / mode_lib->vba.ReturnBWPerState[i][0];
4619 }
4620 mode_lib->vba.TimeCalc = 24.0 / mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0];
4621
4622 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4623 if (mode_lib->vba.BlendingAndTiming[k] == k) {
4624 if (mode_lib->vba.WritebackEnable[k] == true) {
4625 locals->WritebackDelay[i][k] = mode_lib->vba.WritebackLatency
4626 + CalculateWriteBackDelay(
4627 mode_lib->vba.WritebackPixelFormat[k],
4628 mode_lib->vba.WritebackHRatio[k],
4629 mode_lib->vba.WritebackVRatio[k],
4630 mode_lib->vba.WritebackLumaHTaps[k],
4631 mode_lib->vba.WritebackLumaVTaps[k],
4632 mode_lib->vba.WritebackChromaHTaps[k],
4633 mode_lib->vba.WritebackChromaVTaps[k],
4634 mode_lib->vba.WritebackDestinationWidth[k]) / locals->RequiredDISPCLK[i][j];
4635 } else {
4636 locals->WritebackDelay[i][k] = 0.0;
4637 }
4638 for (m = 0; m <= mode_lib->vba.NumberOfActivePlanes - 1; m++) {
4639 if (mode_lib->vba.BlendingAndTiming[m] == k
4640 && mode_lib->vba.WritebackEnable[m]
4641 == true) {
4642 locals->WritebackDelay[i][k] = dml_max(locals->WritebackDelay[i][k],
4643 mode_lib->vba.WritebackLatency + CalculateWriteBackDelay(
4644 mode_lib->vba.WritebackPixelFormat[m],
4645 mode_lib->vba.WritebackHRatio[m],
4646 mode_lib->vba.WritebackVRatio[m],
4647 mode_lib->vba.WritebackLumaHTaps[m],
4648 mode_lib->vba.WritebackLumaVTaps[m],
4649 mode_lib->vba.WritebackChromaHTaps[m],
4650 mode_lib->vba.WritebackChromaVTaps[m],
4651 mode_lib->vba.WritebackDestinationWidth[m]) / locals->RequiredDISPCLK[i][j]);
4652 }
4653 }
4654 }
4655 }
4656 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4657 for (m = 0; m <= mode_lib->vba.NumberOfActivePlanes - 1; m++) {
4658 if (mode_lib->vba.BlendingAndTiming[k] == m) {
4659 locals->WritebackDelay[i][k] = locals->WritebackDelay[i][m];
4660 }
4661 }
4662 }
4663 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4664 for (m = 0; m < locals->NumberOfCursors[k]; m++)
4665 locals->cursor_bw[k] = locals->NumberOfCursors[k] * locals->CursorWidth[k][m] * locals->CursorBPP[k][m]
4666 / 8 / (locals->HTotal[k] / locals->PixelClock[k]) * locals->VRatio[k];
4667 }
4668
4669 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4670 locals->MaximumVStartup[0][0][k] = mode_lib->vba.VTotal[k] - mode_lib->vba.VActive[k]
4671 - dml_max(1.0, dml_ceil(locals->WritebackDelay[i][k] / (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k]), 1.0));
4672 }
4673
4674 mode_lib->vba.NextPrefetchMode = mode_lib->vba.MinPrefetchMode;
4675 do {
4676 mode_lib->vba.PrefetchMode[i][j] = mode_lib->vba.NextPrefetchMode;
4677 mode_lib->vba.NextPrefetchMode = mode_lib->vba.NextPrefetchMode + 1;
4678
4679 mode_lib->vba.TWait = CalculateTWait(
4680 mode_lib->vba.PrefetchMode[i][j],
4681 mode_lib->vba.DRAMClockChangeLatency,
4682 mode_lib->vba.UrgentLatency,
4683 mode_lib->vba.SREnterPlusExitTime);
4684 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4685
4686 if (mode_lib->vba.XFCEnabled[k] == true) {
4687 mode_lib->vba.XFCRemoteSurfaceFlipDelay =
4688 CalculateRemoteSurfaceFlipDelay(
4689 mode_lib,
4690 mode_lib->vba.VRatio[k],
4691 locals->SwathWidthYPerState[i][j][k],
4692 dml_ceil(locals->BytePerPixelInDETY[k], 1.0),
4693 mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k],
4694 mode_lib->vba.XFCTSlvVupdateOffset,
4695 mode_lib->vba.XFCTSlvVupdateWidth,
4696 mode_lib->vba.XFCTSlvVreadyOffset,
4697 mode_lib->vba.XFCXBUFLatencyTolerance,
4698 mode_lib->vba.XFCFillBWOverhead,
4699 mode_lib->vba.XFCSlvChunkSize,
4700 mode_lib->vba.XFCBusTransportTime,
4701 mode_lib->vba.TimeCalc,
4702 mode_lib->vba.TWait,
4703 &mode_lib->vba.SrcActiveDrainRate,
4704 &mode_lib->vba.TInitXFill,
4705 &mode_lib->vba.TslvChk);
4706 } else {
4707 mode_lib->vba.XFCRemoteSurfaceFlipDelay = 0.0;
4708 }
4709 mode_lib->vba.IsErrorResult[i][j][k] =
4710 CalculatePrefetchSchedule(
4711 mode_lib,
4712 mode_lib->vba.RequiredDPPCLK[i][j][k],
4713 mode_lib->vba.RequiredDISPCLK[i][j],
4714 mode_lib->vba.PixelClock[k],
4715 mode_lib->vba.ProjectedDCFCLKDeepSleep[0][0],
4716 mode_lib->vba.DSCDelayPerState[i][k],
4717 mode_lib->vba.NoOfDPP[i][j][k],
4718 mode_lib->vba.ScalerEnabled[k],
4719 mode_lib->vba.NumberOfCursors[k],
4720 mode_lib->vba.DPPCLKDelaySubtotal,
4721 mode_lib->vba.DPPCLKDelaySCL,
4722 mode_lib->vba.DPPCLKDelaySCLLBOnly,
4723 mode_lib->vba.DPPCLKDelayCNVCFormater,
4724 mode_lib->vba.DPPCLKDelayCNVCCursor,
4725 mode_lib->vba.DISPCLKDelaySubtotal,
4726 mode_lib->vba.SwathWidthYPerState[i][j][k]
4727 / mode_lib->vba.HRatio[k],
4728 mode_lib->vba.OutputFormat[k],
4729 mode_lib->vba.VTotal[k]
4730 - mode_lib->vba.VActive[k],
4731 mode_lib->vba.HTotal[k],
4732 mode_lib->vba.MaxInterDCNTileRepeaters,
4733 mode_lib->vba.MaximumVStartup[0][0][k],
4734 mode_lib->vba.GPUVMMaxPageTableLevels,
4735 mode_lib->vba.GPUVMEnable,
4736 mode_lib->vba.DynamicMetadataEnable[k],
4737 mode_lib->vba.DynamicMetadataLinesBeforeActiveRequired[k],
4738 mode_lib->vba.DynamicMetadataTransmittedBytes[k],
4739 mode_lib->vba.DCCEnable[k],
4740 mode_lib->vba.UrgentLatencyPixelDataOnly,
4741 mode_lib->vba.ExtraLatency,
4742 mode_lib->vba.TimeCalc,
4743 mode_lib->vba.PDEAndMetaPTEBytesPerFrame[0][0][k],
4744 mode_lib->vba.MetaRowBytes[0][0][k],
4745 mode_lib->vba.DPTEBytesPerRow[0][0][k],
4746 mode_lib->vba.PrefetchLinesY[0][0][k],
4747 mode_lib->vba.SwathWidthYPerState[i][j][k],
4748 mode_lib->vba.BytePerPixelInDETY[k],
4749 mode_lib->vba.PrefillY[k],
4750 mode_lib->vba.MaxNumSwY[k],
4751 mode_lib->vba.PrefetchLinesC[0][0][k],
4752 mode_lib->vba.BytePerPixelInDETC[k],
4753 mode_lib->vba.PrefillC[k],
4754 mode_lib->vba.MaxNumSwC[k],
4755 mode_lib->vba.SwathHeightYPerState[i][j][k],
4756 mode_lib->vba.SwathHeightCPerState[i][j][k],
4757 mode_lib->vba.TWait,
4758 mode_lib->vba.XFCEnabled[k],
4759 mode_lib->vba.XFCRemoteSurfaceFlipDelay,
4760 mode_lib->vba.Interlace[k],
4761 mode_lib->vba.ProgressiveToInterlaceUnitInOPP,
4762 mode_lib->vba.DSTXAfterScaler,
4763 mode_lib->vba.DSTYAfterScaler,
4764 &mode_lib->vba.LineTimesForPrefetch[k],
4765 &mode_lib->vba.PrefetchBW[k],
4766 &mode_lib->vba.LinesForMetaPTE[k],
4767 &mode_lib->vba.LinesForMetaAndDPTERow[k],
4768 &mode_lib->vba.VRatioPreY[i][j][k],
4769 &mode_lib->vba.VRatioPreC[i][j][k],
4770 &mode_lib->vba.RequiredPrefetchPixelDataBWLuma[i][j][k],
4771 &mode_lib->vba.VStartupRequiredWhenNotEnoughTimeForDynamicMetadata,
4772 &mode_lib->vba.Tno_bw[k],
4773 &mode_lib->vba.VUpdateOffsetPix[k],
4774 &mode_lib->vba.VUpdateWidthPix[k],
4775 &mode_lib->vba.VReadyOffsetPix[k]);
4776 }
4777 mode_lib->vba.MaximumReadBandwidthWithoutPrefetch = 0.0;
4778 mode_lib->vba.MaximumReadBandwidthWithPrefetch = 0.0;
4779 locals->prefetch_vm_bw_valid = true;
4780 locals->prefetch_row_bw_valid = true;
4781 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4782 if (locals->PDEAndMetaPTEBytesPerFrame[0][0][k] == 0)
4783 locals->prefetch_vm_bw[k] = 0;
4784 else if (locals->LinesForMetaPTE[k] > 0)
4785 locals->prefetch_vm_bw[k] = locals->PDEAndMetaPTEBytesPerFrame[0][0][k]
4786 / (locals->LinesForMetaPTE[k] * locals->HTotal[k] / locals->PixelClock[k]);
4787 else {
4788 locals->prefetch_vm_bw[k] = 0;
4789 locals->prefetch_vm_bw_valid = false;
4790 }
4791 if (locals->MetaRowBytes[0][0][k] + locals->DPTEBytesPerRow[0][0][k] == 0)
4792 locals->prefetch_row_bw[k] = 0;
4793 else if (locals->LinesForMetaAndDPTERow[k] > 0)
4794 locals->prefetch_row_bw[k] = (locals->MetaRowBytes[0][0][k] + locals->DPTEBytesPerRow[0][0][k])
4795 / (locals->LinesForMetaAndDPTERow[k] * locals->HTotal[k] / locals->PixelClock[k]);
4796 else {
4797 locals->prefetch_row_bw[k] = 0;
4798 locals->prefetch_row_bw_valid = false;
4799 }
4800
4801 mode_lib->vba.MaximumReadBandwidthWithoutPrefetch = mode_lib->vba.MaximumReadBandwidthWithPrefetch
4802 + mode_lib->vba.cursor_bw[k] + mode_lib->vba.ReadBandwidth[k] + mode_lib->vba.meta_row_bw[k] + mode_lib->vba.dpte_row_bw[k];
4803 mode_lib->vba.MaximumReadBandwidthWithPrefetch =
4804 mode_lib->vba.MaximumReadBandwidthWithPrefetch
4805 + mode_lib->vba.cursor_bw[k]
4806 + dml_max3(
4807 mode_lib->vba.prefetch_vm_bw[k],
4808 mode_lib->vba.prefetch_row_bw[k],
4809 dml_max(mode_lib->vba.ReadBandwidth[k],
4810 mode_lib->vba.RequiredPrefetchPixelDataBWLuma[i][j][k])
4811 + mode_lib->vba.meta_row_bw[k] + mode_lib->vba.dpte_row_bw[k]);
4812 }
4813 locals->BandwidthWithoutPrefetchSupported[i][0] = true;
4814 if (mode_lib->vba.MaximumReadBandwidthWithoutPrefetch > locals->ReturnBWPerState[i][0]) {
4815 locals->BandwidthWithoutPrefetchSupported[i][0] = false;
4816 }
4817
4818 locals->PrefetchSupported[i][j] = true;
4819 if (mode_lib->vba.MaximumReadBandwidthWithPrefetch > locals->ReturnBWPerState[i][0]) {
4820 locals->PrefetchSupported[i][j] = false;
4821 }
4822 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4823 if (locals->LineTimesForPrefetch[k] < 2.0
4824 || locals->LinesForMetaPTE[k] >= 8.0
4825 || locals->LinesForMetaAndDPTERow[k] >= 16.0
4826 || mode_lib->vba.IsErrorResult[i][j][k] == true) {
4827 locals->PrefetchSupported[i][j] = false;
4828 }
4829 }
4830 locals->VRatioInPrefetchSupported[i][j] = true;
4831 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4832 if (locals->VRatioPreY[i][j][k] > 4.0
4833 || locals->VRatioPreC[i][j][k] > 4.0
4834 || mode_lib->vba.IsErrorResult[i][j][k] == true) {
4835 locals->VRatioInPrefetchSupported[i][j] = false;
4836 }
4837 }
4838 } while ((locals->PrefetchSupported[i][j] != true || locals->VRatioInPrefetchSupported[i][j] != true)
4839 && mode_lib->vba.NextPrefetchMode < mode_lib->vba.MaxPrefetchMode);
4840
4841 if (mode_lib->vba.PrefetchSupported[i][j] == true
4842 && mode_lib->vba.VRatioInPrefetchSupported[i][j] == true) {
4843 mode_lib->vba.BandwidthAvailableForImmediateFlip =
4844 mode_lib->vba.ReturnBWPerState[i][0];
4845 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4846 mode_lib->vba.BandwidthAvailableForImmediateFlip =
4847 mode_lib->vba.BandwidthAvailableForImmediateFlip
4848 - mode_lib->vba.cursor_bw[k]
4849 - dml_max(
4850 mode_lib->vba.ReadBandwidth[k] + mode_lib->vba.qual_row_bw[k],
4851 mode_lib->vba.PrefetchBW[k]);
4852 }
4853 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4854 mode_lib->vba.ImmediateFlipBytes[k] = 0.0;
4855 if ((mode_lib->vba.SourcePixelFormat[k] != dm_420_8
4856 && mode_lib->vba.SourcePixelFormat[k] != dm_420_10)) {
4857 mode_lib->vba.ImmediateFlipBytes[k] =
4858 mode_lib->vba.PDEAndMetaPTEBytesPerFrame[0][0][k]
4859 + mode_lib->vba.MetaRowBytes[0][0][k]
4860 + mode_lib->vba.DPTEBytesPerRow[0][0][k];
4861 }
4862 }
4863 mode_lib->vba.TotImmediateFlipBytes = 0.0;
4864 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4865 if ((mode_lib->vba.SourcePixelFormat[k] != dm_420_8
4866 && mode_lib->vba.SourcePixelFormat[k] != dm_420_10)) {
4867 mode_lib->vba.TotImmediateFlipBytes =
4868 mode_lib->vba.TotImmediateFlipBytes
4869 + mode_lib->vba.ImmediateFlipBytes[k];
4870 }
4871 }
4872
4873 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4874 CalculateFlipSchedule(
4875 mode_lib,
4876 mode_lib->vba.ExtraLatency,
4877 mode_lib->vba.UrgentLatencyPixelDataOnly,
4878 mode_lib->vba.GPUVMMaxPageTableLevels,
4879 mode_lib->vba.GPUVMEnable,
4880 mode_lib->vba.BandwidthAvailableForImmediateFlip,
4881 mode_lib->vba.TotImmediateFlipBytes,
4882 mode_lib->vba.SourcePixelFormat[k],
4883 mode_lib->vba.ImmediateFlipBytes[k],
4884 mode_lib->vba.HTotal[k]
4885 / mode_lib->vba.PixelClock[k],
4886 mode_lib->vba.VRatio[k],
4887 mode_lib->vba.Tno_bw[k],
4888 mode_lib->vba.PDEAndMetaPTEBytesPerFrame[0][0][k],
4889 mode_lib->vba.MetaRowBytes[0][0][k],
4890 mode_lib->vba.DPTEBytesPerRow[0][0][k],
4891 mode_lib->vba.DCCEnable[k],
4892 mode_lib->vba.dpte_row_height[k],
4893 mode_lib->vba.meta_row_height[k],
4894 mode_lib->vba.qual_row_bw[k],
4895 &mode_lib->vba.DestinationLinesToRequestVMInImmediateFlip[k],
4896 &mode_lib->vba.DestinationLinesToRequestRowInImmediateFlip[k],
4897 &mode_lib->vba.final_flip_bw[k],
4898 &mode_lib->vba.ImmediateFlipSupportedForPipe[k]);
4899 }
4900 mode_lib->vba.total_dcn_read_bw_with_flip = 0.0;
4901 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4902 mode_lib->vba.total_dcn_read_bw_with_flip =
4903 mode_lib->vba.total_dcn_read_bw_with_flip
4904 + mode_lib->vba.cursor_bw[k]
4905 + dml_max3(
4906 mode_lib->vba.prefetch_vm_bw[k],
4907 mode_lib->vba.prefetch_row_bw[k],
4908 mode_lib->vba.final_flip_bw[k]
4909 + dml_max(
4910 mode_lib->vba.ReadBandwidth[k],
4911 mode_lib->vba.RequiredPrefetchPixelDataBWLuma[i][j][k]));
4912 }
4913 mode_lib->vba.ImmediateFlipSupportedForState[i][j] = true;
4914 if (mode_lib->vba.total_dcn_read_bw_with_flip
4915 > mode_lib->vba.ReturnBWPerState[i][0]) {
4916 mode_lib->vba.ImmediateFlipSupportedForState[i][j] = false;
4917 }
4918 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4919 if (mode_lib->vba.ImmediateFlipSupportedForPipe[k] == false) {
4920 mode_lib->vba.ImmediateFlipSupportedForState[i][j] = false;
4921 }
4922 }
4923 } else {
4924 mode_lib->vba.ImmediateFlipSupportedForState[i][j] = false;
4925 }
4926 }
4927 }
4928
4929 /*Vertical Active BW support*/
4930 mode_lib->vba.MaxTotalVActiveRDBandwidth = 0;
4931 for (k = 0; k < mode_lib->vba.NumberOfActivePlanes; k++)
4932 mode_lib->vba.MaxTotalVActiveRDBandwidth = mode_lib->vba.MaxTotalVActiveRDBandwidth + mode_lib->vba.ReadBandwidth[k];
4933 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4934 mode_lib->vba.MaxTotalVerticalActiveAvailableBandwidth[i][0] = dml_min(mode_lib->vba.ReturnBusWidth *
4935 mode_lib->vba.DCFCLKPerState[i], mode_lib->vba.FabricAndDRAMBandwidthPerState[i] * 1000) *
4936 mode_lib->vba.MaxAveragePercentOfIdealDRAMBWDisplayCanUseInNormalSystemOperation / 100;
4937 if (mode_lib->vba.MaxTotalVActiveRDBandwidth <= mode_lib->vba.MaxTotalVerticalActiveAvailableBandwidth[i][0])
4938 mode_lib->vba.TotalVerticalActiveBandwidthSupport[i][0] = true;
4939 else
4940 mode_lib->vba.TotalVerticalActiveBandwidthSupport[i][0] = false;
4941 }
4942
4943 /*PTE Buffer Size Check*/
4944
4945 for (i = 0; i <= mode_lib->vba.soc.num_states; i++) {
4946 for (j = 0; j < 2; j++) {
4947 locals->PTEBufferSizeNotExceeded[i][j] = true;
4948 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4949 if (locals->PTEBufferSizeNotExceededY[i][j][k] == false
4950 || locals->PTEBufferSizeNotExceededC[i][j][k] == false) {
4951 locals->PTEBufferSizeNotExceeded[i][j] = false;
4952 }
4953 }
4954 }
4955 }
4956 /*Cursor Support Check*/
4957 mode_lib->vba.CursorSupport = true;
4958 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4959 for (j = 0; j < 2; j++) {
4960 if (mode_lib->vba.CursorWidth[k][j] > 0.0) {
4961 if (dml_floor(
4962 dml_floor(
4963 mode_lib->vba.CursorBufferSize
4964 - mode_lib->vba.CursorChunkSize,
4965 mode_lib->vba.CursorChunkSize) * 1024.0
4966 / (mode_lib->vba.CursorWidth[k][j]
4967 * mode_lib->vba.CursorBPP[k][j]
4968 / 8.0),
4969 1.0)
4970 * (mode_lib->vba.HTotal[k] / mode_lib->vba.PixelClock[k])
4971 / mode_lib->vba.VRatio[k] < mode_lib->vba.UrgentLatencyPixelDataOnly
4972 || (mode_lib->vba.CursorBPP[k][j] == 64.0
4973 && mode_lib->vba.Cursor64BppSupport == false)) {
4974 mode_lib->vba.CursorSupport = false;
4975 }
4976 }
4977 }
4978 }
4979 /*Valid Pitch Check*/
4980
4981 mode_lib->vba.PitchSupport = true;
4982 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
4983 locals->AlignedYPitch[k] = dml_ceil(
4984 dml_max(mode_lib->vba.PitchY[k], mode_lib->vba.ViewportWidth[k]),
4985 locals->MacroTileWidthY[k]);
4986 if (locals->AlignedYPitch[k] > mode_lib->vba.PitchY[k]) {
4987 mode_lib->vba.PitchSupport = false;
4988 }
4989 if (mode_lib->vba.DCCEnable[k] == true) {
4990 locals->AlignedDCCMetaPitch[k] = dml_ceil(
4991 dml_max(
4992 mode_lib->vba.DCCMetaPitchY[k],
4993 mode_lib->vba.ViewportWidth[k]),
4994 64.0 * locals->Read256BlockWidthY[k]);
4995 } else {
4996 locals->AlignedDCCMetaPitch[k] = mode_lib->vba.DCCMetaPitchY[k];
4997 }
4998 if (locals->AlignedDCCMetaPitch[k] > mode_lib->vba.DCCMetaPitchY[k]) {
4999 mode_lib->vba.PitchSupport = false;
5000 }
5001 if (mode_lib->vba.SourcePixelFormat[k] != dm_444_64
5002 && mode_lib->vba.SourcePixelFormat[k] != dm_444_32
5003 && mode_lib->vba.SourcePixelFormat[k] != dm_444_16
5004 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_16
5005 && mode_lib->vba.SourcePixelFormat[k] != dm_mono_8) {
5006 locals->AlignedCPitch[k] = dml_ceil(
5007 dml_max(
5008 mode_lib->vba.PitchC[k],
5009 mode_lib->vba.ViewportWidth[k] / 2.0),
5010 locals->MacroTileWidthC[k]);
5011 } else {
5012 locals->AlignedCPitch[k] = mode_lib->vba.PitchC[k];
5013 }
5014 if (locals->AlignedCPitch[k] > mode_lib->vba.PitchC[k]) {
5015 mode_lib->vba.PitchSupport = false;
5016 }
5017 }
5018 /*Mode Support, Voltage State and SOC Configuration*/
5019
5020 for (i = mode_lib->vba.soc.num_states; i >= 0; i--) {
5021 for (j = 0; j < 2; j++) {
5022 enum dm_validation_status status = DML_VALIDATION_OK;
5023
5024 if (mode_lib->vba.ScaleRatioAndTapsSupport != true) {
5025 status = DML_FAIL_SCALE_RATIO_TAP;
5026 } else if (mode_lib->vba.SourceFormatPixelAndScanSupport != true) {
5027 status = DML_FAIL_SOURCE_PIXEL_FORMAT;
5028 } else if (locals->ViewportSizeSupport[i][0] != true) {
5029 status = DML_FAIL_VIEWPORT_SIZE;
5030 } else if (locals->DIOSupport[i] != true) {
5031 status = DML_FAIL_DIO_SUPPORT;
5032 } else if (locals->NotEnoughDSCUnits[i] != false) {
5033 status = DML_FAIL_NOT_ENOUGH_DSC;
5034 } else if (locals->DSCCLKRequiredMoreThanSupported[i] != false) {
5035 status = DML_FAIL_DSC_CLK_REQUIRED;
5036 } else if (locals->UrgentLatencySupport[i][j] != true) {
5037 status = DML_FAIL_URGENT_LATENCY;
5038 } else if (locals->ROBSupport[i][0] != true) {
5039 status = DML_FAIL_REORDERING_BUFFER;
5040 } else if (locals->DISPCLK_DPPCLK_Support[i][j] != true) {
5041 status = DML_FAIL_DISPCLK_DPPCLK;
5042 } else if (locals->TotalAvailablePipesSupport[i][j] != true) {
5043 status = DML_FAIL_TOTAL_AVAILABLE_PIPES;
5044 } else if (mode_lib->vba.NumberOfOTGSupport != true) {
5045 status = DML_FAIL_NUM_OTG;
5046 } else if (mode_lib->vba.WritebackModeSupport != true) {
5047 status = DML_FAIL_WRITEBACK_MODE;
5048 } else if (mode_lib->vba.WritebackLatencySupport != true) {
5049 status = DML_FAIL_WRITEBACK_LATENCY;
5050 } else if (mode_lib->vba.WritebackScaleRatioAndTapsSupport != true) {
5051 status = DML_FAIL_WRITEBACK_SCALE_RATIO_TAP;
5052 } else if (mode_lib->vba.CursorSupport != true) {
5053 status = DML_FAIL_CURSOR_SUPPORT;
5054 } else if (mode_lib->vba.PitchSupport != true) {
5055 status = DML_FAIL_PITCH_SUPPORT;
5056 } else if (locals->PrefetchSupported[i][j] != true) {
5057 status = DML_FAIL_PREFETCH_SUPPORT;
5058 } else if (locals->TotalVerticalActiveBandwidthSupport[i][0] != true) {
5059 status = DML_FAIL_TOTAL_V_ACTIVE_BW;
5060 } else if (locals->VRatioInPrefetchSupported[i][j] != true) {
5061 status = DML_FAIL_V_RATIO_PREFETCH;
5062 } else if (locals->PTEBufferSizeNotExceeded[i][j] != true) {
5063 status = DML_FAIL_PTE_BUFFER_SIZE;
5064 } else if (mode_lib->vba.NonsupportedDSCInputBPC != false) {
5065 status = DML_FAIL_DSC_INPUT_BPC;
5066 }
5067
5068 if (status == DML_VALIDATION_OK) {
5069 locals->ModeSupport[i][j] = true;
5070 } else {
5071 locals->ModeSupport[i][j] = false;
5072 }
5073 locals->ValidationStatus[i] = status;
5074 }
5075 }
5076 {
5077 unsigned int MaximumMPCCombine = 0;
5078 mode_lib->vba.VoltageLevel = mode_lib->vba.soc.num_states + 1;
5079 for (i = mode_lib->vba.VoltageOverrideLevel; i <= mode_lib->vba.soc.num_states; i++) {
5080 if (locals->ModeSupport[i][0] == true || locals->ModeSupport[i][1] == true) {
5081 mode_lib->vba.VoltageLevel = i;
5082 if (locals->ModeSupport[i][1] == true && (locals->ModeSupport[i][0] == false
5083 || mode_lib->vba.WhenToDoMPCCombine == dm_mpc_always_when_possible)) {
5084 MaximumMPCCombine = 1;
5085 } else {
5086 MaximumMPCCombine = 0;
5087 }
5088 break;
5089 }
5090 }
5091 mode_lib->vba.ImmediateFlipSupport =
5092 locals->ImmediateFlipSupportedForState[mode_lib->vba.VoltageLevel][MaximumMPCCombine];
5093 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
5094 mode_lib->vba.DPPPerPlane[k] = locals->NoOfDPP[mode_lib->vba.VoltageLevel][MaximumMPCCombine][k];
5095 locals->DPPCLK[k] = locals->RequiredDPPCLK[mode_lib->vba.VoltageLevel][MaximumMPCCombine][k];
5096 }
5097 mode_lib->vba.DISPCLK = locals->RequiredDISPCLK[mode_lib->vba.VoltageLevel][MaximumMPCCombine];
5098 mode_lib->vba.maxMpcComb = MaximumMPCCombine;
5099 }
5100 mode_lib->vba.DCFCLK = mode_lib->vba.DCFCLKPerState[mode_lib->vba.VoltageLevel];
5101 mode_lib->vba.DRAMSpeed = mode_lib->vba.DRAMSpeedPerState[mode_lib->vba.VoltageLevel];
5102 mode_lib->vba.FabricClock = mode_lib->vba.FabricClockPerState[mode_lib->vba.VoltageLevel];
5103 mode_lib->vba.SOCCLK = mode_lib->vba.SOCCLKPerState[mode_lib->vba.VoltageLevel];
5104 mode_lib->vba.ReturnBW = locals->ReturnBWPerState[mode_lib->vba.VoltageLevel][0];
5105 mode_lib->vba.FabricAndDRAMBandwidth = locals->FabricAndDRAMBandwidthPerState[mode_lib->vba.VoltageLevel];
5106 for (k = 0; k <= mode_lib->vba.NumberOfActivePlanes - 1; k++) {
5107 if (mode_lib->vba.BlendingAndTiming[k] == k) {
5108 mode_lib->vba.ODMCombineEnabled[k] =
5109 locals->ODMCombineEnablePerState[mode_lib->vba.VoltageLevel][k];
5110 } else {
5111 mode_lib->vba.ODMCombineEnabled[k] = 0;
5112 }
5113 mode_lib->vba.DSCEnabled[k] =
5114 locals->RequiresDSC[mode_lib->vba.VoltageLevel][k];
5115 mode_lib->vba.OutputBpp[k] =
5116 locals->OutputBppPerState[mode_lib->vba.VoltageLevel][k];
5117 }
5118 }
5119