mirror of
https://github.com/libretro/RetroArch.git
synced 2024-11-28 02:30:35 +00:00
fc69e73dcf
mode->vrefresh is a display-only value and should not be used for calculations the drm context is now able to differentiate between, for example, 640x480@60.00 and 640x480@59.92 when setting the video mode
204 lines
5.3 KiB
C
204 lines
5.3 KiB
C
/* RetroArch - A frontend for libretro.
|
|
* Copyright (c) 2011-2017 - Daniel De Matteis
|
|
*
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <compat/strl.h>
|
|
|
|
#include <retro_miscellaneous.h>
|
|
|
|
#include "../../verbosity.h"
|
|
|
|
#include "drm_common.h"
|
|
|
|
/* TODO/FIXME - globals */
|
|
drmEventContext g_drm_evctx;
|
|
struct pollfd g_drm_fds;
|
|
uint32_t g_connector_id = 0;
|
|
int g_drm_fd = 0;
|
|
uint32_t g_crtc_id = 0;
|
|
drmModeCrtc *g_orig_crtc = NULL;
|
|
drmModeConnector *g_drm_connector = NULL;
|
|
drmModeModeInfo *g_drm_mode = NULL;
|
|
|
|
/* TODO/FIXME - static globals */
|
|
static drmModeRes *g_drm_resources = NULL;
|
|
static drmModeEncoder *g_drm_encoder = NULL;
|
|
|
|
/* Restore the original CRTC. */
|
|
void drm_restore_crtc(void)
|
|
{
|
|
if (!g_orig_crtc)
|
|
return;
|
|
|
|
drmModeSetCrtc(g_drm_fd, g_orig_crtc->crtc_id,
|
|
g_orig_crtc->buffer_id,
|
|
g_orig_crtc->x,
|
|
g_orig_crtc->y,
|
|
&g_connector_id, 1, &g_orig_crtc->mode);
|
|
|
|
drmModeFreeCrtc(g_orig_crtc);
|
|
g_orig_crtc = NULL;
|
|
}
|
|
|
|
bool drm_get_resources(int fd)
|
|
{
|
|
g_drm_resources = drmModeGetResources(fd);
|
|
if (!g_drm_resources)
|
|
{
|
|
RARCH_WARN("[DRM]: Couldn't get device resources.\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool drm_get_connector(int fd, unsigned monitor_index)
|
|
{
|
|
unsigned i;
|
|
unsigned monitor_index_count = 0;
|
|
unsigned monitor = MAX(monitor_index, 1);
|
|
|
|
/* Enumerate all connectors. */
|
|
|
|
RARCH_LOG("[DRM]: Found %d connectors.\n", g_drm_resources->count_connectors);
|
|
|
|
for (i = 0; (int)i < g_drm_resources->count_connectors; i++)
|
|
{
|
|
drmModeConnectorPtr conn = drmModeGetConnector(
|
|
fd, g_drm_resources->connectors[i]);
|
|
|
|
if (conn)
|
|
{
|
|
bool connected = conn->connection == DRM_MODE_CONNECTED;
|
|
RARCH_LOG("[DRM]: Connector %d connected: %s\n", i, connected ? "yes" : "no");
|
|
RARCH_LOG("[DRM]: Connector %d has %d modes.\n", i, conn->count_modes);
|
|
if (connected && conn->count_modes > 0)
|
|
{
|
|
monitor_index_count++;
|
|
RARCH_LOG("[DRM]: Connector %d assigned to monitor index: #%u.\n", i, monitor_index_count);
|
|
}
|
|
drmModeFreeConnector(conn);
|
|
}
|
|
}
|
|
|
|
monitor_index_count = 0;
|
|
|
|
for (i = 0; (int)i < g_drm_resources->count_connectors; i++)
|
|
{
|
|
g_drm_connector = drmModeGetConnector(fd,
|
|
g_drm_resources->connectors[i]);
|
|
|
|
if (!g_drm_connector)
|
|
continue;
|
|
if (g_drm_connector->connection == DRM_MODE_CONNECTED
|
|
&& g_drm_connector->count_modes > 0)
|
|
{
|
|
monitor_index_count++;
|
|
if (monitor_index_count == monitor)
|
|
break;
|
|
}
|
|
|
|
drmModeFreeConnector(g_drm_connector);
|
|
g_drm_connector = NULL;
|
|
}
|
|
|
|
if (!g_drm_connector)
|
|
{
|
|
RARCH_WARN("[DRM]: Couldn't get device connector.\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
float drm_calc_refresh_rate(drmModeModeInfo *mode)
|
|
{
|
|
float refresh_rate = (mode->clock * 1000.0f) / (mode->htotal * mode->vtotal);
|
|
return refresh_rate;
|
|
}
|
|
|
|
bool drm_get_encoder(int fd)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; (int)i < g_drm_resources->count_encoders; i++)
|
|
{
|
|
g_drm_encoder = drmModeGetEncoder(fd, g_drm_resources->encoders[i]);
|
|
|
|
if (!g_drm_encoder)
|
|
continue;
|
|
|
|
if (g_drm_encoder->encoder_id == g_drm_connector->encoder_id)
|
|
break;
|
|
|
|
drmModeFreeEncoder(g_drm_encoder);
|
|
g_drm_encoder = NULL;
|
|
}
|
|
|
|
if (!g_drm_encoder)
|
|
{
|
|
RARCH_WARN("[DRM]: Couldn't find DRM encoder.\n");
|
|
return false;
|
|
}
|
|
|
|
for (i = 0; (int)i < g_drm_connector->count_modes; i++)
|
|
{
|
|
RARCH_LOG("[DRM]: Mode %d: (%s) %d x %d, %f Hz\n",
|
|
i,
|
|
g_drm_connector->modes[i].name,
|
|
g_drm_connector->modes[i].hdisplay,
|
|
g_drm_connector->modes[i].vdisplay,
|
|
drm_calc_refresh_rate(&g_drm_connector->modes[i]));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void drm_setup(int fd)
|
|
{
|
|
g_crtc_id = g_drm_encoder->crtc_id;
|
|
g_connector_id = g_drm_connector->connector_id;
|
|
g_orig_crtc = drmModeGetCrtc(fd, g_crtc_id);
|
|
if (!g_orig_crtc)
|
|
RARCH_WARN("[DRM]: Cannot find original CRTC.\n");
|
|
}
|
|
|
|
float drm_get_refresh_rate(void *data)
|
|
{
|
|
float refresh_rate = 0.0f;
|
|
|
|
if (g_drm_mode)
|
|
{
|
|
refresh_rate = drm_calc_refresh_rate(g_drm_mode);
|
|
}
|
|
|
|
return refresh_rate;
|
|
}
|
|
|
|
void drm_free(void)
|
|
{
|
|
if (g_drm_encoder)
|
|
drmModeFreeEncoder(g_drm_encoder);
|
|
if (g_drm_connector)
|
|
drmModeFreeConnector(g_drm_connector);
|
|
if (g_drm_resources)
|
|
drmModeFreeResources(g_drm_resources);
|
|
|
|
memset(&g_drm_fds, 0, sizeof(struct pollfd));
|
|
memset(&g_drm_evctx, 0, sizeof(drmEventContext));
|
|
|
|
g_drm_encoder = NULL;
|
|
g_drm_connector = NULL;
|
|
g_drm_resources = NULL;
|
|
}
|