mirror of
https://github.com/torvalds/linux.git
synced 2026-04-27 02:52:27 -04:00
When a connector is created, add a `status` file to allow to update the connector status to: - 1 connector_status_connected - 2 connector_status_disconnected - 3 connector_status_unknown If the device is enabled, updating the status hot-plug or unplugs the connector. Tested-by: Mark Yacoub <markyacoub@google.com> Reviewed-by: Louis Chauvet <louis.chauvet@bootlin.com> Reviewed-by: Harry Wentland <harry.wentland@amd.com> Reviewed-by: Luca Ceresoli <luca.ceresoli@bootlin.com> Signed-off-by: José Expósito <jose.exposito89@gmail.com> Link: https://lore.kernel.org/r/20251016175618.10051-17-jose.exposito89@gmail.com Signed-off-by: Luca Ceresoli <luca.ceresoli@bootlin.com>
97 lines
2.6 KiB
C
97 lines
2.6 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_edid.h>
|
|
#include <drm/drm_managed.h>
|
|
#include <drm/drm_probe_helper.h>
|
|
|
|
#include "vkms_config.h"
|
|
#include "vkms_connector.h"
|
|
|
|
static enum drm_connector_status vkms_connector_detect(struct drm_connector *connector,
|
|
bool force)
|
|
{
|
|
struct drm_device *dev = connector->dev;
|
|
struct vkms_device *vkmsdev = drm_device_to_vkms_device(dev);
|
|
struct vkms_connector *vkms_connector;
|
|
enum drm_connector_status status;
|
|
struct vkms_config_connector *connector_cfg;
|
|
|
|
vkms_connector = drm_connector_to_vkms_connector(connector);
|
|
|
|
/*
|
|
* The connector configuration might not exist if its configfs directory
|
|
* was deleted. Therefore, use the configuration if present or keep the
|
|
* current status if we can not access it anymore.
|
|
*/
|
|
status = connector->status;
|
|
|
|
vkms_config_for_each_connector(vkmsdev->config, connector_cfg) {
|
|
if (connector_cfg->connector == vkms_connector)
|
|
status = vkms_config_connector_get_status(connector_cfg);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
static const struct drm_connector_funcs vkms_connector_funcs = {
|
|
.detect = vkms_connector_detect,
|
|
.fill_modes = drm_helper_probe_single_connector_modes,
|
|
.reset = drm_atomic_helper_connector_reset,
|
|
.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
|
|
};
|
|
|
|
static int vkms_conn_get_modes(struct drm_connector *connector)
|
|
{
|
|
int count;
|
|
|
|
/* Use the default modes list from DRM */
|
|
count = drm_add_modes_noedid(connector, XRES_MAX, YRES_MAX);
|
|
drm_set_preferred_mode(connector, XRES_DEF, YRES_DEF);
|
|
|
|
return count;
|
|
}
|
|
|
|
static struct drm_encoder *vkms_conn_best_encoder(struct drm_connector *connector)
|
|
{
|
|
struct drm_encoder *encoder;
|
|
|
|
drm_connector_for_each_possible_encoder(connector, encoder)
|
|
return encoder;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static const struct drm_connector_helper_funcs vkms_conn_helper_funcs = {
|
|
.get_modes = vkms_conn_get_modes,
|
|
.best_encoder = vkms_conn_best_encoder,
|
|
};
|
|
|
|
struct vkms_connector *vkms_connector_init(struct vkms_device *vkmsdev)
|
|
{
|
|
struct drm_device *dev = &vkmsdev->drm;
|
|
struct vkms_connector *connector;
|
|
int ret;
|
|
|
|
connector = drmm_kzalloc(dev, sizeof(*connector), GFP_KERNEL);
|
|
if (!connector)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
ret = drmm_connector_init(dev, &connector->base, &vkms_connector_funcs,
|
|
DRM_MODE_CONNECTOR_VIRTUAL, NULL);
|
|
if (ret)
|
|
return ERR_PTR(ret);
|
|
|
|
drm_connector_helper_add(&connector->base, &vkms_conn_helper_funcs);
|
|
|
|
return connector;
|
|
}
|
|
|
|
void vkms_trigger_connector_hotplug(struct vkms_device *vkmsdev)
|
|
{
|
|
struct drm_device *dev = &vkmsdev->drm;
|
|
|
|
drm_kms_helper_hotplug_event(dev);
|
|
}
|