xref: /openbmc/linux/drivers/gpu/drm/xen/xen_drm_front_conn.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1c575b7eeSOleksandr Andrushchenko // SPDX-License-Identifier: GPL-2.0 OR MIT
2c575b7eeSOleksandr Andrushchenko 
3c575b7eeSOleksandr Andrushchenko /*
4c575b7eeSOleksandr Andrushchenko  *  Xen para-virtual DRM device
5c575b7eeSOleksandr Andrushchenko  *
6c575b7eeSOleksandr Andrushchenko  * Copyright (C) 2016-2018 EPAM Systems Inc.
7c575b7eeSOleksandr Andrushchenko  *
8c575b7eeSOleksandr Andrushchenko  * Author: Oleksandr Andrushchenko <oleksandr_andrushchenko@epam.com>
9c575b7eeSOleksandr Andrushchenko  */
10c575b7eeSOleksandr Andrushchenko 
11c575b7eeSOleksandr Andrushchenko #include <drm/drm_atomic_helper.h>
122ea2269eSSam Ravnborg #include <drm/drm_drv.h>
13*720cf96dSVille Syrjälä #include <drm/drm_fourcc.h>
14fcd70cd3SDaniel Vetter #include <drm/drm_probe_helper.h>
15c575b7eeSOleksandr Andrushchenko 
16c575b7eeSOleksandr Andrushchenko #include <video/videomode.h>
17c575b7eeSOleksandr Andrushchenko 
18c575b7eeSOleksandr Andrushchenko #include "xen_drm_front.h"
19c575b7eeSOleksandr Andrushchenko #include "xen_drm_front_conn.h"
20c575b7eeSOleksandr Andrushchenko #include "xen_drm_front_kms.h"
21c575b7eeSOleksandr Andrushchenko 
22c575b7eeSOleksandr Andrushchenko static struct xen_drm_front_drm_pipeline *
to_xen_drm_pipeline(struct drm_connector * connector)23c575b7eeSOleksandr Andrushchenko to_xen_drm_pipeline(struct drm_connector *connector)
24c575b7eeSOleksandr Andrushchenko {
25c575b7eeSOleksandr Andrushchenko 	return container_of(connector, struct xen_drm_front_drm_pipeline, conn);
26c575b7eeSOleksandr Andrushchenko }
27c575b7eeSOleksandr Andrushchenko 
28c575b7eeSOleksandr Andrushchenko static const u32 plane_formats[] = {
29c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_RGB565,
30c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_RGB888,
31c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_XRGB8888,
32c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_ARGB8888,
33c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_XRGB4444,
34c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_ARGB4444,
35c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_XRGB1555,
36c575b7eeSOleksandr Andrushchenko 	DRM_FORMAT_ARGB1555,
3712957299SOleksandr Andrushchenko 	DRM_FORMAT_YUYV,
38c575b7eeSOleksandr Andrushchenko };
39c575b7eeSOleksandr Andrushchenko 
xen_drm_front_conn_get_formats(int * format_count)40c575b7eeSOleksandr Andrushchenko const u32 *xen_drm_front_conn_get_formats(int *format_count)
41c575b7eeSOleksandr Andrushchenko {
42c575b7eeSOleksandr Andrushchenko 	*format_count = ARRAY_SIZE(plane_formats);
43c575b7eeSOleksandr Andrushchenko 	return plane_formats;
44c575b7eeSOleksandr Andrushchenko }
45c575b7eeSOleksandr Andrushchenko 
connector_detect(struct drm_connector * connector,struct drm_modeset_acquire_ctx * ctx,bool force)46c575b7eeSOleksandr Andrushchenko static int connector_detect(struct drm_connector *connector,
47c575b7eeSOleksandr Andrushchenko 			    struct drm_modeset_acquire_ctx *ctx,
48c575b7eeSOleksandr Andrushchenko 			    bool force)
49c575b7eeSOleksandr Andrushchenko {
50c575b7eeSOleksandr Andrushchenko 	struct xen_drm_front_drm_pipeline *pipeline =
51c575b7eeSOleksandr Andrushchenko 			to_xen_drm_pipeline(connector);
52c575b7eeSOleksandr Andrushchenko 
53c575b7eeSOleksandr Andrushchenko 	if (drm_dev_is_unplugged(connector->dev))
54c575b7eeSOleksandr Andrushchenko 		pipeline->conn_connected = false;
55c575b7eeSOleksandr Andrushchenko 
56c575b7eeSOleksandr Andrushchenko 	return pipeline->conn_connected ? connector_status_connected :
57c575b7eeSOleksandr Andrushchenko 			connector_status_disconnected;
58c575b7eeSOleksandr Andrushchenko }
59c575b7eeSOleksandr Andrushchenko 
60c575b7eeSOleksandr Andrushchenko #define XEN_DRM_CRTC_VREFRESH_HZ	60
61c575b7eeSOleksandr Andrushchenko 
connector_get_modes(struct drm_connector * connector)62c575b7eeSOleksandr Andrushchenko static int connector_get_modes(struct drm_connector *connector)
63c575b7eeSOleksandr Andrushchenko {
64c575b7eeSOleksandr Andrushchenko 	struct xen_drm_front_drm_pipeline *pipeline =
65c575b7eeSOleksandr Andrushchenko 			to_xen_drm_pipeline(connector);
66c575b7eeSOleksandr Andrushchenko 	struct drm_display_mode *mode;
67c575b7eeSOleksandr Andrushchenko 	struct videomode videomode;
68c575b7eeSOleksandr Andrushchenko 	int width, height;
69c575b7eeSOleksandr Andrushchenko 
70c575b7eeSOleksandr Andrushchenko 	mode = drm_mode_create(connector->dev);
71c575b7eeSOleksandr Andrushchenko 	if (!mode)
72c575b7eeSOleksandr Andrushchenko 		return 0;
73c575b7eeSOleksandr Andrushchenko 
74c575b7eeSOleksandr Andrushchenko 	memset(&videomode, 0, sizeof(videomode));
75c575b7eeSOleksandr Andrushchenko 	videomode.hactive = pipeline->width;
76c575b7eeSOleksandr Andrushchenko 	videomode.vactive = pipeline->height;
77c575b7eeSOleksandr Andrushchenko 	width = videomode.hactive + videomode.hfront_porch +
78c575b7eeSOleksandr Andrushchenko 			videomode.hback_porch + videomode.hsync_len;
79c575b7eeSOleksandr Andrushchenko 	height = videomode.vactive + videomode.vfront_porch +
80c575b7eeSOleksandr Andrushchenko 			videomode.vback_porch + videomode.vsync_len;
81c575b7eeSOleksandr Andrushchenko 	videomode.pixelclock = width * height * XEN_DRM_CRTC_VREFRESH_HZ;
82c575b7eeSOleksandr Andrushchenko 	mode->type = DRM_MODE_TYPE_PREFERRED | DRM_MODE_TYPE_DRIVER;
83c575b7eeSOleksandr Andrushchenko 
84c575b7eeSOleksandr Andrushchenko 	drm_display_mode_from_videomode(&videomode, mode);
85c575b7eeSOleksandr Andrushchenko 	drm_mode_probed_add(connector, mode);
86c575b7eeSOleksandr Andrushchenko 	return 1;
87c575b7eeSOleksandr Andrushchenko }
88c575b7eeSOleksandr Andrushchenko 
89c575b7eeSOleksandr Andrushchenko static const struct drm_connector_helper_funcs connector_helper_funcs = {
90c575b7eeSOleksandr Andrushchenko 	.get_modes = connector_get_modes,
91c575b7eeSOleksandr Andrushchenko 	.detect_ctx = connector_detect,
92c575b7eeSOleksandr Andrushchenko };
93c575b7eeSOleksandr Andrushchenko 
94c575b7eeSOleksandr Andrushchenko static const struct drm_connector_funcs connector_funcs = {
95c575b7eeSOleksandr Andrushchenko 	.fill_modes = drm_helper_probe_single_connector_modes,
96c575b7eeSOleksandr Andrushchenko 	.destroy = drm_connector_cleanup,
97c575b7eeSOleksandr Andrushchenko 	.reset = drm_atomic_helper_connector_reset,
98c575b7eeSOleksandr Andrushchenko 	.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
99c575b7eeSOleksandr Andrushchenko 	.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
100c575b7eeSOleksandr Andrushchenko };
101c575b7eeSOleksandr Andrushchenko 
xen_drm_front_conn_init(struct xen_drm_front_drm_info * drm_info,struct drm_connector * connector)102c575b7eeSOleksandr Andrushchenko int xen_drm_front_conn_init(struct xen_drm_front_drm_info *drm_info,
103c575b7eeSOleksandr Andrushchenko 			    struct drm_connector *connector)
104c575b7eeSOleksandr Andrushchenko {
105c575b7eeSOleksandr Andrushchenko 	struct xen_drm_front_drm_pipeline *pipeline =
106c575b7eeSOleksandr Andrushchenko 			to_xen_drm_pipeline(connector);
107c575b7eeSOleksandr Andrushchenko 
108c575b7eeSOleksandr Andrushchenko 	drm_connector_helper_add(connector, &connector_helper_funcs);
109c575b7eeSOleksandr Andrushchenko 
110c575b7eeSOleksandr Andrushchenko 	pipeline->conn_connected = true;
111c575b7eeSOleksandr Andrushchenko 
112c575b7eeSOleksandr Andrushchenko 	connector->polled = DRM_CONNECTOR_POLL_CONNECT |
113c575b7eeSOleksandr Andrushchenko 			DRM_CONNECTOR_POLL_DISCONNECT;
114c575b7eeSOleksandr Andrushchenko 
115c575b7eeSOleksandr Andrushchenko 	return drm_connector_init(drm_info->drm_dev, connector,
116c575b7eeSOleksandr Andrushchenko 				  &connector_funcs, DRM_MODE_CONNECTOR_VIRTUAL);
117c575b7eeSOleksandr Andrushchenko }
118