HelenOS sources
This source file includes following definitions.
- display_open
- display_callback_create
- display_close
- display_lock
- display_unlock
- display_wnd_params_init
- display_window_create
- display_window_destroy
- display_window_get_gc
- display_window_move_req
- display_window_move
- display_window_get_pos
- display_window_get_max_rect
- display_window_resize_req
- display_window_resize
- display_window_minimize
- display_window_maximize
- display_window_unmaximize
- display_window_set_cursor
- display_window_set_caption
- display_get_event
- display_get_info
- display_ev_pending
- display_cb_conn
- display_get_window
#include <async.h>
#include <display.h>
#include <display/event.h>
#include <errno.h>
#include <fibril_synch.h>
#include <ipc/display.h>
#include <ipc/services.h>
#include <ipcgfx/client.h>
#include <loc.h>
#include <mem.h>
#include <stdlib.h>
#include <str.h>
#include "../private/display.h"
#include "../private/params.h"
static errno_t display_callback_create(display_t *);
static void display_cb_conn(ipc_call_t *, void *);
static errno_t display_get_window(display_t *, sysarg_t, display_window_t **);
errno_t display_open(const char *dsname, display_t **rdisplay)
{
service_id_t display_svc;
display_t *display;
errno_t rc;
display = calloc(1, sizeof(display_t));
if (display == NULL)
return ENOMEM;
fibril_mutex_initialize(&display->lock);
fibril_condvar_initialize(&display->cv);
list_initialize(&display->windows);
if (dsname == NULL)
dsname = SERVICE_NAME_DISPLAY;
rc = loc_service_get_id(dsname, &display_svc, 0);
if (rc != EOK) {
free(display);
return ENOENT;
}
display->sess = loc_service_connect(display_svc, INTERFACE_DISPLAY,
0);
if (display->sess == NULL) {
free(display);
return ENOENT;
}
rc = display_callback_create(display);
if (rc != EOK) {
async_hangup(display->sess);
free(display);
return EIO;
}
*rdisplay = display;
return EOK;
}
static errno_t display_callback_create(display_t *display)
{
async_exch_t *exch = async_exchange_begin(display->sess);
aid_t req = async_send_0(exch, DISPLAY_CALLBACK_CREATE, NULL);
port_id_t port;
errno_t rc = async_create_callback_port(exch, INTERFACE_DISPLAY_CB, 0, 0,
display_cb_conn, display, &port);
async_exchange_end(exch);
if (rc != EOK)
return rc;
errno_t retval;
async_wait_for(req, &retval);
return retval;
}
void display_close(display_t *display)
{
fibril_mutex_lock(&display->lock);
async_hangup(display->sess);
display->sess = NULL;
while (!display->cb_done)
fibril_condvar_wait(&display->cv, &display->lock);
fibril_mutex_unlock(&display->lock);
free(display);
}
void display_lock(display_t *display)
{
fibril_mutex_lock(&display->lock);
}
void display_unlock(display_t *display)
{
fibril_mutex_unlock(&display->lock);
}
void display_wnd_params_init(display_wnd_params_t *params)
{
memset(params, 0, sizeof(*params));
params->caption = "";
}
errno_t display_window_create(display_t *display, display_wnd_params_t *params,
display_wnd_cb_t *cb, void *cb_arg, display_window_t **rwindow)
{
display_window_t *window;
display_wnd_params_enc_t eparams;
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
eparams.rect = params->rect;
eparams.caption_size = str_size(params->caption);
eparams.min_size = params->min_size;
eparams.pos = params->pos;
eparams.flags = params->flags;
eparams.idev_id = params->idev_id;
window = calloc(1, sizeof(display_window_t));
if (window == NULL)
return ENOMEM;
exch = async_exchange_begin(display->sess);
req = async_send_0(exch, DISPLAY_WINDOW_CREATE, &answer);
rc = async_data_write_start(exch, &eparams,
sizeof (display_wnd_params_enc_t));
if (rc != EOK) {
async_exchange_end(exch);
async_forget(req);
free(window);
return rc;
}
rc = async_data_write_start(exch, params->caption,
eparams.caption_size);
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
free(window);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK) {
free(window);
return rc;
}
window->display = display;
window->id = ipc_get_arg1(&answer);
window->cb = cb;
window->cb_arg = cb_arg;
list_append(&window->lwindows, &display->windows);
*rwindow = window;
return EOK;
}
errno_t display_window_destroy(display_window_t *window)
{
async_exch_t *exch;
errno_t rc;
if (window == NULL)
return EOK;
exch = async_exchange_begin(window->display->sess);
rc = async_req_1_0(exch, DISPLAY_WINDOW_DESTROY, window->id);
async_exchange_end(exch);
list_remove(&window->lwindows);
free(window);
return rc;
}
errno_t display_window_get_gc(display_window_t *window, gfx_context_t **rgc)
{
async_sess_t *sess;
async_exch_t *exch;
ipc_gc_t *gc;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
sess = async_connect_me_to(exch, INTERFACE_GC, 0, window->id, &rc);
if (sess == NULL) {
async_exchange_end(exch);
return rc;
}
async_exchange_end(exch);
rc = ipc_gc_create(sess, &gc);
if (rc != EOK) {
async_hangup(sess);
return ENOMEM;
}
*rgc = ipc_gc_get_ctx(gc);
return EOK;
}
errno_t display_window_move_req(display_window_t *window, gfx_coord2_t *pos,
sysarg_t pos_id)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
req = async_send_2(exch, DISPLAY_WINDOW_MOVE_REQ, window->id,
pos_id, &answer);
rc = async_data_write_start(exch, (void *)pos, sizeof (gfx_coord2_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_move(display_window_t *window, gfx_coord2_t *dpos)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
req = async_send_1(exch, DISPLAY_WINDOW_MOVE, window->id, &answer);
rc = async_data_write_start(exch, dpos, sizeof (gfx_coord2_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_get_pos(display_window_t *window, gfx_coord2_t *dpos)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
req = async_send_1(exch, DISPLAY_WINDOW_GET_POS, window->id, &answer);
rc = async_data_read_start(exch, dpos, sizeof (gfx_coord2_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_get_max_rect(display_window_t *window, gfx_rect_t *rect)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
req = async_send_1(exch, DISPLAY_WINDOW_GET_MAX_RECT, window->id,
&answer);
rc = async_data_read_start(exch, rect, sizeof (gfx_rect_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_resize_req(display_window_t *window,
display_wnd_rsztype_t rsztype, gfx_coord2_t *pos, sysarg_t pos_id)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
req = async_send_3(exch, DISPLAY_WINDOW_RESIZE_REQ, window->id,
(sysarg_t) rsztype, pos_id, &answer);
rc = async_data_write_start(exch, (void *)pos, sizeof (gfx_coord2_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_resize(display_window_t *window, gfx_coord2_t *offs,
gfx_rect_t *nrect)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
display_wnd_resize_t wresize;
errno_t rc;
wresize.offs = *offs;
wresize.nrect = *nrect;
exch = async_exchange_begin(window->display->sess);
req = async_send_1(exch, DISPLAY_WINDOW_RESIZE, window->id, &answer);
rc = async_data_write_start(exch, &wresize, sizeof (display_wnd_resize_t));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
errno_t display_window_minimize(display_window_t *window)
{
async_exch_t *exch;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
rc = async_req_1_0(exch, DISPLAY_WINDOW_MINIMIZE, window->id);
async_exchange_end(exch);
return rc;
}
errno_t display_window_maximize(display_window_t *window)
{
async_exch_t *exch;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
rc = async_req_1_0(exch, DISPLAY_WINDOW_MAXIMIZE, window->id);
async_exchange_end(exch);
return rc;
}
errno_t display_window_unmaximize(display_window_t *window)
{
async_exch_t *exch;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
rc = async_req_1_0(exch, DISPLAY_WINDOW_UNMAXIMIZE, window->id);
async_exchange_end(exch);
return rc;
}
errno_t display_window_set_cursor(display_window_t *window,
display_stock_cursor_t cursor)
{
async_exch_t *exch;
errno_t rc;
exch = async_exchange_begin(window->display->sess);
rc = async_req_2_0(exch, DISPLAY_WINDOW_SET_CURSOR, window->id,
cursor);
async_exchange_end(exch);
return rc;
}
errno_t display_window_set_caption(display_window_t *window,
const char *caption)
{
async_exch_t *exch;
aid_t req;
ipc_call_t answer;
size_t cap_size;
errno_t rc;
cap_size = str_size(caption);
exch = async_exchange_begin(window->display->sess);
req = async_send_1(exch, DISPLAY_WINDOW_SET_CAPTION, window->id,
&answer);
rc = async_data_write_start(exch, caption, cap_size);
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
return rc;
}
static errno_t display_get_event(display_t *display, display_window_t **rwindow,
display_wnd_ev_t *event)
{
async_exch_t *exch;
ipc_call_t answer;
aid_t req;
errno_t rc;
sysarg_t wnd_id;
display_window_t *window;
exch = async_exchange_begin(display->sess);
req = async_send_0(exch, DISPLAY_GET_EVENT, &answer);
rc = async_data_read_start(exch, event, sizeof(*event));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
wnd_id = ipc_get_arg1(&answer);
rc = display_get_window(display, wnd_id, &window);
if (rc != EOK)
return EIO;
*rwindow = window;
return EOK;
}
errno_t display_get_info(display_t *display, display_info_t *info)
{
async_exch_t *exch;
ipc_call_t answer;
aid_t req;
errno_t rc;
exch = async_exchange_begin(display->sess);
req = async_send_0(exch, DISPLAY_GET_INFO, &answer);
rc = async_data_read_start(exch, info, sizeof(*info));
async_exchange_end(exch);
if (rc != EOK) {
async_forget(req);
return rc;
}
async_wait_for(req, &rc);
if (rc != EOK)
return rc;
return EOK;
}
static void display_ev_pending(display_t *display, ipc_call_t *icall)
{
errno_t rc;
display_window_t *window = NULL;
display_wnd_ev_t event;
display_lock(display);
while (true) {
if (display->sess != NULL)
rc = display_get_event(display, &window, &event);
else
rc = ENOENT;
if (rc != EOK)
break;
switch (event.etype) {
case wev_close:
if (window->cb != NULL && window->cb->close_event != NULL) {
window->cb->close_event(window->cb_arg);
}
break;
case wev_focus:
if (window->cb != NULL && window->cb->focus_event != NULL) {
window->cb->focus_event(window->cb_arg,
event.ev.focus.nfocus);
}
break;
case wev_kbd:
if (window->cb != NULL && window->cb->kbd_event != NULL) {
window->cb->kbd_event(window->cb_arg,
&event.ev.kbd);
}
break;
case wev_pos:
if (window->cb != NULL && window->cb->pos_event != NULL) {
window->cb->pos_event(window->cb_arg,
&event.ev.pos);
}
break;
case wev_resize:
if (window->cb != NULL && window->cb->resize_event != NULL) {
window->cb->resize_event(window->cb_arg,
&event.ev.resize.rect);
}
break;
case wev_unfocus:
if (window->cb != NULL && window->cb->unfocus_event != NULL) {
window->cb->unfocus_event(window->cb_arg,
event.ev.unfocus.nfocus);
}
break;
}
}
display_unlock(display);
async_answer_0(icall, EOK);
}
static void display_cb_conn(ipc_call_t *icall, void *arg)
{
display_t *display = (display_t *) arg;
while (true) {
ipc_call_t call;
async_get_call(&call);
if (!ipc_get_imethod(&call)) {
async_answer_0(&call, EOK);
goto out;
}
switch (ipc_get_imethod(&call)) {
case DISPLAY_EV_PENDING:
display_ev_pending(display, &call);
break;
default:
async_answer_0(&call, ENOTSUP);
break;
}
}
out:
fibril_mutex_lock(&display->lock);
display->cb_done = true;
fibril_mutex_unlock(&display->lock);
fibril_condvar_broadcast(&display->cv);
}
static errno_t display_get_window(display_t *display, sysarg_t wnd_id,
display_window_t **rwindow)
{
link_t *link;
display_window_t *window;
link = list_first(&display->windows);
while (link != NULL) {
window = list_get_instance(link, display_window_t, lwindows);
if (window->id == wnd_id) {
*rwindow = window;
return EOK;
}
link = list_next(link, &display->windows);
}
return ENOENT;
}
HelenOS homepage, sources at GitHub