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_set_dpms(struct uterm_display *disp, int state)
287 struct uterm_drm_display *ddrm = disp->data;
288 struct uterm_drm_video *vdrm;
290 if (!display_is_conn(disp) || !video_is_awake(disp->video))
293 vdrm = disp->video->data;
294 log_info("setting DPMS of display %p to %s", disp,
295 uterm_dpms_to_name(state));
297 ret = uterm_drm_set_dpms(vdrm->fd, ddrm->conn_id, state);
305 int uterm_drm_display_bind(struct uterm_video *video,
306 struct uterm_display *disp, drmModeRes *res,
307 drmModeConnector *conn, int fd)
309 struct uterm_mode *mode;
311 struct uterm_drm_display *ddrm = disp->data;
313 for (i = 0; i < conn->count_modes; ++i) {
314 ret = mode_new(&mode, &uterm_drm_mode_ops);
317 uterm_drm_mode_set(mode, &conn->modes[i]);
318 mode->next = disp->modes;
321 /* TODO: more sophisticated default-mode selection */
322 if (!disp->default_mode)
323 disp->default_mode = mode;
327 log_warn("no valid mode for display found");
331 ddrm->conn_id = conn->connector_id;
332 disp->flags |= DISPLAY_AVAILABLE;
333 disp->next = video->displays;
334 video->displays = disp;
335 disp->dpms = uterm_drm_get_dpms(fd, conn);
337 log_info("display %p DPMS is %s", disp,
338 uterm_dpms_to_name(disp->dpms));
340 VIDEO_CB(video, disp, UTERM_NEW);
344 void uterm_drm_display_unbind(struct uterm_display *disp)
346 VIDEO_CB(disp->video, disp, UTERM_GONE);
347 uterm_display_deactivate(disp);
349 disp->flags &= ~DISPLAY_AVAILABLE;
352 static void event(struct ev_fd *fd, int mask, void *data)
354 struct uterm_video *video = data;
355 struct uterm_drm_video *vdrm = video->data;
358 /* TODO: run this in a loop with O_NONBLOCK */
359 if (mask & EV_READABLE) {
360 memset(&ev, 0, sizeof(ev));
361 ev.version = DRM_EVENT_CONTEXT_VERSION;
362 ev.page_flip_handler = vdrm->page_flip;
363 drmHandleEvent(vdrm->fd, &ev);
366 /* TODO: forward HUP to caller */
367 if (mask & (EV_HUP | EV_ERR)) {
368 log_err("error or hangup on DRM fd");
369 ev_eloop_rm_fd(vdrm->efd);
375 int uterm_drm_video_init(struct uterm_video *video, const char *node,
376 uterm_drm_page_flip_t pflip, void *data)
378 struct uterm_drm_video *vdrm;
381 log_info("new drm device via %s", node);
383 vdrm = malloc(sizeof(*vdrm));
386 memset(vdrm, 0, sizeof(*vdrm));
389 vdrm->page_flip = pflip;
391 vdrm->fd = open(node, O_RDWR | O_CLOEXEC);
393 log_err("cannot open drm device %s (%d): %m", node, errno);
397 /* TODO: fix the race-condition with DRM-Master-on-open */
398 drmDropMaster(vdrm->fd);
400 ret = ev_eloop_new_fd(video->eloop, &vdrm->efd, vdrm->fd, EV_READABLE,
405 video->flags |= VIDEO_HOTPLUG;
415 void uterm_drm_video_destroy(struct uterm_video *video)
417 struct uterm_drm_video *vdrm = video->data;
419 ev_eloop_rm_fd(vdrm->efd);
424 int uterm_drm_video_find_crtc(struct uterm_video *video, drmModeRes *res,
428 struct uterm_display *iter;
429 struct uterm_drm_display *ddrm;
431 for (i = 0; i < res->count_crtcs; ++i) {
432 if (enc->possible_crtcs & (1 << i)) {
433 crtc = res->crtcs[i];
434 for (iter = video->displays; iter; iter = iter->next) {
436 if (ddrm->crtc_id == crtc)
447 static void bind_display(struct uterm_video *video, drmModeRes *res,
448 drmModeConnector *conn,
449 const struct display_ops *ops)
451 struct uterm_drm_video *vdrm = video->data;
452 struct uterm_display *disp;
455 ret = display_new(&disp, ops, video);
459 ret = uterm_drm_display_bind(video, disp, res, conn, vdrm->fd);
461 uterm_display_unref(disp);
466 static void unbind_display(struct uterm_display *disp)
468 if (!display_is_conn(disp))
471 uterm_drm_display_unbind(disp);
472 uterm_display_unref(disp);
475 int uterm_drm_video_hotplug(struct uterm_video *video,
476 const struct display_ops *ops)
478 struct uterm_drm_video *vdrm = video->data;
480 drmModeConnector *conn;
481 struct uterm_display *disp, *tmp;
482 struct uterm_drm_display *ddrm;
485 if (!video_is_awake(video) || !video_need_hotplug(video))
488 res = drmModeGetResources(vdrm->fd);
490 log_err("cannot retrieve drm resources");
494 for (disp = video->displays; disp; disp = disp->next)
495 disp->flags &= ~DISPLAY_AVAILABLE;
497 for (i = 0; i < res->count_connectors; ++i) {
498 conn = drmModeGetConnector(vdrm->fd, res->connectors[i]);
501 if (conn->connection == DRM_MODE_CONNECTED) {
502 for (disp = video->displays; disp; disp = disp->next) {
504 if (ddrm->conn_id == res->connectors[i]) {
505 disp->flags |= DISPLAY_AVAILABLE;
510 bind_display(video, res, conn, ops);
512 drmModeFreeConnector(conn);
515 drmModeFreeResources(res);
517 while (video->displays) {
518 tmp = video->displays;
519 if (tmp->flags & DISPLAY_AVAILABLE)
522 video->displays = tmp->next;
526 for (disp = video->displays; disp && disp->next; ) {
528 if (tmp->flags & DISPLAY_AVAILABLE) {
531 disp->next = tmp->next;
537 video->flags &= ~VIDEO_HOTPLUG;
541 int uterm_drm_video_wake_up(struct uterm_video *video,
542 const struct display_ops *ops)
545 struct uterm_drm_video *vdrm = video->data;
547 ret = drmSetMaster(vdrm->fd);
549 log_err("cannot set DRM-master");
553 video->flags |= VIDEO_AWAKE;
554 ret = uterm_drm_video_hotplug(video, ops);
556 video->flags &= ~VIDEO_AWAKE;
557 drmDropMaster(vdrm->fd);
564 void uterm_drm_video_sleep(struct uterm_video *video)
566 struct uterm_drm_video *vdrm = video->data;
568 drmDropMaster(vdrm->fd);
569 video->flags &= ~VIDEO_AWAKE;
572 int uterm_drm_video_poll(struct uterm_video *video,
573 const struct display_ops *ops)
575 video->flags |= VIDEO_HOTPLUG;
576 return uterm_drm_video_hotplug(video, ops);