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