2 * uterm - Linux User-Space Terminal
4 * Copyright (c) 2011-2013 David Herrmann <dh.herrmann@googlemail.com>
6 * Permission is hereby granted, free of charge, to any person obtaining
7 * a copy of this software and associated documentation files
8 * (the "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sublicense, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
14 * The above copyright notice and this permission notice shall be included
15 * in all copies or substantial portions of the Software.
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
20 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
21 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
22 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
23 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 * DRM shared functions
37 #include <xf86drmMode.h>
39 #include "uterm_drm_shared_internal.h"
40 #include "uterm_video.h"
41 #include "uterm_video_internal.h"
43 #define LOG_SUBSYSTEM "drm_shared"
45 int uterm_drm_mode_init(struct uterm_mode *mode)
47 struct uterm_drm_mode *m;
49 m = malloc(sizeof(*m));
52 memset(m, 0, sizeof(*m));
58 void uterm_drm_mode_destroy(struct uterm_mode *mode)
63 const char *uterm_drm_mode_get_name(const struct uterm_mode *mode)
65 struct uterm_drm_mode *m = mode->data;
70 unsigned int uterm_drm_mode_get_width(const struct uterm_mode *mode)
72 struct uterm_drm_mode *m = mode->data;
74 return m->info.hdisplay;
77 unsigned int uterm_drm_mode_get_height(const struct uterm_mode *mode)
79 struct uterm_drm_mode *m = mode->data;
81 return m->info.vdisplay;
84 void uterm_drm_mode_set(struct uterm_mode *mode, drmModeModeInfo *info)
86 struct uterm_drm_mode *m = mode->data;
91 const struct mode_ops uterm_drm_mode_ops = {
92 .init = uterm_drm_mode_init,
93 .destroy = uterm_drm_mode_destroy,
94 .get_name = uterm_drm_mode_get_name,
95 .get_width = uterm_drm_mode_get_width,
96 .get_height = uterm_drm_mode_get_height,
99 int uterm_drm_set_dpms(int fd, uint32_t conn_id, int state)
102 drmModeConnector *conn;
103 drmModePropertyRes *prop;
107 set = DRM_MODE_DPMS_ON;
109 case UTERM_DPMS_STANDBY:
110 set = DRM_MODE_DPMS_STANDBY;
112 case UTERM_DPMS_SUSPEND:
113 set = DRM_MODE_DPMS_SUSPEND;
116 set = DRM_MODE_DPMS_OFF;
122 conn = drmModeGetConnector(fd, conn_id);
124 log_err("cannot get display connector");
129 for (i = 0; i < conn->count_props; ++i) {
130 prop = drmModeGetProperty(fd, conn->props[i]);
132 log_error("cannot get DRM property (%d): %m", errno);
136 if (!strcmp(prop->name, "DPMS")) {
137 ret = drmModeConnectorSetProperty(fd, conn_id,
140 log_info("cannot set DPMS");
143 drmModeFreeProperty(prop);
146 drmModeFreeProperty(prop);
149 if (i == conn->count_props) {
150 log_warn("display does not support DPMS");
151 ret = UTERM_DPMS_UNKNOWN;
154 drmModeFreeConnector(conn);
158 int uterm_drm_get_dpms(int fd, drmModeConnector *conn)
161 drmModePropertyRes *prop;
163 for (i = 0; i < conn->count_props; ++i) {
164 prop = drmModeGetProperty(fd, conn->props[i]);
166 log_error("cannot get DRM property (%d): %m", errno);
170 if (!strcmp(prop->name, "DPMS")) {
171 switch (conn->prop_values[i]) {
172 case DRM_MODE_DPMS_ON:
175 case DRM_MODE_DPMS_STANDBY:
176 ret = UTERM_DPMS_STANDBY;
178 case DRM_MODE_DPMS_SUSPEND:
179 ret = UTERM_DPMS_SUSPEND;
181 case DRM_MODE_DPMS_OFF:
183 ret = UTERM_DPMS_OFF;
186 drmModeFreeProperty(prop);
189 drmModeFreeProperty(prop);
192 if (i == conn->count_props)
193 log_warn("display does not support DPMS");
194 return UTERM_DPMS_UNKNOWN;
197 int uterm_drm_display_init(struct uterm_display *disp, void *data)
199 struct uterm_drm_display *d;
201 d = malloc(sizeof(*d));
204 memset(d, 0, sizeof(*d));
211 void uterm_drm_display_destroy(struct uterm_display *disp)
216 int uterm_drm_display_activate(struct uterm_display *disp, int fd)
218 struct uterm_video *video = disp->video;
219 struct uterm_drm_display *ddrm = disp->data;
221 drmModeConnector *conn;
225 res = drmModeGetResources(fd);
227 log_err("cannot get resources for display %p", disp);
230 conn = drmModeGetConnector(fd, ddrm->conn_id);
232 log_err("cannot get connector for display %p", disp);
233 drmModeFreeResources(res);
238 for (i = 0; i < conn->count_encoders; ++i) {
239 enc = drmModeGetEncoder(fd, conn->encoders[i]);
242 crtc = uterm_drm_video_find_crtc(video, res, enc);
243 drmModeFreeEncoder(enc);
248 drmModeFreeConnector(conn);
249 drmModeFreeResources(res);
252 log_warn("cannot find crtc for new display");
256 ddrm->crtc_id = crtc;
257 if (ddrm->saved_crtc)
258 drmModeFreeCrtc(ddrm->saved_crtc);
259 ddrm->saved_crtc = drmModeGetCrtc(fd, ddrm->crtc_id);
264 void uterm_drm_display_deactivate(struct uterm_display *disp, int fd)
266 struct uterm_drm_display *ddrm = disp->data;
268 if (ddrm->saved_crtc) {
269 if (disp->video->flags & VIDEO_AWAKE) {
270 drmModeSetCrtc(fd, ddrm->saved_crtc->crtc_id,
271 ddrm->saved_crtc->buffer_id,
275 &ddrm->saved_crtc->mode);
277 drmModeFreeCrtc(ddrm->saved_crtc);
278 ddrm->saved_crtc = NULL;
284 int uterm_drm_display_bind(struct uterm_video *video,
285 struct uterm_display *disp, drmModeRes *res,
286 drmModeConnector *conn, int fd)
288 struct uterm_mode *mode;
290 struct uterm_drm_display *ddrm = disp->data;
292 for (i = 0; i < conn->count_modes; ++i) {
293 ret = mode_new(&mode, &uterm_drm_mode_ops);
296 uterm_drm_mode_set(mode, &conn->modes[i]);
297 mode->next = disp->modes;
300 /* TODO: more sophisticated default-mode selection */
301 if (!disp->default_mode)
302 disp->default_mode = mode;
306 log_warn("no valid mode for display found");
310 ddrm->conn_id = conn->connector_id;
311 disp->flags |= DISPLAY_AVAILABLE;
312 disp->next = video->displays;
313 video->displays = disp;
314 disp->dpms = uterm_drm_get_dpms(fd, conn);
316 log_info("display %p DPMS is %s", disp,
317 uterm_dpms_to_name(disp->dpms));
319 VIDEO_CB(video, disp, UTERM_NEW);
323 void uterm_drm_display_unbind(struct uterm_display *disp)
325 VIDEO_CB(disp->video, disp, UTERM_GONE);
326 uterm_display_deactivate(disp);
328 disp->flags &= ~DISPLAY_AVAILABLE;
331 static void event(struct ev_fd *fd, int mask, void *data)
333 struct uterm_video *video = data;
334 struct uterm_drm_video *vdrm = video->data;
337 /* TODO: run this in a loop with O_NONBLOCK */
338 if (mask & EV_READABLE) {
339 memset(&ev, 0, sizeof(ev));
340 ev.version = DRM_EVENT_CONTEXT_VERSION;
341 ev.page_flip_handler = vdrm->page_flip;
342 drmHandleEvent(vdrm->fd, &ev);
345 /* TODO: forward HUP to caller */
346 if (mask & (EV_HUP | EV_ERR)) {
347 log_err("error or hangup on DRM fd");
348 ev_eloop_rm_fd(vdrm->efd);
354 int uterm_drm_video_init(struct uterm_video *video, const char *node,
355 uterm_drm_page_flip_t pflip, void *data)
357 struct uterm_drm_video *vdrm;
360 log_info("new drm device via %s", node);
362 vdrm = malloc(sizeof(*vdrm));
365 memset(vdrm, 0, sizeof(*vdrm));
368 vdrm->page_flip = pflip;
370 vdrm->fd = open(node, O_RDWR | O_CLOEXEC);
372 log_err("cannot open drm device %s (%d): %m", node, errno);
376 /* TODO: fix the race-condition with DRM-Master-on-open */
377 drmDropMaster(vdrm->fd);
379 ret = ev_eloop_new_fd(video->eloop, &vdrm->efd, vdrm->fd, EV_READABLE,
384 video->flags |= VIDEO_HOTPLUG;
394 void uterm_drm_video_destroy(struct uterm_video *video)
396 struct uterm_drm_video *vdrm = video->data;
398 ev_eloop_rm_fd(vdrm->efd);
403 int uterm_drm_video_find_crtc(struct uterm_video *video, drmModeRes *res,
407 struct uterm_display *iter;
408 struct uterm_drm_display *ddrm;
410 for (i = 0; i < res->count_crtcs; ++i) {
411 if (enc->possible_crtcs & (1 << i)) {
412 crtc = res->crtcs[i];
413 for (iter = video->displays; iter; iter = iter->next) {
415 if (ddrm->crtc_id == crtc)
426 static void bind_display(struct uterm_video *video, drmModeRes *res,
427 drmModeConnector *conn,
428 const struct display_ops *ops)
430 struct uterm_drm_video *vdrm = video->data;
431 struct uterm_display *disp;
434 ret = display_new(&disp, ops, video);
438 ret = uterm_drm_display_bind(video, disp, res, conn, vdrm->fd);
440 uterm_display_unref(disp);
445 static void unbind_display(struct uterm_display *disp)
447 if (!display_is_conn(disp))
450 uterm_drm_display_unbind(disp);
451 uterm_display_unref(disp);
454 int uterm_drm_video_hotplug(struct uterm_video *video,
455 const struct display_ops *ops)
457 struct uterm_drm_video *vdrm = video->data;
459 drmModeConnector *conn;
460 struct uterm_display *disp, *tmp;
461 struct uterm_drm_display *ddrm;
464 if (!video_is_awake(video) || !video_need_hotplug(video))
467 res = drmModeGetResources(vdrm->fd);
469 log_err("cannot retrieve drm resources");
473 for (disp = video->displays; disp; disp = disp->next)
474 disp->flags &= ~DISPLAY_AVAILABLE;
476 for (i = 0; i < res->count_connectors; ++i) {
477 conn = drmModeGetConnector(vdrm->fd, res->connectors[i]);
480 if (conn->connection == DRM_MODE_CONNECTED) {
481 for (disp = video->displays; disp; disp = disp->next) {
483 if (ddrm->conn_id == res->connectors[i]) {
484 disp->flags |= DISPLAY_AVAILABLE;
489 bind_display(video, res, conn, ops);
491 drmModeFreeConnector(conn);
494 drmModeFreeResources(res);
496 while (video->displays) {
497 tmp = video->displays;
498 if (tmp->flags & DISPLAY_AVAILABLE)
501 video->displays = tmp->next;
505 for (disp = video->displays; disp && disp->next; ) {
507 if (tmp->flags & DISPLAY_AVAILABLE) {
510 disp->next = tmp->next;
516 video->flags &= ~VIDEO_HOTPLUG;
520 int uterm_drm_video_wake_up(struct uterm_video *video,
521 const struct display_ops *ops)
524 struct uterm_drm_video *vdrm = video->data;
526 ret = drmSetMaster(vdrm->fd);
528 log_err("cannot set DRM-master");
532 video->flags |= VIDEO_AWAKE;
533 ret = uterm_drm_video_hotplug(video, ops);
535 video->flags &= ~VIDEO_AWAKE;
536 drmDropMaster(vdrm->fd);
543 void uterm_drm_video_sleep(struct uterm_video *video)
545 struct uterm_drm_video *vdrm = video->data;
547 drmDropMaster(vdrm->fd);
548 video->flags &= ~VIDEO_AWAKE;
551 int uterm_drm_video_poll(struct uterm_video *video,
552 const struct display_ops *ops)
554 video->flags |= VIDEO_HOTPLUG;
555 return uterm_drm_video_hotplug(video, ops);