mirror of
https://github.com/libretro/RetroArch.git
synced 2024-11-27 10:10:57 +00:00
4b2f27eff8
Fixed SR close match refresh bug. Added menu high resolution option. Fixed desktop restore bug cuusing endless resolution change requests. Fixed file conflicts Added destop restore resolution back in for manu only. Pulled Switchres fixes. Added better PI rsolution support. Ver 0.7 SR2 (Switchres API) Implimantation Removed HH experimetal check. This is better done via teh switchres.ini at present. Fixed refresh rate bug. Now new resolution and refesh is added correctly. Removed SR deinit from menu restore. Meanu now stays at last content resolution. Ver 0.6.2 SR2 (Switchres API) Implimantation Fixed super resolution bug casuing abnormal video size and aspect ratio Fixed logging issue casuing seg falts on RA exit Ver 0.6 SR2 (Switchres API) Implimantation Ver 0.2 SR2 (Switchres API) Implimantation Added forced super resolutions. Added Multi-monitor/monitor selection support. Added desktop resolution restore when switching back to menu only. Added new menu items for 31KHz standard and 120hz monitor profiles. Added new menu item INI. load monitor profile from switchrss.ini. Fixed winraw driver. Coordinates new refreshed after a resolution change. Fixed Menu aspect ratio in super resolutions. Removed static glabals. These have been added to videocrt_switch struct. Ver 0.1 SR2 (Switchres API) Implimantation Removed old CRTSwitchRes method. Added new SR2 API implimantaion. Resolution swithcing is now done by switchres libs. Both Linux and Windows working with native and super resolutions. Working multi-monitor support with monitor index selection. Working 31KHz support with standard and 120Hxz modes. The monitor index selection is still done via the RA UI. Only choose native and 15KHz form the CRT options in the RA UI as all options are now set in the switchres.ini. All other CRT optoins in the RA UI currently do nothing. Added SR wrapper to fix compile issues. Added back RPi functionality Fixed windows resize/scaling issues on resolution change Thanks @Calamity no more need for crt_switch_driver_refresh() Fix broken case after prevous commit Monitor preset options 15/31KHz now active. Added new meu option. Moitor persets can now be choosen fom the RA UI. 15KHz and 31KHz will set arcade_15 and aracde_31 respectivly. New option INI, if this is chosen your monitor preset will be selected from your switchres.ini file. Added 3KHhz, 120Hz. for old RA users. Renamed 31KHz to 31 KHz, Standard Fixed winraw input coordinates after switching resolution. Code cleanup Fixed menu aspect ratio issue Added menu resolution restore after closing content Fixed aspect ratio after menu resolution restore. code clean up Fxed menu Resulition Restore Aspect Ratio. When SR uses non integer scalled resolution. super width bug with restoring menu resolution fix added super resolution check after setting desktop resolutoion variables when menu active only sr_deinit() used to restore desktop mode. Fixed menu sr_deinit bug. now setting sr_active false Removed static globals, added them to video_switch struct Fixex compile bug due to comment // Fixed compile issues doe to c++ comments in teh switchres_wrapper.h Temporarily removed SR2 logging to fix compile isses for c90 added logging back in. Removed support for winnt and osx Added define for C89. Disabled SR if defined C89 Removed all RA compile fixes fro C89 C90 etc. Swithing now working again. Put Switchres behind HAVE_SR2. HAVE_SR2=no by default. --enable-sr2 Ver 0.5 SR2 Implimentation. Ver 0.4 SR2 Implimantation. Bake SR inside RA Removed temporary log files Disable switchres when C89/C99 builds. Removed C89 and C90 checks for SR Fixed switchres_wrapper.h location Ver 0.3 SR2 Implimenation Dissable logging for C89 __STDC__ Fix For RPi fixed missing EOL fixed RPi function definition added vidrocrt_switch stuct to RPI funcion fixed xoffset for RPi Removed old RPi function call SR disabled for videocore until VC4 switching ported Reverted back to state 5c8a56c Bake SR inside RA Use native win32 api for threads. Fix static lib linking LIBERROR would be defined twice otherwise + improper function names prefixed by __imp_ Added lidstc++ to makefile.common for switchres Fixed RPi switching. Disabled Switchres for videocore unill it is ported. removed RAA.log. Should not exist Added check for when SR fails to set mode with an aspect ratio fix. added video driver re init for RPi GB, GBA and GBC core check, adjusted reseolutions and scale. Please turn on integer scalling in the RA UI Added logas back in. Checking STDC verstion >= C11 Fixed c89 for loop declaration. Code clean up. Added new functions Fix resolution switching bug introduces with HH code clean up. Fixed menu restore bug on closw content after code clean up Moved SR logging to relevant RA logs Update makefile. Checks for X11 and xrandr fixed makefile Use native win32 api for threads. Fix static lib linking LIBERROR would be defined twice otherwise + improper function names prefixed by __imp_ Update switchres_wrapper.* header comments Update year copyright DRMKMS: build only if libdrm has the required version XRANDR: build only if xrandr is available Simplified maklefile Fixed RPI compile error with unsued functions. As before Disable Griffin. No switching support available. Never has been Removed log file 1 Added Win32 static define Added SR source Removed Videocore check on destroy SR Moved SR deinit to trigger earlier on RA exit. Fixed compile error after upstream rebase Fixed aspect ration bug cused by super resolutions. Temporarily disbabled SR logging Re inabled runtim eSR loggind. Disableed all RARCH logging on retro_deinit_drivers Removed srdeinit from menu restore. Menu stays in current reolution until a fix can be found Fixed refresh rate changes when no reolution change is detected. Forgot to add teh resolution cahge in with the refresh change oops Fixed endless no detection log. Removed HH check. This can been better adjusted using the switchres.ini fixed compile issue Added better PI crt switching and fixed typo Pulled Swicthres fixes. Updated desktop restore resolution. removed unused makefile Lockec menu refresh to 60hz fixed missing new line Fixed file conflicts Forced 640x480@60 for menu Added high resolution menu option Removed item logg checker Fixed typos Removed unused functions Fixed SR close match refesh bug. Fixed typo
772 lines
27 KiB
C++
772 lines
27 KiB
C++
/**************************************************************
|
|
|
|
modeline.cpp - Modeline generation and scoring routines
|
|
|
|
---------------------------------------------------------
|
|
|
|
Switchres Modeline generation engine for emulation
|
|
|
|
License GPL-2.0+
|
|
Copyright 2010-2021 Chris Kennedy, Antonio Giner,
|
|
Alexandre Wodarczyk, Gil Delescluse
|
|
|
|
**************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <cstddef>
|
|
#include "modeline.h"
|
|
#include "log.h"
|
|
|
|
#define max(a,b)({ __typeof__ (a) _a = (a);__typeof__ (b) _b = (b);_a > _b ? _a : _b; })
|
|
#define min(a,b)({ __typeof__ (a) _a = (a);__typeof__ (b) _b = (b);_a < _b ? _a : _b; })
|
|
|
|
|
|
//============================================================
|
|
// PROTOTYPES
|
|
//============================================================
|
|
|
|
int get_line_params(modeline *mode, monitor_range *range, int char_size);
|
|
int scale_into_range (int value, int lower_limit, int higher_limit);
|
|
int scale_into_range (double value, double lower_limit, double higher_limit);
|
|
int scale_into_aspect (int source_res, int tot_res, double original_monitor_aspect, double users_monitor_aspect, double *best_diff);
|
|
int stretch_into_range(double vfreq, monitor_range *range, double borders, bool interlace_allowed, double *interlace);
|
|
int total_lines_for_yres(int yres, double vfreq, monitor_range *range, double borders, double interlace);
|
|
double max_vfreq_for_yres (int yres, monitor_range *range, double borders, double interlace);
|
|
|
|
//============================================================
|
|
// modeline_create
|
|
//============================================================
|
|
|
|
int modeline_create(modeline *s_mode, modeline *t_mode, monitor_range *range, generator_settings *cs)
|
|
{
|
|
double vfreq_real = 0;
|
|
double interlace = 1;
|
|
double doublescan = 1;
|
|
double scan_factor = 1;
|
|
int x_scale = 0;
|
|
int y_scale = 0;
|
|
int v_scale = 0;
|
|
double x_diff = 0;
|
|
double y_diff = 0;
|
|
double v_diff = 0;
|
|
double y_ratio = 0;
|
|
double x_ratio = 0;
|
|
double borders = 0;
|
|
t_mode->result.weight = 0;
|
|
|
|
// ≈≈≈ Vertical refresh ≈≈≈
|
|
// try to fit vertical frequency into current range
|
|
v_scale = scale_into_range(t_mode->vfreq, range->vfreq_min, range->vfreq_max);
|
|
|
|
if (!v_scale && (t_mode->type & V_FREQ_EDITABLE))
|
|
{
|
|
t_mode->vfreq = t_mode->vfreq < range->vfreq_min? range->vfreq_min : range->vfreq_max;
|
|
v_scale = 1;
|
|
}
|
|
else if (v_scale != 1 && !(t_mode->type & V_FREQ_EDITABLE))
|
|
{
|
|
t_mode->result.weight |= R_OUT_OF_RANGE;
|
|
return -1;
|
|
}
|
|
|
|
// ≈≈≈ Vertical resolution ≈≈≈
|
|
// try to fit active lines in the progressive range first
|
|
if (range->progressive_lines_min && (!t_mode->interlace || (t_mode->type & SCAN_EDITABLE)))
|
|
y_scale = scale_into_range(t_mode->vactive, range->progressive_lines_min, range->progressive_lines_max);
|
|
|
|
// if not possible, try to fit in the interlaced range, if any
|
|
if (!y_scale && range->interlaced_lines_min && cs->interlace && (t_mode->interlace || (t_mode->type & SCAN_EDITABLE)))
|
|
{
|
|
y_scale = scale_into_range(t_mode->vactive, range->interlaced_lines_min, range->interlaced_lines_max);
|
|
interlace = 2;
|
|
}
|
|
|
|
// if we succeeded, let's see if we can apply integer scaling
|
|
if (y_scale == 1 || (y_scale > 1 && (t_mode->type & Y_RES_EDITABLE)))
|
|
{
|
|
// check if we should apply doublescan
|
|
if (cs->doublescan && y_scale % 2 == 0)
|
|
{
|
|
y_scale /= 2;
|
|
doublescan = 0.5;
|
|
}
|
|
scan_factor = interlace * doublescan;
|
|
|
|
// Calculate top border in case of multi-standard consumer TVs
|
|
if (cs->v_shift_correct)
|
|
borders = (range->progressive_lines_max - t_mode->vactive * y_scale / interlace) * (1.0 / range->hfreq_min) / 2;
|
|
|
|
// calculate expected achievable refresh for this height
|
|
vfreq_real = min(t_mode->vfreq * v_scale, max_vfreq_for_yres(t_mode->vactive * y_scale, range, borders, scan_factor));
|
|
if (vfreq_real != t_mode->vfreq * v_scale && !(t_mode->type & V_FREQ_EDITABLE))
|
|
{
|
|
t_mode->result.weight |= R_OUT_OF_RANGE;
|
|
return -1;
|
|
}
|
|
|
|
// calculate the ratio that our scaled yres represents with respect to the original height
|
|
y_ratio = double(t_mode->vactive) * y_scale / s_mode->vactive;
|
|
int y_source_scaled = s_mode->vactive * floor(y_ratio);
|
|
|
|
// if our original height doesn't fit the target height, we're forced to stretch
|
|
if (!y_source_scaled)
|
|
t_mode->result.weight |= R_RES_STRETCH;
|
|
|
|
// otherwise we try to perform integer scaling
|
|
else
|
|
{
|
|
// exclude lcd ranges from raw border computation
|
|
if (t_mode->type & V_FREQ_EDITABLE && range->progressive_lines_max - range->progressive_lines_min > 0)
|
|
{
|
|
// calculate y borders considering physical lines (instead of logical resolution)
|
|
int tot_yres = total_lines_for_yres(t_mode->vactive * y_scale, vfreq_real, range, borders, scan_factor);
|
|
int tot_source = total_lines_for_yres(y_source_scaled, t_mode->vfreq * v_scale, range, borders, scan_factor);
|
|
y_diff = tot_yres > tot_source?double(tot_yres % tot_source) / tot_yres * 100:0;
|
|
|
|
// we penalize for the logical lines we need to add in order to meet the user's lower active lines limit
|
|
int y_min = interlace == 2?range->interlaced_lines_min:range->progressive_lines_min;
|
|
int tot_rest = (y_min >= y_source_scaled / doublescan)? y_min % int(y_source_scaled / doublescan):0;
|
|
y_diff += double(tot_rest) / tot_yres * 100;
|
|
}
|
|
else
|
|
y_diff = double((t_mode->vactive * y_scale) % y_source_scaled) / (t_mode->vactive * y_scale) * 100;
|
|
|
|
// we save the integer ratio between source and target resolutions, this will be used for prescaling
|
|
y_scale = floor(y_ratio);
|
|
|
|
// now if the borders obtained are low enough (< 10%) we'll finally apply integer scaling
|
|
// otherwise we'll stretch the original resolution over the target one
|
|
if (!(y_ratio >= 1.0 && y_ratio < 16.0 && y_diff < 10.0))
|
|
t_mode->result.weight |= R_RES_STRETCH;
|
|
}
|
|
}
|
|
|
|
// otherwise, check if we're allowed to apply fractional scaling
|
|
else if (t_mode->type & Y_RES_EDITABLE)
|
|
t_mode->result.weight |= R_RES_STRETCH;
|
|
|
|
// if there's nothing we can do, we're out of range
|
|
else
|
|
{
|
|
t_mode->result.weight |= R_OUT_OF_RANGE;
|
|
return -1;
|
|
}
|
|
|
|
// ≈≈≈ Horizontal resolution ≈≈≈
|
|
// make the best possible adjustment of xres depending on what happened in the previous steps
|
|
// let's start with the SCALED case
|
|
if (!(t_mode->result.weight & R_RES_STRETCH))
|
|
{
|
|
// apply integer scaling to yres
|
|
if (t_mode->type & Y_RES_EDITABLE) t_mode->vactive *= y_scale;
|
|
|
|
// if we can, let's apply the same scaling to both directions
|
|
if (t_mode->type & X_RES_EDITABLE)
|
|
{
|
|
x_scale = y_scale;
|
|
double aspect_corrector = max(1.0f, cs->monitor_aspect / (cs->rotation? (1.0/(STANDARD_CRT_ASPECT)) : (STANDARD_CRT_ASPECT)));
|
|
t_mode->hactive = normalize(double(t_mode->hactive) * double(x_scale) * aspect_corrector, 8);
|
|
}
|
|
|
|
// otherwise, try to get the best out of our current xres
|
|
else
|
|
{
|
|
x_scale = t_mode->hactive / s_mode->hactive;
|
|
// if the source width fits our xres, try applying integer scaling
|
|
if (x_scale)
|
|
{
|
|
x_scale = scale_into_aspect(s_mode->hactive, t_mode->hactive, cs->rotation?1.0/(STANDARD_CRT_ASPECT):STANDARD_CRT_ASPECT, cs->monitor_aspect, &x_diff);
|
|
if (x_diff > 15.0 && t_mode->hactive < cs->super_width)
|
|
t_mode->result.weight |= R_RES_STRETCH;
|
|
}
|
|
// otherwise apply fractional scaling
|
|
else
|
|
t_mode->result.weight |= R_RES_STRETCH;
|
|
}
|
|
}
|
|
|
|
// if the result was fractional scaling in any of the previous steps, deal with it
|
|
if (t_mode->result.weight & R_RES_STRETCH)
|
|
{
|
|
if (t_mode->type & Y_RES_EDITABLE)
|
|
{
|
|
// always try to use the interlaced range first if it exists, for better resolution
|
|
t_mode->vactive = stretch_into_range(t_mode->vfreq * v_scale, range, borders, cs->interlace, &interlace);
|
|
|
|
// check in case we couldn't achieve the desired refresh
|
|
vfreq_real = min(t_mode->vfreq * v_scale, max_vfreq_for_yres(t_mode->vactive, range, borders, interlace));
|
|
}
|
|
|
|
// check if we can create a normal aspect resolution
|
|
if (t_mode->type & X_RES_EDITABLE)
|
|
t_mode->hactive = max(t_mode->hactive, normalize(STANDARD_CRT_ASPECT * t_mode->vactive, 8));
|
|
|
|
// calculate integer scale for prescaling
|
|
x_scale = max(1, scale_into_aspect(s_mode->hactive, t_mode->hactive, cs->rotation?1.0/(STANDARD_CRT_ASPECT):STANDARD_CRT_ASPECT, cs->monitor_aspect, &x_diff));
|
|
y_scale = max(1, floor(double(t_mode->vactive) / s_mode->vactive));
|
|
|
|
scan_factor = interlace;
|
|
doublescan = 1;
|
|
}
|
|
|
|
x_ratio = double(t_mode->hactive) / s_mode->hactive;
|
|
y_ratio = double(t_mode->vactive) / s_mode->vactive;
|
|
v_scale = max(round_near(vfreq_real / s_mode->vfreq), 1);
|
|
v_diff = (vfreq_real / v_scale) - s_mode->vfreq;
|
|
if (fabs(v_diff) > cs->refresh_tolerance)
|
|
t_mode->result.weight |= R_V_FREQ_OFF;
|
|
|
|
// ≈≈≈ Modeline generation ≈≈≈
|
|
// compute new modeline if we are allowed to
|
|
if (t_mode->type & V_FREQ_EDITABLE)
|
|
{
|
|
double margin = 0;
|
|
double vblank_lines = 0;
|
|
double vvt_ini = 0;
|
|
|
|
// Get resulting refresh
|
|
t_mode->vfreq = vfreq_real;
|
|
|
|
// Get total vertical lines
|
|
vvt_ini = total_lines_for_yres(t_mode->vactive, t_mode->vfreq, range, borders, scan_factor) + (interlace == 2?0.5:0);
|
|
|
|
// Calculate horizontal frequency
|
|
t_mode->hfreq = t_mode->vfreq * vvt_ini;
|
|
|
|
horizontal_values:
|
|
|
|
// Fill horizontal part of modeline
|
|
get_line_params(t_mode, range, cs->pixel_precision? 1 : 8);
|
|
|
|
// Calculate pixel clock
|
|
t_mode->pclock = t_mode->htotal * t_mode->hfreq;
|
|
if (t_mode->pclock <= cs->pclock_min)
|
|
{
|
|
if (t_mode->type & X_RES_EDITABLE)
|
|
{
|
|
x_scale *= 2;
|
|
t_mode->hactive *= 2;
|
|
goto horizontal_values;
|
|
}
|
|
else
|
|
{
|
|
t_mode->result.weight |= R_OUT_OF_RANGE;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// Vertical blanking
|
|
t_mode->vtotal = vvt_ini * scan_factor;
|
|
vblank_lines = int(t_mode->hfreq * (range->vertical_blank + borders)) + (interlace == 2?0.5:0);
|
|
margin = (t_mode->vtotal - t_mode->vactive - vblank_lines * scan_factor) / (cs->v_shift_correct? 1 : 2);
|
|
|
|
t_mode->vbegin = t_mode->vactive + max(round_near(t_mode->hfreq * range->vfront_porch * scan_factor + margin), 1);
|
|
t_mode->vend = t_mode->vbegin + max(round_near(t_mode->hfreq * range->vsync_pulse * scan_factor), 1);
|
|
|
|
// Recalculate final vfreq
|
|
t_mode->vfreq = (t_mode->hfreq / t_mode->vtotal) * scan_factor;
|
|
|
|
t_mode->hsync = range->hsync_polarity;
|
|
t_mode->vsync = range->vsync_polarity;
|
|
t_mode->interlace = interlace == 2?1:0;
|
|
t_mode->doublescan = doublescan == 1?0:1;
|
|
|
|
// Apply interlace fixes
|
|
if (cs->interlace_force_even && interlace == 2)
|
|
{
|
|
t_mode->vbegin = (t_mode->vbegin / 2) * 2;
|
|
t_mode->vend = (t_mode->vend / 2) * 2;
|
|
t_mode->vtotal++;
|
|
}
|
|
}
|
|
|
|
// finally, store result
|
|
t_mode->result.scan_penalty = (s_mode->interlace != t_mode->interlace? 1:0) + (s_mode->doublescan != t_mode->doublescan? 1:0);
|
|
t_mode->result.x_scale = x_scale;
|
|
t_mode->result.y_scale = y_scale;
|
|
t_mode->result.v_scale = v_scale;
|
|
t_mode->result.x_diff = x_diff;
|
|
t_mode->result.y_diff = y_diff;
|
|
t_mode->result.v_diff = v_diff;
|
|
t_mode->result.x_ratio = x_ratio;
|
|
t_mode->result.y_ratio = y_ratio;
|
|
t_mode->result.v_ratio = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
//============================================================
|
|
// get_line_params
|
|
//============================================================
|
|
|
|
int get_line_params(modeline *mode, monitor_range *range, int char_size)
|
|
{
|
|
int hhi, hhf, hht;
|
|
int hh, hs, he, ht;
|
|
double line_time, char_time, new_char_time;
|
|
double hfront_porch_min, hsync_pulse_min, hback_porch_min;
|
|
|
|
hfront_porch_min = range->hfront_porch * .90;
|
|
hsync_pulse_min = range->hsync_pulse * .90;
|
|
hback_porch_min = range->hback_porch * .90;
|
|
|
|
line_time = 1 / mode->hfreq * 1000000;
|
|
|
|
hh = round(mode->hactive / char_size);
|
|
hs = he = ht = 1;
|
|
|
|
do {
|
|
char_time = line_time / (hh + hs + he + ht);
|
|
if (hs * char_time < hfront_porch_min ||
|
|
fabs((hs + 1) * char_time - range->hfront_porch) < fabs(hs * char_time - range->hfront_porch))
|
|
hs++;
|
|
|
|
if (he * char_time < hsync_pulse_min ||
|
|
fabs((he + 1) * char_time - range->hsync_pulse) < fabs(he * char_time - range->hsync_pulse))
|
|
he++;
|
|
|
|
if (ht * char_time < hback_porch_min ||
|
|
fabs((ht + 1) * char_time - range->hback_porch) < fabs(ht * char_time - range->hback_porch))
|
|
ht++;
|
|
|
|
new_char_time = line_time / (hh + hs + he + ht);
|
|
} while (new_char_time != char_time);
|
|
|
|
hhi = (hh + hs) * char_size;
|
|
hhf = (hh + hs + he) * char_size;
|
|
hht = (hh + hs + he + ht) * char_size;
|
|
|
|
mode->hbegin = hhi;
|
|
mode->hend = hhf;
|
|
mode->htotal = hht;
|
|
|
|
return 0;
|
|
}
|
|
|
|
//============================================================
|
|
// scale_into_range
|
|
//============================================================
|
|
|
|
int scale_into_range (int value, int lower_limit, int higher_limit)
|
|
{
|
|
int scale = 1;
|
|
while (value * scale < lower_limit) scale ++;
|
|
if (value * scale <= higher_limit)
|
|
return scale;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
//============================================================
|
|
// scale_into_range
|
|
//============================================================
|
|
|
|
int scale_into_range (double value, double lower_limit, double higher_limit)
|
|
{
|
|
int scale = 1;
|
|
while (value * scale < lower_limit) scale ++;
|
|
if (value * scale <= higher_limit)
|
|
return scale;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
//============================================================
|
|
// scale_into_aspect
|
|
//============================================================
|
|
|
|
int scale_into_aspect (int source_res, int tot_res, double original_monitor_aspect, double users_monitor_aspect, double *best_diff)
|
|
{
|
|
int scale = 1, best_scale = 1;
|
|
double diff = 0;
|
|
*best_diff = 0;
|
|
|
|
while (source_res * scale <= tot_res)
|
|
{
|
|
diff = fabs(1.0 - (users_monitor_aspect / (double(tot_res) / double(source_res * scale) * original_monitor_aspect))) * 100.0;
|
|
if (diff < *best_diff || *best_diff == 0)
|
|
{
|
|
*best_diff = diff;
|
|
best_scale = scale;
|
|
}
|
|
scale ++;
|
|
}
|
|
return best_scale;
|
|
}
|
|
|
|
//============================================================
|
|
// stretch_into_range
|
|
//============================================================
|
|
|
|
int stretch_into_range(double vfreq, monitor_range *range, double borders, bool interlace_allowed, double *interlace)
|
|
{
|
|
int yres, lower_limit;
|
|
|
|
if (range->interlaced_lines_min && interlace_allowed)
|
|
{
|
|
yres = range->interlaced_lines_max;
|
|
lower_limit = range->interlaced_lines_min;
|
|
*interlace = 2;
|
|
}
|
|
else
|
|
{
|
|
yres = range->progressive_lines_max;
|
|
lower_limit = range->progressive_lines_min;
|
|
}
|
|
|
|
while (yres > lower_limit && max_vfreq_for_yres(yres, range, borders, *interlace) < vfreq)
|
|
yres -= 8;
|
|
|
|
return yres;
|
|
}
|
|
|
|
|
|
//============================================================
|
|
// total_lines_for_yres
|
|
//============================================================
|
|
|
|
int total_lines_for_yres(int yres, double vfreq, monitor_range *range, double borders, double interlace)
|
|
{
|
|
int vvt = max(yres / interlace + round_near(vfreq * yres / (interlace * (1.0 - vfreq * (range->vertical_blank + borders))) * (range->vertical_blank + borders)), 1);
|
|
while ((vfreq * vvt < range->hfreq_min) && (vfreq * (vvt + 1) < range->hfreq_max)) vvt++;
|
|
return vvt;
|
|
}
|
|
|
|
//============================================================
|
|
// max_vfreq_for_yres
|
|
//============================================================
|
|
|
|
double max_vfreq_for_yres (int yres, monitor_range *range, double borders, double interlace)
|
|
{
|
|
return range->hfreq_max / (yres / interlace + round_near(range->hfreq_max * (range->vertical_blank + borders)));
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_print
|
|
//============================================================
|
|
|
|
char * modeline_print(modeline *mode, char *modeline, int flags)
|
|
{
|
|
char label[48]={'\x00'};
|
|
char params[192]={'\x00'};
|
|
|
|
if (flags & MS_LABEL)
|
|
sprintf(label, "\"%dx%d_%d%s %.6fKHz %.6fHz\"", mode->hactive, mode->vactive, mode->refresh, mode->interlace?"i":"", mode->hfreq/1000, mode->vfreq);
|
|
|
|
if (flags & MS_LABEL_SDL)
|
|
sprintf(label, "\"%dx%d_%.6f\"", mode->hactive, mode->vactive, mode->vfreq);
|
|
|
|
if (flags & MS_PARAMS)
|
|
sprintf(params, " %.6f %d %d %d %d %d %d %d %d %s %s %s %s", double(mode->pclock)/1000000.0, mode->hactive, mode->hbegin, mode->hend, mode->htotal, mode->vactive, mode->vbegin, mode->vend, mode->vtotal,
|
|
mode->interlace?"interlace":"", mode->doublescan?"doublescan":"", mode->hsync?"+hsync":"-hsync", mode->vsync?"+vsync":"-vsync");
|
|
|
|
sprintf(modeline, "%s%s", label, params);
|
|
|
|
return modeline;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_result
|
|
//============================================================
|
|
|
|
char * modeline_result(modeline *mode, char *result)
|
|
{
|
|
log_verbose(" rng(%d): ", mode->range);
|
|
|
|
if (mode->result.weight & R_OUT_OF_RANGE)
|
|
sprintf(result, " out of range");
|
|
|
|
else
|
|
sprintf(result, "%4d x%4d_%3.6f%s%s %3.6f [%s] scale(%d, %d, %d) diff(%.2f, %.2f, %.4f) ratio(%.3f, %.3f)",
|
|
mode->hactive, mode->vactive, mode->vfreq, mode->interlace?"i":"p", mode->doublescan?"d":"", mode->hfreq/1000, mode->result.weight & R_RES_STRETCH?"fract":"integ",
|
|
mode->result.x_scale, mode->result.y_scale, mode->result.v_scale, mode->result.x_diff, mode->result.y_diff, mode->result.v_diff, mode->result.x_ratio, mode->result.y_ratio);
|
|
return result;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_compare
|
|
//============================================================
|
|
|
|
int modeline_compare(modeline *t, modeline *best)
|
|
{
|
|
bool vector = (t->hactive == (int)t->result.x_ratio);
|
|
|
|
if (t->result.weight < best->result.weight)
|
|
return 1;
|
|
|
|
else if (t->result.weight <= best->result.weight)
|
|
{
|
|
double t_v_diff = fabs(t->result.v_diff);
|
|
double b_v_diff = fabs(best->result.v_diff);
|
|
|
|
if (t->result.weight & R_RES_STRETCH || vector)
|
|
{
|
|
double t_y_score = t->result.y_ratio * (t->interlace?(2.0/3.0):1.0);
|
|
double b_y_score = best->result.y_ratio * (best->interlace?(2.0/3.0):1.0);
|
|
|
|
if ((t_v_diff < b_v_diff) ||
|
|
((t_v_diff == b_v_diff) && (t_y_score > b_y_score)) ||
|
|
((t_v_diff == b_v_diff) && (t_y_score == b_y_score) && (t->result.x_ratio > best->result.x_ratio)))
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
int t_y_score = t->result.y_scale + t->result.scan_penalty;
|
|
int b_y_score = best->result.y_scale + best->result.scan_penalty;
|
|
double xy_diff = roundf((t->result.x_diff + t->result.y_diff) * 100) / 100;
|
|
double best_xy_diff = roundf((best->result.x_diff + best->result.y_diff) * 100) / 100;
|
|
|
|
if ((t_y_score < b_y_score) ||
|
|
((t_y_score == b_y_score) && (xy_diff < best_xy_diff)) ||
|
|
((t_y_score == b_y_score) && (xy_diff == best_xy_diff) && (t->result.x_scale < best->result.x_scale)) ||
|
|
((t_y_score == b_y_score) && (xy_diff == best_xy_diff) && (t->result.x_scale == best->result.x_scale) && (t_v_diff < b_v_diff)))
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_vesa_gtf
|
|
// Based on the VESA GTF spreadsheet by Andy Morrish 1/5/97
|
|
//============================================================
|
|
|
|
int modeline_vesa_gtf(modeline *m)
|
|
{
|
|
int C, M;
|
|
int v_sync_lines, v_porch_lines_min, v_front_porch_lines, v_back_porch_lines, v_sync_v_back_porch_lines, v_total_lines;
|
|
int h_sync_width_percent, h_sync_width_pixels, h_blanking_pixels, h_front_porch_pixels, h_total_pixels;
|
|
double v_freq, v_freq_est, v_freq_real, v_sync_v_back_porch;
|
|
double h_freq, h_period, h_period_real, h_ideal_blanking;
|
|
double pixel_freq, interlace;
|
|
|
|
// Check if there's a value defined for vfreq. We're assuming input vfreq is the total field vfreq regardless interlace
|
|
v_freq = m->vfreq? m->vfreq:double(m->refresh);
|
|
|
|
// These values are GTF defined defaults
|
|
v_sync_lines = 3;
|
|
v_porch_lines_min = 1;
|
|
v_front_porch_lines = v_porch_lines_min;
|
|
v_sync_v_back_porch = 550;
|
|
h_sync_width_percent = 8;
|
|
M = 128.0 / 256 * 600;
|
|
C = ((40 - 20) * 128.0 / 256) + 20;
|
|
|
|
// GTF calculation
|
|
interlace = m->interlace?0.5:0;
|
|
h_period = ((1.0 / v_freq) - (v_sync_v_back_porch / 1000000)) / ((double)m->height + v_front_porch_lines + interlace) * 1000000;
|
|
v_sync_v_back_porch_lines = round_near(v_sync_v_back_porch / h_period);
|
|
v_back_porch_lines = v_sync_v_back_porch_lines - v_sync_lines;
|
|
v_total_lines = m->height + v_front_porch_lines + v_sync_lines + v_back_porch_lines;
|
|
v_freq_est = (1.0 / h_period) / v_total_lines * 1000000;
|
|
h_period_real = h_period / (v_freq / v_freq_est);
|
|
v_freq_real = (1.0 / h_period_real) / v_total_lines * 1000000;
|
|
h_ideal_blanking = double(C - (M * h_period_real / 1000));
|
|
h_blanking_pixels = round_near(m->width * h_ideal_blanking /(100 - h_ideal_blanking) / (2 * 8)) * (2 * 8);
|
|
h_total_pixels = m->width + h_blanking_pixels;
|
|
pixel_freq = h_total_pixels / h_period_real * 1000000;
|
|
h_freq = 1000000 / h_period_real;
|
|
h_sync_width_pixels = round_near(h_sync_width_percent * h_total_pixels / 100 / 8) * 8;
|
|
h_front_porch_pixels = (h_blanking_pixels / 2) - h_sync_width_pixels;
|
|
|
|
// Results
|
|
m->hactive = m->width;
|
|
m->hbegin = m->hactive + h_front_porch_pixels;
|
|
m->hend = m->hbegin + h_sync_width_pixels;
|
|
m->htotal = h_total_pixels;
|
|
m->vactive = m->height;
|
|
m->vbegin = m->vactive + v_front_porch_lines;
|
|
m->vend = m->vbegin + v_sync_lines;
|
|
m->vtotal = v_total_lines;
|
|
m->hfreq = h_freq;
|
|
m->vfreq = v_freq_real;
|
|
m->pclock = pixel_freq;
|
|
m->hsync = 0;
|
|
m->vsync = 1;
|
|
|
|
return true;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_parse
|
|
//============================================================
|
|
|
|
int modeline_parse(const char *user_modeline, modeline *mode)
|
|
{
|
|
char modeline_txt[256]={'\x00'};
|
|
|
|
if (!strcmp(user_modeline, "auto"))
|
|
return false;
|
|
|
|
// Remove quotes
|
|
char *quote_start, *quote_end;
|
|
quote_start = strstr((char*)user_modeline, "\"");
|
|
if (quote_start)
|
|
{
|
|
quote_start++;
|
|
quote_end = strstr(quote_start, "\"");
|
|
if (!quote_end || *quote_end++ == 0)
|
|
return false;
|
|
user_modeline = quote_end;
|
|
}
|
|
|
|
// Get timing flags
|
|
mode->interlace = strstr(user_modeline, "interlace")?1:0;
|
|
mode->doublescan = strstr(user_modeline, "doublescan")?1:0;
|
|
mode->hsync = strstr(user_modeline, "+hsync")?1:0;
|
|
mode->vsync = strstr(user_modeline, "+vsync")?1:0;
|
|
|
|
// Get timing values
|
|
double pclock;
|
|
int e = sscanf(user_modeline, " %lf %d %d %d %d %d %d %d %d",
|
|
&pclock,
|
|
&mode->hactive, &mode->hbegin, &mode->hend, &mode->htotal,
|
|
&mode->vactive, &mode->vbegin, &mode->vend, &mode->vtotal);
|
|
|
|
if (e != 9)
|
|
{
|
|
log_error("SwitchRes: missing parameter in user modeline\n %s\n", user_modeline);
|
|
memset(mode, 0, sizeof(struct modeline));
|
|
return false;
|
|
}
|
|
|
|
// Calculate timings
|
|
mode->pclock = pclock * 1000000.0;
|
|
mode->hfreq = mode->pclock / mode->htotal;
|
|
mode->vfreq = mode->hfreq / mode->vtotal * (mode->interlace?2:1);
|
|
mode->refresh = mode->vfreq;
|
|
mode->width = mode->hactive;
|
|
mode->height = mode->vactive;
|
|
log_verbose("SwitchRes: user modeline %s\n", modeline_print(mode, modeline_txt, MS_FULL));
|
|
|
|
return true;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_to_monitor_range
|
|
//============================================================
|
|
|
|
int modeline_to_monitor_range(monitor_range *range, modeline *mode)
|
|
{
|
|
if (range->vfreq_min == 0)
|
|
{
|
|
range->vfreq_min = mode->vfreq - 0.2;
|
|
range->vfreq_max = mode->vfreq + 0.2;
|
|
}
|
|
|
|
double line_time = 1 / mode->hfreq;
|
|
double pixel_time = line_time / mode->htotal * 1000000;
|
|
|
|
range->hfront_porch = pixel_time * (mode->hbegin - mode->hactive);
|
|
range->hsync_pulse = pixel_time * (mode->hend - mode->hbegin);
|
|
range->hback_porch = pixel_time * (mode->htotal - mode->hend);
|
|
|
|
range->vfront_porch = line_time * (mode->vbegin - mode->vactive);
|
|
range->vsync_pulse = line_time * (mode->vend - mode->vbegin);
|
|
range->vback_porch = line_time * (mode->vtotal - mode->vend);
|
|
range->vertical_blank = range->vfront_porch + range->vsync_pulse + range->vback_porch;
|
|
|
|
range->hsync_polarity = mode->hsync;
|
|
range->vsync_polarity = mode->vsync;
|
|
|
|
range->progressive_lines_min = mode->interlace?0:mode->vactive;
|
|
range->progressive_lines_max = mode->interlace?0:mode->vactive;
|
|
range->interlaced_lines_min = mode->interlace?mode->vactive:0;
|
|
range->interlaced_lines_max= mode->interlace?mode->vactive:0;
|
|
|
|
range->hfreq_min = range->vfreq_min * mode->vtotal;
|
|
range->hfreq_max = range->vfreq_max * mode->vtotal;
|
|
|
|
return 1;
|
|
}
|
|
|
|
//============================================================
|
|
// modeline_is_different
|
|
//============================================================
|
|
|
|
int modeline_is_different(modeline *n, modeline *p)
|
|
{
|
|
// Remove on last fields in modeline comparison
|
|
return memcmp(n, p, offsetof(struct modeline, vfreq));
|
|
}
|
|
|
|
//============================================================
|
|
// monitor_fill_vesa_gtf
|
|
//============================================================
|
|
|
|
int monitor_fill_vesa_gtf(monitor_range *range, const char *max_lines)
|
|
{
|
|
int lines = 0;
|
|
sscanf(max_lines, "vesa_%d", &lines);
|
|
|
|
if (!lines)
|
|
return 0;
|
|
|
|
int i = 0;
|
|
if (lines >= 480)
|
|
i += monitor_fill_vesa_range(&range[i], 384, 480);
|
|
if (lines >= 600)
|
|
i += monitor_fill_vesa_range(&range[i], 480, 600);
|
|
if (lines >= 768)
|
|
i += monitor_fill_vesa_range(&range[i], 600, 768);
|
|
if (lines >= 1024)
|
|
i += monitor_fill_vesa_range(&range[i], 768, 1024);
|
|
|
|
return i;
|
|
}
|
|
|
|
//============================================================
|
|
// monitor_fill_vesa_range
|
|
//============================================================
|
|
|
|
int monitor_fill_vesa_range(monitor_range *range, int lines_min, int lines_max)
|
|
{
|
|
modeline mode;
|
|
memset(&mode, 0, sizeof(modeline));
|
|
|
|
mode.width = real_res(STANDARD_CRT_ASPECT * lines_max);
|
|
mode.height = lines_max;
|
|
mode.refresh = 60;
|
|
range->vfreq_min = 50;
|
|
range->vfreq_max = 65;
|
|
|
|
modeline_vesa_gtf(&mode);
|
|
modeline_to_monitor_range(range, &mode);
|
|
|
|
range->progressive_lines_min = lines_min;
|
|
range->hfreq_min = mode.hfreq - 500;
|
|
range->hfreq_max = mode.hfreq + 500;
|
|
monitor_show_range(range);
|
|
|
|
return 1;
|
|
}
|
|
|
|
//============================================================
|
|
// round_near
|
|
//============================================================
|
|
|
|
int round_near(double number)
|
|
{
|
|
return number < 0.0 ? ceil(number - 0.5) : floor(number + 0.5);
|
|
}
|
|
|
|
//============================================================
|
|
// normalize
|
|
//============================================================
|
|
|
|
int normalize(int a, int b)
|
|
{
|
|
int c, d;
|
|
c = a % b;
|
|
d = a / b;
|
|
if (c) d++;
|
|
return d * b;
|
|
}
|
|
|
|
//============================================================
|
|
// real_res
|
|
//============================================================
|
|
|
|
int real_res(int x) {return (int) (x / 8) * 8;}
|