1 /* 2 * Copyright 2016 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 "dm_services.h" 27 #include "dc.h" 28 #include "mod_freesync.h" 29 #include "core_types.h" 30 31 #define MOD_FREESYNC_MAX_CONCURRENT_STREAMS 32 32 33 #define MIN_REFRESH_RANGE_IN_US 10000000 34 /* Refresh rate ramp at a fixed rate of 65 Hz/second */ 35 #define STATIC_SCREEN_RAMP_DELTA_REFRESH_RATE_PER_FRAME ((1000 / 60) * 65) 36 /* Number of elements in the render times cache array */ 37 #define RENDER_TIMES_MAX_COUNT 10 38 /* Threshold to exit BTR (to avoid frequent enter-exits at the lower limit) */ 39 #define BTR_EXIT_MARGIN 2000 40 /* Threshold to change BTR multiplier (to avoid frequent changes) */ 41 #define BTR_DRIFT_MARGIN 2000 42 /*Threshold to exit fixed refresh rate*/ 43 #define FIXED_REFRESH_EXIT_MARGIN_IN_HZ 4 44 /* Number of consecutive frames to check before entering/exiting fixed refresh*/ 45 #define FIXED_REFRESH_ENTER_FRAME_COUNT 5 46 #define FIXED_REFRESH_EXIT_FRAME_COUNT 5 47 48 struct core_freesync { 49 struct mod_freesync public; 50 struct dc *dc; 51 }; 52 53 void setFieldWithMask(unsigned char *dest, unsigned int mask, unsigned int value) 54 { 55 unsigned int shift = 0; 56 57 if (!mask || !dest) 58 return; 59 60 while (!((mask >> shift) & 1)) 61 shift++; 62 63 //reset 64 *dest = *dest & ~mask; 65 //set 66 //dont let value span past mask 67 value = value & (mask >> shift); 68 //insert value 69 *dest = *dest | (value << shift); 70 } 71 72 // VTEM Byte Offset 73 #define VRR_VTEM_PB0 0 74 #define VRR_VTEM_PB1 1 75 #define VRR_VTEM_PB2 2 76 #define VRR_VTEM_PB3 3 77 #define VRR_VTEM_PB4 4 78 #define VRR_VTEM_PB5 5 79 #define VRR_VTEM_PB6 6 80 81 #define VRR_VTEM_MD0 7 82 #define VRR_VTEM_MD1 8 83 #define VRR_VTEM_MD2 9 84 #define VRR_VTEM_MD3 10 85 86 87 // VTEM Byte Masks 88 //PB0 89 #define MASK__VRR_VTEM_PB0__RESERVED0 0x01 90 #define MASK__VRR_VTEM_PB0__SYNC 0x02 91 #define MASK__VRR_VTEM_PB0__VFR 0x04 92 #define MASK__VRR_VTEM_PB0__AFR 0x08 93 #define MASK__VRR_VTEM_PB0__DS_TYPE 0x30 94 //0: Periodic pseudo-static EM Data Set 95 //1: Periodic dynamic EM Data Set 96 //2: Unique EM Data Set 97 //3: Reserved 98 #define MASK__VRR_VTEM_PB0__END 0x40 99 #define MASK__VRR_VTEM_PB0__NEW 0x80 100 101 //PB1 102 #define MASK__VRR_VTEM_PB1__RESERVED1 0xFF 103 104 //PB2 105 #define MASK__VRR_VTEM_PB2__ORGANIZATION_ID 0xFF 106 //0: This is a Vendor Specific EM Data Set 107 //1: This EM Data Set is defined by This Specification (HDMI 2.1 r102.clean) 108 //2: This EM Data Set is defined by CTA-861-G 109 //3: This EM Data Set is defined by VESA 110 //PB3 111 #define MASK__VRR_VTEM_PB3__DATA_SET_TAG_MSB 0xFF 112 //PB4 113 #define MASK__VRR_VTEM_PB4__DATA_SET_TAG_LSB 0xFF 114 //PB5 115 #define MASK__VRR_VTEM_PB5__DATA_SET_LENGTH_MSB 0xFF 116 //PB6 117 #define MASK__VRR_VTEM_PB6__DATA_SET_LENGTH_LSB 0xFF 118 119 120 121 //PB7-27 (20 bytes): 122 //PB7 = MD0 123 #define MASK__VRR_VTEM_MD0__VRR_EN 0x01 124 #define MASK__VRR_VTEM_MD0__M_CONST 0x02 125 #define MASK__VRR_VTEM_MD0__RESERVED2 0x0C 126 #define MASK__VRR_VTEM_MD0__FVA_FACTOR_M1 0xF0 127 128 //MD1 129 #define MASK__VRR_VTEM_MD1__BASE_VFRONT 0xFF 130 131 //MD2 132 #define MASK__VRR_VTEM_MD2__BASE_REFRESH_RATE_98 0x03 133 #define MASK__VRR_VTEM_MD2__RB 0x04 134 #define MASK__VRR_VTEM_MD2__RESERVED3 0xF8 135 136 //MD3 137 #define MASK__VRR_VTEM_MD3__BASE_REFRESH_RATE_07 0xFF 138 139 140 #define MOD_FREESYNC_TO_CORE(mod_freesync)\ 141 container_of(mod_freesync, struct core_freesync, public) 142 143 struct mod_freesync *mod_freesync_create(struct dc *dc) 144 { 145 struct core_freesync *core_freesync = 146 kzalloc(sizeof(struct core_freesync), GFP_KERNEL); 147 148 if (core_freesync == NULL) 149 goto fail_alloc_context; 150 151 if (dc == NULL) 152 goto fail_construct; 153 154 core_freesync->dc = dc; 155 return &core_freesync->public; 156 157 fail_construct: 158 kfree(core_freesync); 159 160 fail_alloc_context: 161 return NULL; 162 } 163 164 void mod_freesync_destroy(struct mod_freesync *mod_freesync) 165 { 166 struct core_freesync *core_freesync = NULL; 167 if (mod_freesync == NULL) 168 return; 169 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 170 kfree(core_freesync); 171 } 172 173 #if 0 /* unused currently */ 174 static unsigned int calc_refresh_in_uhz_from_duration( 175 unsigned int duration_in_ns) 176 { 177 unsigned int refresh_in_uhz = 178 ((unsigned int)(div64_u64((1000000000ULL * 1000000), 179 duration_in_ns))); 180 return refresh_in_uhz; 181 } 182 #endif 183 184 static unsigned int calc_duration_in_us_from_refresh_in_uhz( 185 unsigned int refresh_in_uhz) 186 { 187 unsigned int duration_in_us = 188 ((unsigned int)(div64_u64((1000000000ULL * 1000), 189 refresh_in_uhz))); 190 return duration_in_us; 191 } 192 193 static unsigned int calc_duration_in_us_from_v_total( 194 const struct dc_stream_state *stream, 195 const struct mod_vrr_params *in_vrr, 196 unsigned int v_total) 197 { 198 unsigned int duration_in_us = 199 (unsigned int)(div64_u64(((unsigned long long)(v_total) 200 * 10000) * stream->timing.h_total, 201 stream->timing.pix_clk_100hz)); 202 203 return duration_in_us; 204 } 205 206 static unsigned int calc_v_total_from_refresh( 207 const struct dc_stream_state *stream, 208 unsigned int refresh_in_uhz) 209 { 210 unsigned int v_total = stream->timing.v_total; 211 unsigned int frame_duration_in_ns; 212 213 frame_duration_in_ns = 214 ((unsigned int)(div64_u64((1000000000ULL * 1000000), 215 refresh_in_uhz))); 216 217 v_total = div64_u64(div64_u64(((unsigned long long)( 218 frame_duration_in_ns) * (stream->timing.pix_clk_100hz / 10)), 219 stream->timing.h_total), 1000000); 220 221 /* v_total cannot be less than nominal */ 222 if (v_total < stream->timing.v_total) { 223 ASSERT(v_total < stream->timing.v_total); 224 v_total = stream->timing.v_total; 225 } 226 227 return v_total; 228 } 229 230 static unsigned int calc_v_total_from_duration( 231 const struct dc_stream_state *stream, 232 const struct mod_vrr_params *vrr, 233 unsigned int duration_in_us) 234 { 235 unsigned int v_total = 0; 236 237 if (duration_in_us < vrr->min_duration_in_us) 238 duration_in_us = vrr->min_duration_in_us; 239 240 if (duration_in_us > vrr->max_duration_in_us) 241 duration_in_us = vrr->max_duration_in_us; 242 243 v_total = div64_u64(div64_u64(((unsigned long long)( 244 duration_in_us) * (stream->timing.pix_clk_100hz / 10)), 245 stream->timing.h_total), 1000); 246 247 /* v_total cannot be less than nominal */ 248 if (v_total < stream->timing.v_total) { 249 ASSERT(v_total < stream->timing.v_total); 250 v_total = stream->timing.v_total; 251 } 252 253 return v_total; 254 } 255 256 static void update_v_total_for_static_ramp( 257 struct core_freesync *core_freesync, 258 const struct dc_stream_state *stream, 259 struct mod_vrr_params *in_out_vrr) 260 { 261 unsigned int v_total = 0; 262 unsigned int current_duration_in_us = 263 calc_duration_in_us_from_v_total( 264 stream, in_out_vrr, 265 in_out_vrr->adjust.v_total_max); 266 unsigned int target_duration_in_us = 267 calc_duration_in_us_from_refresh_in_uhz( 268 in_out_vrr->fixed.target_refresh_in_uhz); 269 bool ramp_direction_is_up = (current_duration_in_us > 270 target_duration_in_us) ? true : false; 271 272 /* Calc ratio between new and current frame duration with 3 digit */ 273 unsigned int frame_duration_ratio = div64_u64(1000000, 274 (1000 + div64_u64(((unsigned long long)( 275 STATIC_SCREEN_RAMP_DELTA_REFRESH_RATE_PER_FRAME) * 276 current_duration_in_us), 277 1000000))); 278 279 /* Calculate delta between new and current frame duration in us */ 280 unsigned int frame_duration_delta = div64_u64(((unsigned long long)( 281 current_duration_in_us) * 282 (1000 - frame_duration_ratio)), 1000); 283 284 /* Adjust frame duration delta based on ratio between current and 285 * standard frame duration (frame duration at 60 Hz refresh rate). 286 */ 287 unsigned int ramp_rate_interpolated = div64_u64(((unsigned long long)( 288 frame_duration_delta) * current_duration_in_us), 16666); 289 290 /* Going to a higher refresh rate (lower frame duration) */ 291 if (ramp_direction_is_up) { 292 /* reduce frame duration */ 293 current_duration_in_us -= ramp_rate_interpolated; 294 295 /* adjust for frame duration below min */ 296 if (current_duration_in_us <= target_duration_in_us) { 297 in_out_vrr->fixed.ramping_active = false; 298 in_out_vrr->fixed.ramping_done = true; 299 current_duration_in_us = 300 calc_duration_in_us_from_refresh_in_uhz( 301 in_out_vrr->fixed.target_refresh_in_uhz); 302 } 303 /* Going to a lower refresh rate (larger frame duration) */ 304 } else { 305 /* increase frame duration */ 306 current_duration_in_us += ramp_rate_interpolated; 307 308 /* adjust for frame duration above max */ 309 if (current_duration_in_us >= target_duration_in_us) { 310 in_out_vrr->fixed.ramping_active = false; 311 in_out_vrr->fixed.ramping_done = true; 312 current_duration_in_us = 313 calc_duration_in_us_from_refresh_in_uhz( 314 in_out_vrr->fixed.target_refresh_in_uhz); 315 } 316 } 317 318 v_total = div64_u64(div64_u64(((unsigned long long)( 319 current_duration_in_us) * (stream->timing.pix_clk_100hz / 10)), 320 stream->timing.h_total), 1000); 321 322 in_out_vrr->adjust.v_total_min = v_total; 323 in_out_vrr->adjust.v_total_max = v_total; 324 } 325 326 static void apply_below_the_range(struct core_freesync *core_freesync, 327 const struct dc_stream_state *stream, 328 unsigned int last_render_time_in_us, 329 struct mod_vrr_params *in_out_vrr) 330 { 331 unsigned int inserted_frame_duration_in_us = 0; 332 unsigned int mid_point_frames_ceil = 0; 333 unsigned int mid_point_frames_floor = 0; 334 unsigned int frame_time_in_us = 0; 335 unsigned int delta_from_mid_point_in_us_1 = 0xFFFFFFFF; 336 unsigned int delta_from_mid_point_in_us_2 = 0xFFFFFFFF; 337 unsigned int frames_to_insert = 0; 338 unsigned int min_frame_duration_in_ns = 0; 339 unsigned int max_render_time_in_us = in_out_vrr->max_duration_in_us; 340 unsigned int delta_from_mid_point_delta_in_us; 341 342 min_frame_duration_in_ns = ((unsigned int) (div64_u64( 343 (1000000000ULL * 1000000), 344 in_out_vrr->max_refresh_in_uhz))); 345 346 /* Program BTR */ 347 if (last_render_time_in_us + BTR_EXIT_MARGIN < max_render_time_in_us) { 348 /* Exit Below the Range */ 349 if (in_out_vrr->btr.btr_active) { 350 in_out_vrr->btr.frame_counter = 0; 351 in_out_vrr->btr.btr_active = false; 352 } 353 } else if (last_render_time_in_us > max_render_time_in_us) { 354 /* Enter Below the Range */ 355 in_out_vrr->btr.btr_active = true; 356 } 357 358 /* BTR set to "not active" so disengage */ 359 if (!in_out_vrr->btr.btr_active) { 360 in_out_vrr->btr.inserted_duration_in_us = 0; 361 in_out_vrr->btr.frames_to_insert = 0; 362 in_out_vrr->btr.frame_counter = 0; 363 364 /* Restore FreeSync */ 365 in_out_vrr->adjust.v_total_min = 366 calc_v_total_from_refresh(stream, 367 in_out_vrr->max_refresh_in_uhz); 368 in_out_vrr->adjust.v_total_max = 369 calc_v_total_from_refresh(stream, 370 in_out_vrr->min_refresh_in_uhz); 371 /* BTR set to "active" so engage */ 372 } else { 373 374 /* Calculate number of midPoint frames that could fit within 375 * the render time interval- take ceil of this value 376 */ 377 mid_point_frames_ceil = (last_render_time_in_us + 378 in_out_vrr->btr.mid_point_in_us - 1) / 379 in_out_vrr->btr.mid_point_in_us; 380 381 if (mid_point_frames_ceil > 0) { 382 frame_time_in_us = last_render_time_in_us / 383 mid_point_frames_ceil; 384 delta_from_mid_point_in_us_1 = 385 (in_out_vrr->btr.mid_point_in_us > 386 frame_time_in_us) ? 387 (in_out_vrr->btr.mid_point_in_us - frame_time_in_us) : 388 (frame_time_in_us - in_out_vrr->btr.mid_point_in_us); 389 } 390 391 /* Calculate number of midPoint frames that could fit within 392 * the render time interval- take floor of this value 393 */ 394 mid_point_frames_floor = last_render_time_in_us / 395 in_out_vrr->btr.mid_point_in_us; 396 397 if (mid_point_frames_floor > 0) { 398 399 frame_time_in_us = last_render_time_in_us / 400 mid_point_frames_floor; 401 delta_from_mid_point_in_us_2 = 402 (in_out_vrr->btr.mid_point_in_us > 403 frame_time_in_us) ? 404 (in_out_vrr->btr.mid_point_in_us - frame_time_in_us) : 405 (frame_time_in_us - in_out_vrr->btr.mid_point_in_us); 406 } 407 408 /* Choose number of frames to insert based on how close it 409 * can get to the mid point of the variable range. 410 */ 411 if (delta_from_mid_point_in_us_1 < delta_from_mid_point_in_us_2) { 412 frames_to_insert = mid_point_frames_ceil; 413 delta_from_mid_point_delta_in_us = delta_from_mid_point_in_us_2 - 414 delta_from_mid_point_in_us_1; 415 } else { 416 frames_to_insert = mid_point_frames_floor; 417 delta_from_mid_point_delta_in_us = delta_from_mid_point_in_us_1 - 418 delta_from_mid_point_in_us_2; 419 } 420 421 /* Prefer current frame multiplier when BTR is enabled unless it drifts 422 * too far from the midpoint 423 */ 424 if (in_out_vrr->btr.frames_to_insert != 0 && 425 delta_from_mid_point_delta_in_us < BTR_DRIFT_MARGIN) { 426 if (((last_render_time_in_us / in_out_vrr->btr.frames_to_insert) < 427 in_out_vrr->max_duration_in_us) && 428 ((last_render_time_in_us / in_out_vrr->btr.frames_to_insert) > 429 in_out_vrr->min_duration_in_us)) 430 frames_to_insert = in_out_vrr->btr.frames_to_insert; 431 } 432 433 /* Either we've calculated the number of frames to insert, 434 * or we need to insert min duration frames 435 */ 436 if (frames_to_insert > 0) 437 inserted_frame_duration_in_us = last_render_time_in_us / 438 frames_to_insert; 439 440 if (inserted_frame_duration_in_us < 441 (1000000 / in_out_vrr->max_refresh_in_uhz)) 442 inserted_frame_duration_in_us = 443 (1000000 / in_out_vrr->max_refresh_in_uhz); 444 445 /* Cache the calculated variables */ 446 in_out_vrr->btr.inserted_duration_in_us = 447 inserted_frame_duration_in_us; 448 in_out_vrr->btr.frames_to_insert = frames_to_insert; 449 in_out_vrr->btr.frame_counter = frames_to_insert; 450 } 451 } 452 453 static void apply_fixed_refresh(struct core_freesync *core_freesync, 454 const struct dc_stream_state *stream, 455 unsigned int last_render_time_in_us, 456 struct mod_vrr_params *in_out_vrr) 457 { 458 bool update = false; 459 unsigned int max_render_time_in_us = in_out_vrr->max_duration_in_us; 460 461 //Compute the exit refresh rate and exit frame duration 462 unsigned int exit_refresh_rate_in_milli_hz = ((1000000000/max_render_time_in_us) 463 + (1000*FIXED_REFRESH_EXIT_MARGIN_IN_HZ)); 464 unsigned int exit_frame_duration_in_us = 1000000000/exit_refresh_rate_in_milli_hz; 465 466 if (last_render_time_in_us < exit_frame_duration_in_us) { 467 /* Exit Fixed Refresh mode */ 468 if (in_out_vrr->fixed.fixed_active) { 469 in_out_vrr->fixed.frame_counter++; 470 471 if (in_out_vrr->fixed.frame_counter > 472 FIXED_REFRESH_EXIT_FRAME_COUNT) { 473 in_out_vrr->fixed.frame_counter = 0; 474 in_out_vrr->fixed.fixed_active = false; 475 in_out_vrr->fixed.target_refresh_in_uhz = 0; 476 update = true; 477 } 478 } 479 } else if (last_render_time_in_us > max_render_time_in_us) { 480 /* Enter Fixed Refresh mode */ 481 if (!in_out_vrr->fixed.fixed_active) { 482 in_out_vrr->fixed.frame_counter++; 483 484 if (in_out_vrr->fixed.frame_counter > 485 FIXED_REFRESH_ENTER_FRAME_COUNT) { 486 in_out_vrr->fixed.frame_counter = 0; 487 in_out_vrr->fixed.fixed_active = true; 488 in_out_vrr->fixed.target_refresh_in_uhz = 489 in_out_vrr->max_refresh_in_uhz; 490 update = true; 491 } 492 } 493 } 494 495 if (update) { 496 if (in_out_vrr->fixed.fixed_active) { 497 in_out_vrr->adjust.v_total_min = 498 calc_v_total_from_refresh( 499 stream, in_out_vrr->max_refresh_in_uhz); 500 in_out_vrr->adjust.v_total_max = 501 in_out_vrr->adjust.v_total_min; 502 } else { 503 in_out_vrr->adjust.v_total_min = 504 calc_v_total_from_refresh(stream, 505 in_out_vrr->max_refresh_in_uhz); 506 in_out_vrr->adjust.v_total_max = 507 calc_v_total_from_refresh(stream, 508 in_out_vrr->min_refresh_in_uhz); 509 } 510 } 511 } 512 513 static bool vrr_settings_require_update(struct core_freesync *core_freesync, 514 struct mod_freesync_config *in_config, 515 unsigned int min_refresh_in_uhz, 516 unsigned int max_refresh_in_uhz, 517 struct mod_vrr_params *in_vrr) 518 { 519 if (in_vrr->state != in_config->state) { 520 return true; 521 } else if (in_vrr->state == VRR_STATE_ACTIVE_FIXED && 522 in_vrr->fixed.target_refresh_in_uhz != 523 in_config->min_refresh_in_uhz) { 524 return true; 525 } else if (in_vrr->min_refresh_in_uhz != min_refresh_in_uhz) { 526 return true; 527 } else if (in_vrr->max_refresh_in_uhz != max_refresh_in_uhz) { 528 return true; 529 } 530 531 return false; 532 } 533 534 bool mod_freesync_get_vmin_vmax(struct mod_freesync *mod_freesync, 535 const struct dc_stream_state *stream, 536 unsigned int *vmin, 537 unsigned int *vmax) 538 { 539 *vmin = stream->adjust.v_total_min; 540 *vmax = stream->adjust.v_total_max; 541 542 return true; 543 } 544 545 bool mod_freesync_get_v_position(struct mod_freesync *mod_freesync, 546 struct dc_stream_state *stream, 547 unsigned int *nom_v_pos, 548 unsigned int *v_pos) 549 { 550 struct core_freesync *core_freesync = NULL; 551 struct crtc_position position; 552 553 if (mod_freesync == NULL) 554 return false; 555 556 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 557 558 if (dc_stream_get_crtc_position(core_freesync->dc, &stream, 1, 559 &position.vertical_count, 560 &position.nominal_vcount)) { 561 562 *nom_v_pos = position.nominal_vcount; 563 *v_pos = position.vertical_count; 564 565 return true; 566 } 567 568 return false; 569 } 570 571 static void build_vrr_infopacket_header_vtem(enum signal_type signal, 572 struct dc_info_packet *infopacket) 573 { 574 // HEADER 575 576 // HB0, HB1, HB2 indicates PacketType VTEMPacket 577 infopacket->hb0 = 0x7F; 578 infopacket->hb1 = 0xC0; 579 infopacket->hb2 = 0x00; //sequence_index 580 581 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB0], MASK__VRR_VTEM_PB0__VFR, 1); 582 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB2], MASK__VRR_VTEM_PB2__ORGANIZATION_ID, 1); 583 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB3], MASK__VRR_VTEM_PB3__DATA_SET_TAG_MSB, 0); 584 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB4], MASK__VRR_VTEM_PB4__DATA_SET_TAG_LSB, 1); 585 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB5], MASK__VRR_VTEM_PB5__DATA_SET_LENGTH_MSB, 0); 586 setFieldWithMask(&infopacket->sb[VRR_VTEM_PB6], MASK__VRR_VTEM_PB6__DATA_SET_LENGTH_LSB, 4); 587 } 588 589 static void build_vrr_infopacket_header_v1(enum signal_type signal, 590 struct dc_info_packet *infopacket, 591 unsigned int *payload_size) 592 { 593 if (dc_is_hdmi_signal(signal)) { 594 595 /* HEADER */ 596 597 /* HB0 = Packet Type = 0x83 (Source Product 598 * Descriptor InfoFrame) 599 */ 600 infopacket->hb0 = DC_HDMI_INFOFRAME_TYPE_SPD; 601 602 /* HB1 = Version = 0x01 */ 603 infopacket->hb1 = 0x01; 604 605 /* HB2 = [Bits 7:5 = 0] [Bits 4:0 = Length = 0x08] */ 606 infopacket->hb2 = 0x08; 607 608 *payload_size = 0x08; 609 610 } else if (dc_is_dp_signal(signal)) { 611 612 /* HEADER */ 613 614 /* HB0 = Secondary-data Packet ID = 0 - Only non-zero 615 * when used to associate audio related info packets 616 */ 617 infopacket->hb0 = 0x00; 618 619 /* HB1 = Packet Type = 0x83 (Source Product 620 * Descriptor InfoFrame) 621 */ 622 infopacket->hb1 = DC_HDMI_INFOFRAME_TYPE_SPD; 623 624 /* HB2 = [Bits 7:0 = Least significant eight bits - 625 * For INFOFRAME, the value must be 1Bh] 626 */ 627 infopacket->hb2 = 0x1B; 628 629 /* HB3 = [Bits 7:2 = INFOFRAME SDP Version Number = 0x1] 630 * [Bits 1:0 = Most significant two bits = 0x00] 631 */ 632 infopacket->hb3 = 0x04; 633 634 *payload_size = 0x1B; 635 } 636 } 637 638 static void build_vrr_infopacket_header_v2(enum signal_type signal, 639 struct dc_info_packet *infopacket, 640 unsigned int *payload_size) 641 { 642 if (dc_is_hdmi_signal(signal)) { 643 644 /* HEADER */ 645 646 /* HB0 = Packet Type = 0x83 (Source Product 647 * Descriptor InfoFrame) 648 */ 649 infopacket->hb0 = DC_HDMI_INFOFRAME_TYPE_SPD; 650 651 /* HB1 = Version = 0x02 */ 652 infopacket->hb1 = 0x02; 653 654 /* HB2 = [Bits 7:5 = 0] [Bits 4:0 = Length = 0x09] */ 655 infopacket->hb2 = 0x09; 656 657 *payload_size = 0x0A; 658 659 } else if (dc_is_dp_signal(signal)) { 660 661 /* HEADER */ 662 663 /* HB0 = Secondary-data Packet ID = 0 - Only non-zero 664 * when used to associate audio related info packets 665 */ 666 infopacket->hb0 = 0x00; 667 668 /* HB1 = Packet Type = 0x83 (Source Product 669 * Descriptor InfoFrame) 670 */ 671 infopacket->hb1 = DC_HDMI_INFOFRAME_TYPE_SPD; 672 673 /* HB2 = [Bits 7:0 = Least significant eight bits - 674 * For INFOFRAME, the value must be 1Bh] 675 */ 676 infopacket->hb2 = 0x1B; 677 678 /* HB3 = [Bits 7:2 = INFOFRAME SDP Version Number = 0x2] 679 * [Bits 1:0 = Most significant two bits = 0x00] 680 */ 681 infopacket->hb3 = 0x08; 682 683 *payload_size = 0x1B; 684 } 685 } 686 687 static void build_vrr_vtem_infopacket_data(const struct dc_stream_state *stream, 688 const struct mod_vrr_params *vrr, 689 struct dc_info_packet *infopacket) 690 { 691 unsigned int fieldRateInHz; 692 693 if (vrr->state == VRR_STATE_ACTIVE_VARIABLE || 694 vrr->state == VRR_STATE_ACTIVE_FIXED) { 695 setFieldWithMask(&infopacket->sb[VRR_VTEM_MD0], MASK__VRR_VTEM_MD0__VRR_EN, 1); 696 } else { 697 setFieldWithMask(&infopacket->sb[VRR_VTEM_MD0], MASK__VRR_VTEM_MD0__VRR_EN, 0); 698 } 699 700 if (!stream->timing.vic) { 701 setFieldWithMask(&infopacket->sb[VRR_VTEM_MD1], MASK__VRR_VTEM_MD1__BASE_VFRONT, 702 stream->timing.v_front_porch); 703 704 705 /* TODO: In dal2, we check mode flags for a reduced blanking timing. 706 * Need a way to relay that information to this function. 707 * if("ReducedBlanking") 708 * { 709 * setFieldWithMask(&infopacket->sb[VRR_VTEM_MD2], MASK__VRR_VTEM_MD2__RB, 1; 710 * } 711 */ 712 713 //TODO: DAL2 does FixPoint and rounding. Here we might need to account for that 714 fieldRateInHz = (stream->timing.pix_clk_100hz * 100)/ 715 (stream->timing.h_total * stream->timing.v_total); 716 717 setFieldWithMask(&infopacket->sb[VRR_VTEM_MD2], MASK__VRR_VTEM_MD2__BASE_REFRESH_RATE_98, 718 fieldRateInHz >> 8); 719 setFieldWithMask(&infopacket->sb[VRR_VTEM_MD3], MASK__VRR_VTEM_MD3__BASE_REFRESH_RATE_07, 720 fieldRateInHz); 721 722 } 723 infopacket->valid = true; 724 } 725 726 static void build_vrr_infopacket_data(const struct mod_vrr_params *vrr, 727 struct dc_info_packet *infopacket) 728 { 729 /* PB1 = 0x1A (24bit AMD IEEE OUI (0x00001A) - Byte 0) */ 730 infopacket->sb[1] = 0x1A; 731 732 /* PB2 = 0x00 (24bit AMD IEEE OUI (0x00001A) - Byte 1) */ 733 infopacket->sb[2] = 0x00; 734 735 /* PB3 = 0x00 (24bit AMD IEEE OUI (0x00001A) - Byte 2) */ 736 infopacket->sb[3] = 0x00; 737 738 /* PB4 = Reserved */ 739 740 /* PB5 = Reserved */ 741 742 /* PB6 = [Bits 7:3 = Reserved] */ 743 744 /* PB6 = [Bit 0 = FreeSync Supported] */ 745 if (vrr->state != VRR_STATE_UNSUPPORTED) 746 infopacket->sb[6] |= 0x01; 747 748 /* PB6 = [Bit 1 = FreeSync Enabled] */ 749 if (vrr->state != VRR_STATE_DISABLED && 750 vrr->state != VRR_STATE_UNSUPPORTED) 751 infopacket->sb[6] |= 0x02; 752 753 /* PB6 = [Bit 2 = FreeSync Active] */ 754 if (vrr->state == VRR_STATE_ACTIVE_VARIABLE || 755 vrr->state == VRR_STATE_ACTIVE_FIXED) 756 infopacket->sb[6] |= 0x04; 757 758 /* PB7 = FreeSync Minimum refresh rate (Hz) */ 759 infopacket->sb[7] = (unsigned char)(vrr->min_refresh_in_uhz / 1000000); 760 761 /* PB8 = FreeSync Maximum refresh rate (Hz) 762 * Note: We should never go above the field rate of the mode timing set. 763 */ 764 infopacket->sb[8] = (unsigned char)(vrr->max_refresh_in_uhz / 1000000); 765 766 767 //FreeSync HDR 768 infopacket->sb[9] = 0; 769 infopacket->sb[10] = 0; 770 } 771 772 static void build_vrr_infopacket_fs2_data(enum color_transfer_func app_tf, 773 struct dc_info_packet *infopacket) 774 { 775 if (app_tf != TRANSFER_FUNC_UNKNOWN) { 776 infopacket->valid = true; 777 778 infopacket->sb[6] |= 0x08; // PB6 = [Bit 3 = Native Color Active] 779 780 if (app_tf == TRANSFER_FUNC_GAMMA_22) { 781 infopacket->sb[9] |= 0x04; // PB6 = [Bit 2 = Gamma 2.2 EOTF Active] 782 } 783 } 784 } 785 786 static void build_vrr_infopacket_checksum(unsigned int *payload_size, 787 struct dc_info_packet *infopacket) 788 { 789 /* Calculate checksum */ 790 unsigned int idx = 0; 791 unsigned char checksum = 0; 792 793 checksum += infopacket->hb0; 794 checksum += infopacket->hb1; 795 checksum += infopacket->hb2; 796 checksum += infopacket->hb3; 797 798 for (idx = 1; idx <= *payload_size; idx++) 799 checksum += infopacket->sb[idx]; 800 801 /* PB0 = Checksum (one byte complement) */ 802 infopacket->sb[0] = (unsigned char)(0x100 - checksum); 803 804 infopacket->valid = true; 805 } 806 807 static void build_vrr_infopacket_v1(enum signal_type signal, 808 const struct mod_vrr_params *vrr, 809 struct dc_info_packet *infopacket) 810 { 811 /* SPD info packet for FreeSync */ 812 unsigned int payload_size = 0; 813 814 build_vrr_infopacket_header_v1(signal, infopacket, &payload_size); 815 build_vrr_infopacket_data(vrr, infopacket); 816 build_vrr_infopacket_checksum(&payload_size, infopacket); 817 818 infopacket->valid = true; 819 } 820 821 static void build_vrr_infopacket_v2(enum signal_type signal, 822 const struct mod_vrr_params *vrr, 823 enum color_transfer_func app_tf, 824 struct dc_info_packet *infopacket) 825 { 826 unsigned int payload_size = 0; 827 828 build_vrr_infopacket_header_v2(signal, infopacket, &payload_size); 829 build_vrr_infopacket_data(vrr, infopacket); 830 831 build_vrr_infopacket_fs2_data(app_tf, infopacket); 832 833 build_vrr_infopacket_checksum(&payload_size, infopacket); 834 835 infopacket->valid = true; 836 } 837 838 static void build_vrr_infopacket_vtem(const struct dc_stream_state *stream, 839 const struct mod_vrr_params *vrr, 840 struct dc_info_packet *infopacket) 841 { 842 //VTEM info packet for HdmiVrr 843 844 memset(infopacket, 0, sizeof(struct dc_info_packet)); 845 846 //VTEM Packet is structured differently 847 build_vrr_infopacket_header_vtem(stream->signal, infopacket); 848 build_vrr_vtem_infopacket_data(stream, vrr, infopacket); 849 850 infopacket->valid = true; 851 } 852 853 void mod_freesync_build_vrr_infopacket(struct mod_freesync *mod_freesync, 854 const struct dc_stream_state *stream, 855 const struct mod_vrr_params *vrr, 856 enum vrr_packet_type packet_type, 857 enum color_transfer_func app_tf, 858 struct dc_info_packet *infopacket) 859 { 860 /* SPD info packet for FreeSync 861 * VTEM info packet for HdmiVRR 862 * Check if Freesync is supported. Return if false. If true, 863 * set the corresponding bit in the info packet 864 */ 865 if (!vrr->supported || (!vrr->send_info_frame && packet_type != PACKET_TYPE_VTEM)) 866 return; 867 868 switch (packet_type) { 869 case PACKET_TYPE_FS2: 870 build_vrr_infopacket_v2(stream->signal, vrr, app_tf, infopacket); 871 break; 872 case PACKET_TYPE_VTEM: 873 build_vrr_infopacket_vtem(stream, vrr, infopacket); 874 break; 875 case PACKET_TYPE_VRR: 876 case PACKET_TYPE_FS1: 877 default: 878 build_vrr_infopacket_v1(stream->signal, vrr, infopacket); 879 } 880 } 881 882 void mod_freesync_build_vrr_params(struct mod_freesync *mod_freesync, 883 const struct dc_stream_state *stream, 884 struct mod_freesync_config *in_config, 885 struct mod_vrr_params *in_out_vrr) 886 { 887 struct core_freesync *core_freesync = NULL; 888 unsigned long long nominal_field_rate_in_uhz = 0; 889 unsigned int refresh_range = 0; 890 unsigned int min_refresh_in_uhz = 0; 891 unsigned int max_refresh_in_uhz = 0; 892 893 if (mod_freesync == NULL) 894 return; 895 896 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 897 898 /* Calculate nominal field rate for stream */ 899 nominal_field_rate_in_uhz = 900 mod_freesync_calc_nominal_field_rate(stream); 901 902 min_refresh_in_uhz = in_config->min_refresh_in_uhz; 903 max_refresh_in_uhz = in_config->max_refresh_in_uhz; 904 905 // Don't allow min > max 906 if (min_refresh_in_uhz > max_refresh_in_uhz) 907 min_refresh_in_uhz = max_refresh_in_uhz; 908 909 // Full range may be larger than current video timing, so cap at nominal 910 if (max_refresh_in_uhz > nominal_field_rate_in_uhz) 911 max_refresh_in_uhz = nominal_field_rate_in_uhz; 912 913 // Full range may be larger than current video timing, so cap at nominal 914 if (min_refresh_in_uhz > nominal_field_rate_in_uhz) 915 min_refresh_in_uhz = nominal_field_rate_in_uhz; 916 917 if (!vrr_settings_require_update(core_freesync, 918 in_config, min_refresh_in_uhz, max_refresh_in_uhz, 919 in_out_vrr)) 920 return; 921 922 in_out_vrr->state = in_config->state; 923 in_out_vrr->send_info_frame = in_config->vsif_supported; 924 925 if (in_config->state == VRR_STATE_UNSUPPORTED) { 926 in_out_vrr->state = VRR_STATE_UNSUPPORTED; 927 in_out_vrr->supported = false; 928 in_out_vrr->adjust.v_total_min = stream->timing.v_total; 929 in_out_vrr->adjust.v_total_max = stream->timing.v_total; 930 931 return; 932 933 } else { 934 in_out_vrr->min_refresh_in_uhz = min_refresh_in_uhz; 935 in_out_vrr->max_duration_in_us = 936 calc_duration_in_us_from_refresh_in_uhz( 937 min_refresh_in_uhz); 938 939 in_out_vrr->max_refresh_in_uhz = max_refresh_in_uhz; 940 in_out_vrr->min_duration_in_us = 941 calc_duration_in_us_from_refresh_in_uhz( 942 max_refresh_in_uhz); 943 944 refresh_range = in_out_vrr->max_refresh_in_uhz - 945 in_out_vrr->min_refresh_in_uhz; 946 947 in_out_vrr->supported = true; 948 } 949 950 in_out_vrr->fixed.ramping_active = in_config->ramping; 951 952 in_out_vrr->btr.btr_enabled = in_config->btr; 953 if (in_out_vrr->max_refresh_in_uhz < 954 2 * in_out_vrr->min_refresh_in_uhz) 955 in_out_vrr->btr.btr_enabled = false; 956 in_out_vrr->btr.btr_active = false; 957 in_out_vrr->btr.inserted_duration_in_us = 0; 958 in_out_vrr->btr.frames_to_insert = 0; 959 in_out_vrr->btr.frame_counter = 0; 960 in_out_vrr->btr.mid_point_in_us = 961 in_out_vrr->min_duration_in_us + 962 (in_out_vrr->max_duration_in_us - 963 in_out_vrr->min_duration_in_us) / 2; 964 965 if (in_out_vrr->state == VRR_STATE_UNSUPPORTED) { 966 in_out_vrr->adjust.v_total_min = stream->timing.v_total; 967 in_out_vrr->adjust.v_total_max = stream->timing.v_total; 968 } else if (in_out_vrr->state == VRR_STATE_DISABLED) { 969 in_out_vrr->adjust.v_total_min = stream->timing.v_total; 970 in_out_vrr->adjust.v_total_max = stream->timing.v_total; 971 } else if (in_out_vrr->state == VRR_STATE_INACTIVE) { 972 in_out_vrr->adjust.v_total_min = stream->timing.v_total; 973 in_out_vrr->adjust.v_total_max = stream->timing.v_total; 974 } else if (in_out_vrr->state == VRR_STATE_ACTIVE_VARIABLE && 975 refresh_range >= MIN_REFRESH_RANGE_IN_US) { 976 in_out_vrr->adjust.v_total_min = 977 calc_v_total_from_refresh(stream, 978 in_out_vrr->max_refresh_in_uhz); 979 in_out_vrr->adjust.v_total_max = 980 calc_v_total_from_refresh(stream, 981 in_out_vrr->min_refresh_in_uhz); 982 } else if (in_out_vrr->state == VRR_STATE_ACTIVE_FIXED) { 983 in_out_vrr->fixed.target_refresh_in_uhz = 984 in_out_vrr->min_refresh_in_uhz; 985 if (in_out_vrr->fixed.ramping_active && 986 in_out_vrr->fixed.fixed_active) { 987 /* Do not update vtotals if ramping is already active 988 * in order to continue ramp from current refresh. 989 */ 990 in_out_vrr->fixed.fixed_active = true; 991 } else { 992 in_out_vrr->fixed.fixed_active = true; 993 in_out_vrr->adjust.v_total_min = 994 calc_v_total_from_refresh(stream, 995 in_out_vrr->fixed.target_refresh_in_uhz); 996 in_out_vrr->adjust.v_total_max = 997 in_out_vrr->adjust.v_total_min; 998 } 999 } else { 1000 in_out_vrr->state = VRR_STATE_INACTIVE; 1001 in_out_vrr->adjust.v_total_min = stream->timing.v_total; 1002 in_out_vrr->adjust.v_total_max = stream->timing.v_total; 1003 } 1004 } 1005 1006 void mod_freesync_handle_preflip(struct mod_freesync *mod_freesync, 1007 const struct dc_plane_state *plane, 1008 const struct dc_stream_state *stream, 1009 unsigned int curr_time_stamp_in_us, 1010 struct mod_vrr_params *in_out_vrr) 1011 { 1012 struct core_freesync *core_freesync = NULL; 1013 unsigned int last_render_time_in_us = 0; 1014 unsigned int average_render_time_in_us = 0; 1015 1016 if (mod_freesync == NULL) 1017 return; 1018 1019 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 1020 1021 if (in_out_vrr->supported && 1022 in_out_vrr->state == VRR_STATE_ACTIVE_VARIABLE) { 1023 unsigned int i = 0; 1024 unsigned int oldest_index = plane->time.index + 1; 1025 1026 if (oldest_index >= DC_PLANE_UPDATE_TIMES_MAX) 1027 oldest_index = 0; 1028 1029 last_render_time_in_us = curr_time_stamp_in_us - 1030 plane->time.prev_update_time_in_us; 1031 1032 // Sum off all entries except oldest one 1033 for (i = 0; i < DC_PLANE_UPDATE_TIMES_MAX; i++) { 1034 average_render_time_in_us += 1035 plane->time.time_elapsed_in_us[i]; 1036 } 1037 average_render_time_in_us -= 1038 plane->time.time_elapsed_in_us[oldest_index]; 1039 1040 // Add render time for current flip 1041 average_render_time_in_us += last_render_time_in_us; 1042 average_render_time_in_us /= DC_PLANE_UPDATE_TIMES_MAX; 1043 1044 if (in_out_vrr->btr.btr_enabled) { 1045 apply_below_the_range(core_freesync, 1046 stream, 1047 last_render_time_in_us, 1048 in_out_vrr); 1049 } else { 1050 apply_fixed_refresh(core_freesync, 1051 stream, 1052 last_render_time_in_us, 1053 in_out_vrr); 1054 } 1055 1056 } 1057 } 1058 1059 void mod_freesync_handle_v_update(struct mod_freesync *mod_freesync, 1060 const struct dc_stream_state *stream, 1061 struct mod_vrr_params *in_out_vrr) 1062 { 1063 struct core_freesync *core_freesync = NULL; 1064 1065 if ((mod_freesync == NULL) || (stream == NULL) || (in_out_vrr == NULL)) 1066 return; 1067 1068 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 1069 1070 if (in_out_vrr->supported == false) 1071 return; 1072 1073 /* Below the Range Logic */ 1074 1075 /* Only execute if in fullscreen mode */ 1076 if (in_out_vrr->state == VRR_STATE_ACTIVE_VARIABLE && 1077 in_out_vrr->btr.btr_active) { 1078 /* TODO: pass in flag for Pre-DCE12 ASIC 1079 * in order for frame variable duration to take affect, 1080 * it needs to be done one VSYNC early, which is at 1081 * frameCounter == 1. 1082 * For DCE12 and newer updates to V_TOTAL_MIN/MAX 1083 * will take affect on current frame 1084 */ 1085 if (in_out_vrr->btr.frames_to_insert == 1086 in_out_vrr->btr.frame_counter) { 1087 in_out_vrr->adjust.v_total_min = 1088 calc_v_total_from_duration(stream, 1089 in_out_vrr, 1090 in_out_vrr->btr.inserted_duration_in_us); 1091 in_out_vrr->adjust.v_total_max = 1092 in_out_vrr->adjust.v_total_min; 1093 } 1094 1095 if (in_out_vrr->btr.frame_counter > 0) 1096 in_out_vrr->btr.frame_counter--; 1097 1098 /* Restore FreeSync */ 1099 if (in_out_vrr->btr.frame_counter == 0) { 1100 in_out_vrr->adjust.v_total_min = 1101 calc_v_total_from_refresh(stream, 1102 in_out_vrr->max_refresh_in_uhz); 1103 in_out_vrr->adjust.v_total_max = 1104 calc_v_total_from_refresh(stream, 1105 in_out_vrr->min_refresh_in_uhz); 1106 } 1107 } 1108 1109 /* If in fullscreen freesync mode or in video, do not program 1110 * static screen ramp values 1111 */ 1112 if (in_out_vrr->state == VRR_STATE_ACTIVE_VARIABLE) 1113 in_out_vrr->fixed.ramping_active = false; 1114 1115 /* Gradual Static Screen Ramping Logic */ 1116 /* Execute if ramp is active and user enabled freesync static screen*/ 1117 if (in_out_vrr->state == VRR_STATE_ACTIVE_FIXED && 1118 in_out_vrr->fixed.ramping_active) { 1119 update_v_total_for_static_ramp( 1120 core_freesync, stream, in_out_vrr); 1121 } 1122 } 1123 1124 void mod_freesync_get_settings(struct mod_freesync *mod_freesync, 1125 const struct mod_vrr_params *vrr, 1126 unsigned int *v_total_min, unsigned int *v_total_max, 1127 unsigned int *event_triggers, 1128 unsigned int *window_min, unsigned int *window_max, 1129 unsigned int *lfc_mid_point_in_us, 1130 unsigned int *inserted_frames, 1131 unsigned int *inserted_duration_in_us) 1132 { 1133 struct core_freesync *core_freesync = NULL; 1134 1135 if (mod_freesync == NULL) 1136 return; 1137 1138 core_freesync = MOD_FREESYNC_TO_CORE(mod_freesync); 1139 1140 if (vrr->supported) { 1141 *v_total_min = vrr->adjust.v_total_min; 1142 *v_total_max = vrr->adjust.v_total_max; 1143 *event_triggers = 0; 1144 *lfc_mid_point_in_us = vrr->btr.mid_point_in_us; 1145 *inserted_frames = vrr->btr.frames_to_insert; 1146 *inserted_duration_in_us = vrr->btr.inserted_duration_in_us; 1147 } 1148 } 1149 1150 unsigned long long mod_freesync_calc_nominal_field_rate( 1151 const struct dc_stream_state *stream) 1152 { 1153 unsigned long long nominal_field_rate_in_uhz = 0; 1154 1155 /* Calculate nominal field rate for stream */ 1156 nominal_field_rate_in_uhz = stream->timing.pix_clk_100hz / 10; 1157 nominal_field_rate_in_uhz *= 1000ULL * 1000ULL * 1000ULL; 1158 nominal_field_rate_in_uhz = div_u64(nominal_field_rate_in_uhz, 1159 stream->timing.h_total); 1160 nominal_field_rate_in_uhz = div_u64(nominal_field_rate_in_uhz, 1161 stream->timing.v_total); 1162 1163 return nominal_field_rate_in_uhz; 1164 } 1165 1166 bool mod_freesync_is_valid_range(struct mod_freesync *mod_freesync, 1167 const struct dc_stream_state *stream, 1168 uint32_t min_refresh_cap_in_uhz, 1169 uint32_t max_refresh_cap_in_uhz, 1170 uint32_t min_refresh_request_in_uhz, 1171 uint32_t max_refresh_request_in_uhz) 1172 { 1173 /* Calculate nominal field rate for stream */ 1174 unsigned long long nominal_field_rate_in_uhz = 1175 mod_freesync_calc_nominal_field_rate(stream); 1176 1177 /* Typically nominal refresh calculated can have some fractional part. 1178 * Allow for some rounding error of actual video timing by taking floor 1179 * of caps and request. Round the nominal refresh rate. 1180 * 1181 * Dividing will convert everything to units in Hz although input 1182 * variable name is in uHz! 1183 * 1184 * Also note, this takes care of rounding error on the nominal refresh 1185 * so by rounding error we only expect it to be off by a small amount, 1186 * such as < 0.1 Hz. i.e. 143.9xxx or 144.1xxx. 1187 * 1188 * Example 1. Caps Min = 40 Hz, Max = 144 Hz 1189 * Request Min = 40 Hz, Max = 144 Hz 1190 * Nominal = 143.5x Hz rounded to 144 Hz 1191 * This function should allow this as valid request 1192 * 1193 * Example 2. Caps Min = 40 Hz, Max = 144 Hz 1194 * Request Min = 40 Hz, Max = 144 Hz 1195 * Nominal = 144.4x Hz rounded to 144 Hz 1196 * This function should allow this as valid request 1197 * 1198 * Example 3. Caps Min = 40 Hz, Max = 144 Hz 1199 * Request Min = 40 Hz, Max = 144 Hz 1200 * Nominal = 120.xx Hz rounded to 120 Hz 1201 * This function should return NOT valid since the requested 1202 * max is greater than current timing's nominal 1203 * 1204 * Example 4. Caps Min = 40 Hz, Max = 120 Hz 1205 * Request Min = 40 Hz, Max = 120 Hz 1206 * Nominal = 144.xx Hz rounded to 144 Hz 1207 * This function should return NOT valid since the nominal 1208 * is greater than the capability's max refresh 1209 */ 1210 nominal_field_rate_in_uhz = 1211 div_u64(nominal_field_rate_in_uhz + 500000, 1000000); 1212 min_refresh_cap_in_uhz /= 1000000; 1213 max_refresh_cap_in_uhz /= 1000000; 1214 min_refresh_request_in_uhz /= 1000000; 1215 max_refresh_request_in_uhz /= 1000000; 1216 1217 // Check nominal is within range 1218 if (nominal_field_rate_in_uhz > max_refresh_cap_in_uhz || 1219 nominal_field_rate_in_uhz < min_refresh_cap_in_uhz) 1220 return false; 1221 1222 // If nominal is less than max, limit the max allowed refresh rate 1223 if (nominal_field_rate_in_uhz < max_refresh_cap_in_uhz) 1224 max_refresh_cap_in_uhz = nominal_field_rate_in_uhz; 1225 1226 // Don't allow min > max 1227 if (min_refresh_request_in_uhz > max_refresh_request_in_uhz) 1228 return false; 1229 1230 // Check min is within range 1231 if (min_refresh_request_in_uhz > max_refresh_cap_in_uhz || 1232 min_refresh_request_in_uhz < min_refresh_cap_in_uhz) 1233 return false; 1234 1235 // Check max is within range 1236 if (max_refresh_request_in_uhz > max_refresh_cap_in_uhz || 1237 max_refresh_request_in_uhz < min_refresh_cap_in_uhz) 1238 return false; 1239 1240 // For variable range, check for at least 10 Hz range 1241 if ((max_refresh_request_in_uhz != min_refresh_request_in_uhz) && 1242 (max_refresh_request_in_uhz - min_refresh_request_in_uhz < 10)) 1243 return false; 1244 1245 return true; 1246 } 1247 1248