Systemd/src/libsystemd-terminal/grdev.c

1363 lines
39 KiB
C

/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright (C) 2014 David Herrmann <dh.herrmann@gmail.com>
systemd is free software; you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version.
systemd is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/
#include <inttypes.h>
#include <libudev.h>
#include <stdbool.h>
#include <stdlib.h>
#include <systemd/sd-bus.h>
#include <systemd/sd-event.h>
#include <systemd/sd-login.h>
#include "grdev.h"
#include "grdev-internal.h"
#include "hashmap.h"
#include "login-shared.h"
#include "macro.h"
#include "udev-util.h"
#include "util.h"
static void pipe_enable(grdev_pipe *pipe);
static void pipe_disable(grdev_pipe *pipe);
static void card_modified(grdev_card *card);
static void session_frame(grdev_session *session, grdev_display *display);
/*
* Displays
*/
static inline grdev_tile *tile_leftmost(grdev_tile *tile) {
if (!tile)
return NULL;
while (tile->type == GRDEV_TILE_NODE && tile->node.child_list)
tile = tile->node.child_list;
return tile;
}
#define TILE_FOREACH(_root, _i) \
for (_i = tile_leftmost(_root); _i; _i = tile_leftmost(_i->children_by_node_next) ? : _i->parent)
#define TILE_FOREACH_SAFE(_root, _i, _next) \
for (_i = tile_leftmost(_root); _i && ((_next = tile_leftmost(_i->children_by_node_next) ? : _i->parent), true); _i = _next)
static void tile_link(grdev_tile *tile, grdev_tile *parent) {
grdev_display *display;
grdev_tile *t;
assert(tile);
assert(!tile->parent);
assert(!tile->display);
assert(parent);
assert(parent->type == GRDEV_TILE_NODE);
display = parent->display;
assert(!display || !display->enabled);
++parent->node.n_children;
LIST_PREPEND(children_by_node, parent->node.child_list, tile);
tile->parent = parent;
if (display) {
display->modified = true;
TILE_FOREACH(tile, t) {
t->display = display;
if (t->type == GRDEV_TILE_LEAF) {
++display->n_leafs;
if (display->enabled)
pipe_enable(t->leaf.pipe);
}
}
}
}
static void tile_unlink(grdev_tile *tile) {
grdev_tile *parent, *t;
grdev_display *display;
assert(tile);
display = tile->display;
parent = tile->parent;
if (!parent) {
assert(!display);
return;
}
assert(parent->type == GRDEV_TILE_NODE);
assert(parent->display == display);
assert(parent->node.n_children > 0);
--parent->node.n_children;
LIST_REMOVE(children_by_node, parent->node.child_list, tile);
tile->parent = NULL;
if (display) {
display->modified = true;
TILE_FOREACH(tile, t) {
t->display = NULL;
if (t->type == GRDEV_TILE_LEAF) {
--display->n_leafs;
t->leaf.pipe->cache = NULL;
pipe_disable(t->leaf.pipe);
}
}
}
/* Tile trees are driven by leafs. Internal nodes have no owner, thus,
* we must take care to not leave them around. Therefore, whenever we
* unlink any part of a tree, we also destroy the parent, in case it's
* now stale.
* Parents are stale if they have no children and either have no display
* or if they are intermediate nodes (i.e, they have a parent).
* This means, you can easily create trees, but you can never partially
* move or destruct them so far. They're always reduced to minimal form
* if you cut them. This might change later, but so far we didn't need
* partial destruction or the ability to move whole trees. */
if (parent->node.n_children < 1 && (parent->parent || !parent->display))
grdev_tile_free(parent);
}
static int tile_new(grdev_tile **out) {
_cleanup_(grdev_tile_freep) grdev_tile *tile = NULL;
assert(out);
tile = new0(grdev_tile, 1);
if (!tile)
return -ENOMEM;
tile->type = (unsigned)-1;
*out = tile;
tile = NULL;
return 0;
}
int grdev_tile_new_leaf(grdev_tile **out, grdev_pipe *pipe) {
_cleanup_(grdev_tile_freep) grdev_tile *tile = NULL;
int r;
assert_return(out, -EINVAL);
assert_return(pipe, -EINVAL);
assert_return(!pipe->tile, -EINVAL);
r = tile_new(&tile);
if (r < 0)
return r;
tile->type = GRDEV_TILE_LEAF;
tile->leaf.pipe = pipe;
if (out)
*out = tile;
tile = NULL;
return 0;
}
int grdev_tile_new_node(grdev_tile **out) {
_cleanup_(grdev_tile_freep) grdev_tile *tile = NULL;
int r;
assert_return(out, -EINVAL);
r = tile_new(&tile);
if (r < 0)
return r;
tile->type = GRDEV_TILE_NODE;
*out = tile;
tile = NULL;
return 0;
}
grdev_tile *grdev_tile_free(grdev_tile *tile) {
if (!tile)
return NULL;
tile_unlink(tile);
switch (tile->type) {
case GRDEV_TILE_LEAF:
assert(!tile->parent);
assert(!tile->display);
assert(tile->leaf.pipe);
break;
case GRDEV_TILE_NODE:
assert(!tile->parent);
assert(!tile->display);
assert(tile->node.n_children == 0);
break;
}
free(tile);
return NULL;
}
grdev_display *grdev_find_display(grdev_session *session, const char *name) {
assert_return(session, NULL);
assert_return(name, NULL);
return hashmap_get(session->display_map, name);
}
int grdev_display_new(grdev_display **out, grdev_session *session, const char *name) {
_cleanup_(grdev_display_freep) grdev_display *display = NULL;
int r;
assert(session);
assert(name);
display = new0(grdev_display, 1);
if (!display)
return -ENOMEM;
display->session = session;
display->name = strdup(name);
if (!display->name)
return -ENOMEM;
r = grdev_tile_new_node(&display->tile);
if (r < 0)
return r;
display->tile->display = display;
r = hashmap_put(session->display_map, display->name, display);
if (r < 0)
return r;
if (out)
*out = display;
display = NULL;
return 0;
}
grdev_display *grdev_display_free(grdev_display *display) {
if (!display)
return NULL;
assert(!display->public);
assert(!display->enabled);
assert(!display->modified);
assert(display->n_leafs == 0);
assert(display->n_pipes == 0);
if (display->name)
hashmap_remove_value(display->session->display_map, display->name, display);
if (display->tile) {
display->tile->display = NULL;
grdev_tile_free(display->tile);
}
free(display->pipes);
free(display->name);
free(display);
return NULL;
}
void grdev_display_set_userdata(grdev_display *display, void *userdata) {
assert(display);
display->userdata = userdata;
}
void *grdev_display_get_userdata(grdev_display *display) {
assert_return(display, NULL);
return display->userdata;
}
const char *grdev_display_get_name(grdev_display *display) {
assert_return(display, NULL);
return display->name;
}
uint32_t grdev_display_get_width(grdev_display *display) {
assert_return(display, 0);
return display->width;
}
uint32_t grdev_display_get_height(grdev_display *display) {
assert_return(display, 0);
return display->height;
}
bool grdev_display_is_enabled(grdev_display *display) {
return display && display->enabled;
}
void grdev_display_enable(grdev_display *display) {
grdev_tile *t;
assert(display);
if (!display->enabled) {
display->enabled = true;
TILE_FOREACH(display->tile, t)
if (t->type == GRDEV_TILE_LEAF)
pipe_enable(t->leaf.pipe);
}
}
void grdev_display_disable(grdev_display *display) {
grdev_tile *t;
assert(display);
if (display->enabled) {
display->enabled = false;
TILE_FOREACH(display->tile, t)
if (t->type == GRDEV_TILE_LEAF)
pipe_disable(t->leaf.pipe);
}
}
const grdev_display_target *grdev_display_next_target(grdev_display *display, const grdev_display_target *prev) {
grdev_display_cache *cache;
size_t idx;
assert_return(display, NULL);
assert_return(!display->modified, NULL);
assert_return(display->enabled, NULL);
if (prev) {
cache = container_of(prev, grdev_display_cache, target);
assert(cache->pipe);
assert(cache->pipe->tile->display == display);
assert(display->pipes >= cache);
idx = cache - display->pipes + 1;
} else {
idx = 0;
}
for (cache = display->pipes + idx; idx < display->n_pipes; ++idx, ++cache) {
grdev_display_target *target;
grdev_pipe *pipe;
grdev_fb *fb;
pipe = cache->pipe;
target = &cache->target;
if (!pipe->running || !pipe->enabled)
continue;
/* find suitable back-buffer */
if (!pipe->back) {
if (!pipe->vtable->target)
continue;
if (!(fb = pipe->vtable->target(pipe)))
continue;
assert(fb == pipe->back);
}
target->front = pipe->front;
target->back = pipe->back;
return target;
}
return NULL;
}
void grdev_display_flip_target(grdev_display *display, const grdev_display_target *target) {
grdev_display_cache *cache;
assert(display);
assert(!display->modified);
assert(display->enabled);
assert(target);
cache = container_of(target, grdev_display_cache, target);
assert(cache->pipe);
assert(cache->pipe->tile->display == display);
cache->pipe->flip = true;
}
static void display_cache_apply(grdev_display_cache *c, grdev_tile *l) {
uint32_t x, y, width, height;
grdev_display_target *t;
assert(c);
assert(l);
assert(l->cache_w >= c->target.width + c->target.x);
assert(l->cache_h >= c->target.height + c->target.y);
t = &c->target;
/* rotate child */
t->rotate = (t->rotate + l->rotate) & 0x3;
x = t->x;
y = t->y;
width = t->width;
height = t->height;
switch (l->rotate) {
case GRDEV_ROTATE_0:
break;
case GRDEV_ROTATE_90:
t->x = l->cache_h - (height + y);
t->y = x;
t->width = height;
t->height = width;
break;
case GRDEV_ROTATE_180:
t->x = l->cache_w - (width + x);
t->y = l->cache_h - (height + y);
break;
case GRDEV_ROTATE_270:
t->x = y;
t->y = l->cache_w - (width + x);
t->width = height;
t->height = width;
break;
}
/* flip child */
t->flip ^= l->flip;
if (l->flip & GRDEV_FLIP_HORIZONTAL)
t->x = l->cache_w - (t->width + t->x);
if (l->flip & GRDEV_FLIP_VERTICAL)
t->y = l->cache_h - (t->height + t->y);
/* move child */
t->x += l->x;
t->y += l->y;
}
static void display_cache_targets(grdev_display *display) {
grdev_display_cache *c;
grdev_tile *tile;
assert(display);
/* depth-first with children before parent */
for (tile = tile_leftmost(display->tile);
tile;
tile = tile_leftmost(tile->children_by_node_next) ? : tile->parent) {
if (tile->type == GRDEV_TILE_LEAF) {
grdev_pipe *p;
/* We're at a leaf and no parent has been cached, yet.
* Copy the pipe information into the target cache and
* update our global pipe-caches if required. */
assert(tile->leaf.pipe);
assert(display->n_pipes + 1 <= display->max_pipes);
p = tile->leaf.pipe;
c = &display->pipes[display->n_pipes++];
zero(*c);
c->pipe = p;
c->pipe->cache = c;
c->target.width = p->width;
c->target.height = p->height;
tile->cache_w = p->width;
tile->cache_h = p->height;
/* all new tiles are incomplete due to geometry changes */
c->incomplete = true;
display_cache_apply(c, tile);
} else {
grdev_tile *child, *l;
/* We're now at a node with all its children already
* computed (depth-first, child before parent). We
* first need to know the size of our tile, then we
* recurse into all leafs and update their cache. */
tile->cache_w = 0;
tile->cache_h = 0;
LIST_FOREACH(children_by_node, child, tile->node.child_list) {
if (child->x + child->cache_w > tile->cache_w)
tile->cache_w = child->x + child->cache_w;
if (child->y + child->cache_h > tile->cache_h)
tile->cache_h = child->y + child->cache_h;
}
assert(tile->cache_w > 0);
assert(tile->cache_h > 0);
TILE_FOREACH(tile, l)
if (l->type == GRDEV_TILE_LEAF)
display_cache_apply(l->leaf.pipe->cache, tile);
}
}
}
static bool display_cache(grdev_display *display) {
grdev_tile *tile;
size_t n;
void *t;
int r;
assert(display);
if (!display->modified)
return false;
display->modified = false;
display->framed = false;
display->n_pipes = 0;
display->width = 0;
display->height = 0;
if (display->n_leafs < 1)
return false;
TILE_FOREACH(display->tile, tile)
if (tile->type == GRDEV_TILE_LEAF)
tile->leaf.pipe->cache = NULL;
if (display->n_leafs > display->max_pipes) {
n = ALIGN_POWER2(display->n_leafs);
if (!n) {
r = -ENOMEM;
goto out;
}
t = realloc_multiply(display->pipes, sizeof(*display->pipes), n);
if (!t) {
r = -ENOMEM;
goto out;
}
display->pipes = t;
display->max_pipes = n;
}
display_cache_targets(display);
display->width = display->tile->cache_w;
display->height = display->tile->cache_h;
r = 0;
out:
if (r < 0)
log_debug_errno(r, "grdev: %s/%s: cannot cache pipes: %m",
display->session->name, display->name);
return true;
}
/*
* Pipes
*/
grdev_pipe *grdev_find_pipe(grdev_card *card, const char *name) {
assert_return(card, NULL);
assert_return(name, NULL);
return hashmap_get(card->pipe_map, name);
}
static int pipe_vsync_fn(sd_event_source *src, uint64_t usec, void *userdata) {
grdev_pipe *pipe = userdata;
grdev_pipe_frame(pipe);
return 0;
}
int grdev_pipe_add(grdev_pipe *pipe, const char *name, size_t n_fbs) {
int r;
assert_return(pipe, -EINVAL);
assert_return(pipe->vtable, -EINVAL);
assert_return(pipe->vtable->free, -EINVAL);
assert_return(pipe->card, -EINVAL);
assert_return(pipe->card->session, -EINVAL);
assert_return(!pipe->cache, -EINVAL);
assert_return(pipe->width > 0, -EINVAL);
assert_return(pipe->height > 0, -EINVAL);
assert_return(pipe->vrefresh > 0, -EINVAL);
assert_return(!pipe->enabled, -EINVAL);
assert_return(!pipe->running, -EINVAL);
assert_return(name, -EINVAL);
pipe->name = strdup(name);
if (!pipe->name)
return -ENOMEM;
if (n_fbs > 0) {
pipe->fbs = new0(grdev_fb*, n_fbs);
if (!pipe->fbs)
return -ENOMEM;
pipe->max_fbs = n_fbs;
}
r = grdev_tile_new_leaf(&pipe->tile, pipe);
if (r < 0)
return r;
r = sd_event_add_time(pipe->card->session->context->event,
&pipe->vsync_src,
CLOCK_MONOTONIC,
0,
10 * USEC_PER_MSEC,
pipe_vsync_fn,
pipe);
if (r < 0)
return r;
r = sd_event_source_set_enabled(pipe->vsync_src, SD_EVENT_OFF);
if (r < 0)
return r;
r = hashmap_put(pipe->card->pipe_map, pipe->name, pipe);
if (r < 0)
return r;
card_modified(pipe->card);
return 0;
}
grdev_pipe *grdev_pipe_free(grdev_pipe *pipe) {
grdev_pipe tmp;
if (!pipe)
return NULL;
assert(pipe->card);
assert(pipe->vtable);
assert(pipe->vtable->free);
if (pipe->name)
hashmap_remove_value(pipe->card->pipe_map, pipe->name, pipe);
if (pipe->tile)
tile_unlink(pipe->tile);
assert(!pipe->cache);
tmp = *pipe;
pipe->vtable->free(pipe);
sd_event_source_unref(tmp.vsync_src);
grdev_tile_free(tmp.tile);
card_modified(tmp.card);
free(tmp.fbs);
free(tmp.name);
return NULL;
}
static void pipe_enable(grdev_pipe *pipe) {
assert(pipe);
if (!pipe->enabled) {
pipe->enabled = true;
if (pipe->vtable->enable)
pipe->vtable->enable(pipe);
}
}
static void pipe_disable(grdev_pipe *pipe) {
assert(pipe);
if (pipe->enabled) {
pipe->enabled = false;
if (pipe->vtable->disable)
pipe->vtable->disable(pipe);
}
}
void grdev_pipe_ready(grdev_pipe *pipe, bool running) {
assert(pipe);
/* grdev_pipe_ready() is used by backends to notify about pipe state
* changed. If a pipe is ready, it can be fully used by us (available,
* enabled and accessible). Backends can disable pipes at any time
* (like for async revocation), but can only enable them from parent
* context. Otherwise, we might call user-callbacks recursively. */
if (pipe->running == running)
return;
pipe->running = running;
/* runtime events for unused pipes are not interesting */
if (pipe->cache && pipe->enabled) {
grdev_display *display = pipe->tile->display;
assert(display);
if (running)
session_frame(display->session, display);
else
pipe->cache->incomplete = true;
}
}
void grdev_pipe_frame(grdev_pipe *pipe) {
grdev_display *display;
assert(pipe);
/* if pipe is unused, ignore any frame events */
if (!pipe->cache || !pipe->enabled)
return;
display = pipe->tile->display;
assert(display);
grdev_pipe_schedule(pipe, 0);
session_frame(display->session, display);
}
void grdev_pipe_schedule(grdev_pipe *pipe, uint64_t frames) {
int r;
uint64_t ts;
if (!frames) {
sd_event_source_set_enabled(pipe->vsync_src, SD_EVENT_OFF);
return;
}
r = sd_event_now(pipe->card->session->context->event, CLOCK_MONOTONIC, &ts);
if (r < 0)
goto error;
ts += frames * USEC_PER_MSEC * 1000ULL / pipe->vrefresh;
r = sd_event_source_set_time(pipe->vsync_src, ts);
if (r < 0)
goto error;
r = sd_event_source_set_enabled(pipe->vsync_src, SD_EVENT_ONESHOT);
if (r < 0)
goto error;
return;
error:
log_debug_errno(r, "grdev: %s/%s/%s: cannot schedule vsync timer: %m",
pipe->card->session->name, pipe->card->name, pipe->name);
}
/*
* Cards
*/
grdev_card *grdev_find_card(grdev_session *session, const char *name) {
assert_return(session, NULL);
assert_return(name, NULL);
return hashmap_get(session->card_map, name);
}
int grdev_card_add(grdev_card *card, const char *name) {
int r;
assert_return(card, -EINVAL);
assert_return(card->vtable, -EINVAL);
assert_return(card->vtable->free, -EINVAL);
assert_return(card->session, -EINVAL);
assert_return(name, -EINVAL);
card->name = strdup(name);
if (!card->name)
return -ENOMEM;
card->pipe_map = hashmap_new(&string_hash_ops);
if (!card->pipe_map)
return -ENOMEM;
r = hashmap_put(card->session->card_map, card->name, card);
if (r < 0)
return r;
return 0;
}
grdev_card *grdev_card_free(grdev_card *card) {
grdev_card tmp;
if (!card)
return NULL;
assert(!card->enabled);
assert(card->vtable);
assert(card->vtable->free);
if (card->name)
hashmap_remove_value(card->session->card_map, card->name, card);
tmp = *card;
card->vtable->free(card);
assert(hashmap_size(tmp.pipe_map) == 0);
hashmap_free(tmp.pipe_map);
free(tmp.name);
return NULL;
}
static void card_modified(grdev_card *card) {
assert(card);
assert(card->session->n_pins > 0);
card->modified = true;
}
static void grdev_card_enable(grdev_card *card) {
assert(card);
if (!card->enabled) {
card->enabled = true;
if (card->vtable->enable)
card->vtable->enable(card);
}
}
static void grdev_card_disable(grdev_card *card) {
assert(card);
if (card->enabled) {
card->enabled = false;
if (card->vtable->disable)
card->vtable->disable(card);
}
}
/*
* Sessions
*/
static void session_raise(grdev_session *session, grdev_event *event) {
session->event_fn(session, session->userdata, event);
}
static void session_raise_display_add(grdev_session *session, grdev_display *display) {
grdev_event event = {
.type = GRDEV_EVENT_DISPLAY_ADD,
.display_add = {
.display = display,
},
};
session_raise(session, &event);
}
static void session_raise_display_remove(grdev_session *session, grdev_display *display) {
grdev_event event = {
.type = GRDEV_EVENT_DISPLAY_REMOVE,
.display_remove = {
.display = display,
},
};
session_raise(session, &event);
}
static void session_raise_display_change(grdev_session *session, grdev_display *display) {
grdev_event event = {
.type = GRDEV_EVENT_DISPLAY_CHANGE,
.display_change = {
.display = display,
},
};
session_raise(session, &event);
}
static void session_raise_display_frame(grdev_session *session, grdev_display *display) {
grdev_event event = {
.type = GRDEV_EVENT_DISPLAY_FRAME,
.display_frame = {
.display = display,
},
};
session_raise(session, &event);
}
static void session_add_card(grdev_session *session, grdev_card *card) {
assert(session);
assert(card);
log_debug("grdev: %s: add card '%s'", session->name, card->name);
/* Cards are not exposed to users, but managed internally. Cards are
* enabled if the session is enabled, and will track that state. The
* backend can probe the card at any time, but only if enabled. It
* will then add pipes according to hardware state.
* That is, the card may create pipes as soon as we enable it here. */
if (session->enabled)
grdev_card_enable(card);
}
static void session_remove_card(grdev_session *session, grdev_card *card) {
assert(session);
assert(card);
log_debug("grdev: %s: remove card '%s'", session->name, card->name);
/* As cards are not exposed, it can never be accessed by outside
* users and we can simply remove it. Disabling the card does not
* necessarily drop all pipes of the card. This is usually deferred
* to card destruction (as pipes are cached as long as FDs remain
* open). Therefore, the card destruction might cause pipes, and thus
* visible displays, to be removed. */
grdev_card_disable(card);
grdev_card_free(card);
}
static void session_add_display(grdev_session *session, grdev_display *display) {
assert(session);
assert(display);
assert(!display->enabled);
log_debug("grdev: %s: add display '%s'", session->name, display->name);
/* Displays are the main entity for public API users. We create them
* independent of card backends and they wrap any underlying display
* architecture. Displays are public at all times, thus, may be entered
* by outside users at any time. */
display->public = true;
session_raise_display_add(session, display);
}
static void session_remove_display(grdev_session *session, grdev_display *display) {
assert(session);
assert(display);
log_debug("grdev: %s: remove display '%s'", session->name, display->name);
/* Displays are public, so we have to be careful when removing them.
* We first tell users about their removal, disable them and then drop
* them. We now, after the notification, no external access will
* happen. Therefore, we can release the tiles afterwards safely. */
if (display->public) {
display->public = false;
session_raise_display_remove(session, display);
}
grdev_display_disable(display);
grdev_display_free(display);
}
static void session_change_display(grdev_session *session, grdev_display *display) {
bool changed;
assert(session);
assert(display);
changed = display_cache(display);
if (display->n_leafs == 0) {
session_remove_display(session, display);
} else if (!display->public) {
session_add_display(session, display);
session_frame(session, display);
} else if (changed) {
session_raise_display_change(session, display);
session_frame(session, display);
} else if (display->framed) {
session_frame(session, display);
}
}
static void session_frame(grdev_session *session, grdev_display *display) {
assert(session);
assert(display);
display->framed = false;
if (!display->enabled || !session->enabled)
return;
if (session->n_pins > 0)
display->framed = true;
else
session_raise_display_frame(session, display);
}
int grdev_session_new(grdev_session **out,
grdev_context *context,
unsigned int flags,
const char *name,
grdev_event_fn event_fn,
void *userdata) {
_cleanup_(grdev_session_freep) grdev_session *session = NULL;
int r;
assert(out);
assert(context);
assert(name);
assert(event_fn);
assert_return(session_id_valid(name) == !(flags & GRDEV_SESSION_CUSTOM), -EINVAL);
assert_return(!(flags & GRDEV_SESSION_CUSTOM) || !(flags & GRDEV_SESSION_MANAGED), -EINVAL);
assert_return(!(flags & GRDEV_SESSION_MANAGED) || context->sysbus, -EINVAL);
session = new0(grdev_session, 1);
if (!session)
return -ENOMEM;
session->context = grdev_context_ref(context);
session->custom = flags & GRDEV_SESSION_CUSTOM;
session->managed = flags & GRDEV_SESSION_MANAGED;
session->event_fn = event_fn;
session->userdata = userdata;
session->name = strdup(name);
if (!session->name)
return -ENOMEM;
if (session->managed) {
r = sd_bus_path_encode("/org/freedesktop/login1/session",
session->name, &session->path);
if (r < 0)
return r;
}
session->card_map = hashmap_new(&string_hash_ops);
if (!session->card_map)
return -ENOMEM;
session->display_map = hashmap_new(&string_hash_ops);
if (!session->display_map)
return -ENOMEM;
r = hashmap_put(context->session_map, session->name, session);
if (r < 0)
return r;
*out = session;
session = NULL;
return 0;
}
grdev_session *grdev_session_free(grdev_session *session) {
grdev_card *card;
if (!session)
return NULL;
grdev_session_disable(session);
while ((card = hashmap_first(session->card_map)))
session_remove_card(session, card);
assert(hashmap_size(session->display_map) == 0);
if (session->name)
hashmap_remove_value(session->context->session_map, session->name, session);
hashmap_free(session->display_map);
hashmap_free(session->card_map);
session->context = grdev_context_unref(session->context);
free(session->path);
free(session->name);
free(session);
return NULL;
}
bool grdev_session_is_enabled(grdev_session *session) {
return session && session->enabled;
}
void grdev_session_enable(grdev_session *session) {
grdev_card *card;
Iterator iter;
assert(session);
if (!session->enabled) {
session->enabled = true;
HASHMAP_FOREACH(card, session->card_map, iter)
grdev_card_enable(card);
}
}
void grdev_session_disable(grdev_session *session) {
grdev_card *card;
Iterator iter;
assert(session);
if (session->enabled) {
session->enabled = false;
HASHMAP_FOREACH(card, session->card_map, iter)
grdev_card_disable(card);
}
}
void grdev_session_commit(grdev_session *session) {
grdev_card *card;
Iterator iter;
assert(session);
if (!session->enabled)
return;
HASHMAP_FOREACH(card, session->card_map, iter)
if (card->vtable->commit)
card->vtable->commit(card);
}
void grdev_session_restore(grdev_session *session) {
grdev_card *card;
Iterator iter;
assert(session);
if (!session->enabled)
return;
HASHMAP_FOREACH(card, session->card_map, iter)
if (card->vtable->restore)
card->vtable->restore(card);
}
void grdev_session_add_drm(grdev_session *session, struct udev_device *ud) {
grdev_card *card;
dev_t devnum;
int r;
assert(session);
assert(ud);
devnum = udev_device_get_devnum(ud);
if (devnum == 0)
return grdev_session_hotplug_drm(session, ud);
card = grdev_find_drm_card(session, devnum);
if (card)
return;
r = grdev_drm_card_new(&card, session, ud);
if (r < 0) {
log_debug_errno(r, "grdev: %s: cannot add DRM device for %s: %m",
session->name, udev_device_get_syspath(ud));
return;
}
session_add_card(session, card);
}
void grdev_session_remove_drm(grdev_session *session, struct udev_device *ud) {
grdev_card *card;
dev_t devnum;
assert(session);
assert(ud);
devnum = udev_device_get_devnum(ud);
if (devnum == 0)
return grdev_session_hotplug_drm(session, ud);
card = grdev_find_drm_card(session, devnum);
if (!card)
return;
session_remove_card(session, card);
}
void grdev_session_hotplug_drm(grdev_session *session, struct udev_device *ud) {
grdev_card *card = NULL;
struct udev_device *p;
dev_t devnum;
assert(session);
assert(ud);
for (p = ud; p; p = udev_device_get_parent_with_subsystem_devtype(p, "drm", NULL)) {
devnum = udev_device_get_devnum(ud);
if (devnum == 0)
continue;
card = grdev_find_drm_card(session, devnum);
if (card)
break;
}
if (!card)
return;
grdev_drm_card_hotplug(card, ud);
}
static void session_configure(grdev_session *session) {
grdev_display *display;
grdev_tile *tile;
grdev_card *card;
grdev_pipe *pipe;
Iterator i, j;
int r;
assert(session);
/*
* Whenever backends add or remove pipes, we set session->modified and
* require them to pin the session while modifying it. On release, we
* reconfigure the device and re-assign displays to all modified pipes.
*
* So far, we configure each pipe as a separate display. We do not
* support user-configuration, nor have we gotten any reports from
* users with multi-pipe monitors (4k on DP-1.2 MST and so on). Until
* we get reports, we keep the logic to a minimum.
*/
/* create new displays for all unconfigured pipes */
HASHMAP_FOREACH(card, session->card_map, i) {
if (!card->modified)
continue;
card->modified = false;
HASHMAP_FOREACH(pipe, card->pipe_map, j) {
tile = pipe->tile;
if (tile->display)
continue;
assert(!tile->parent);
display = grdev_find_display(session, pipe->name);
if (display && display->tile) {
log_debug("grdev: %s/%s: occupied display for pipe %s",
session->name, card->name, pipe->name);
continue;
} else if (!display) {
r = grdev_display_new(&display, session, pipe->name);
if (r < 0) {
log_debug_errno(r, "grdev: %s/%s: cannot create display for pipe %s: %m",
session->name, card->name, pipe->name);
continue;
}
}
tile_link(pipe->tile, display->tile);
}
}
/* update displays */
HASHMAP_FOREACH(display, session->display_map, i)
session_change_display(session, display);
}
grdev_session *grdev_session_pin(grdev_session *session) {
assert(session);
++session->n_pins;
return session;
}
grdev_session *grdev_session_unpin(grdev_session *session) {
if (!session)
return NULL;
assert(session->n_pins > 0);
if (--session->n_pins == 0)
session_configure(session);
return NULL;
}
/*
* Contexts
*/
int grdev_context_new(grdev_context **out, sd_event *event, sd_bus *sysbus) {
_cleanup_(grdev_context_unrefp) grdev_context *context = NULL;
assert_return(out, -EINVAL);
assert_return(event, -EINVAL);
context = new0(grdev_context, 1);
if (!context)
return -ENOMEM;
context->ref = 1;
context->event = sd_event_ref(event);
if (sysbus)
context->sysbus = sd_bus_ref(sysbus);
context->session_map = hashmap_new(&string_hash_ops);
if (!context->session_map)
return -ENOMEM;
*out = context;
context = NULL;
return 0;
}
static void context_cleanup(grdev_context *context) {
assert(hashmap_size(context->session_map) == 0);
hashmap_free(context->session_map);
context->sysbus = sd_bus_unref(context->sysbus);
context->event = sd_event_unref(context->event);
free(context);
}
grdev_context *grdev_context_ref(grdev_context *context) {
assert_return(context, NULL);
assert_return(context->ref > 0, NULL);
++context->ref;
return context;
}
grdev_context *grdev_context_unref(grdev_context *context) {
if (!context)
return NULL;
assert_return(context->ref > 0, NULL);
if (--context->ref == 0)
context_cleanup(context);
return NULL;
}