This still needs all the bells and whistles from the egl-kms mesa branch, but it makes it a lot easier to work on wayland.dev
parent
fc783d4071
commit
ce5325d3ed
@ -0,0 +1,666 @@ |
||||
/*
|
||||
* Copyright © 2008-2010 Kristian Høgsberg |
||||
* |
||||
* This program is free software; you can redistribute it and/or modify |
||||
* it under the terms of the GNU General Public License as published by |
||||
* the Free Software Foundation; either version 2 of the License, or |
||||
* (at your option) any later version. |
||||
* |
||||
* This program 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 General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU General Public License |
||||
* along with this program; if not, write to the Free Software Foundation, |
||||
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. |
||||
*/ |
||||
|
||||
#include <stddef.h> |
||||
#include <stdio.h> |
||||
#include <stdlib.h> |
||||
#include <string.h> |
||||
#include <fcntl.h> |
||||
#include <unistd.h> |
||||
#include <errno.h> |
||||
#include <sys/time.h> |
||||
|
||||
#include <xcb/xcb.h> |
||||
#include <xcb/dri2.h> |
||||
#include <xcb/xfixes.h> |
||||
|
||||
#define GL_GLEXT_PROTOTYPES |
||||
#define EGL_EGLEXT_PROTOTYPES |
||||
#include <GLES2/gl2.h> |
||||
#include <GLES2/gl2ext.h> |
||||
#include <EGL/egl.h> |
||||
#include <EGL/eglext.h> |
||||
|
||||
#include "wayland.h" |
||||
#include "wayland-protocol.h" |
||||
#include "compositor.h" |
||||
|
||||
#define ARRAY_SIZE(array) (sizeof(array) / sizeof(array[0])) |
||||
|
||||
struct x11_compositor { |
||||
struct wlsc_compositor base; |
||||
|
||||
xcb_connection_t *conn; |
||||
xcb_screen_t *screen; |
||||
xcb_cursor_t null_cursor; |
||||
int dri2_major; |
||||
int dri2_minor; |
||||
int drm_fd; |
||||
struct wl_event_source *xcb_source; |
||||
struct { |
||||
xcb_atom_t wm_protocols; |
||||
xcb_atom_t wm_normal_hints; |
||||
xcb_atom_t wm_size_hints; |
||||
xcb_atom_t wm_delete_window; |
||||
xcb_atom_t net_wm_name; |
||||
xcb_atom_t utf8_string; |
||||
} atom; |
||||
}; |
||||
|
||||
struct x11_output { |
||||
struct wlsc_output base; |
||||
|
||||
xcb_xfixes_region_t region; |
||||
xcb_window_t window; |
||||
GLuint rbo; |
||||
EGLImageKHR image; |
||||
xcb_rectangle_t damage[16]; |
||||
int damage_count; |
||||
}; |
||||
|
||||
struct x11_input { |
||||
struct wlsc_input_device base; |
||||
}; |
||||
|
||||
|
||||
static void |
||||
x11_input_create(struct x11_compositor *c) |
||||
{ |
||||
struct x11_input *input; |
||||
|
||||
input = malloc(sizeof *input); |
||||
if (input == NULL) |
||||
return; |
||||
|
||||
memset(input, 0, sizeof *input); |
||||
wlsc_input_device_init(&input->base, &c->base); |
||||
|
||||
c->base.input_device = &input->base; |
||||
} |
||||
|
||||
|
||||
static int |
||||
dri2_connect(struct x11_compositor *c) |
||||
{ |
||||
xcb_xfixes_query_version_reply_t *xfixes_query; |
||||
xcb_xfixes_query_version_cookie_t xfixes_query_cookie; |
||||
xcb_dri2_query_version_reply_t *dri2_query; |
||||
xcb_dri2_query_version_cookie_t dri2_query_cookie; |
||||
xcb_dri2_connect_reply_t *connect; |
||||
xcb_dri2_connect_cookie_t connect_cookie; |
||||
xcb_generic_error_t *error; |
||||
|
||||
xcb_prefetch_extension_data (c->conn, &xcb_xfixes_id); |
||||
xcb_prefetch_extension_data (c->conn, &xcb_dri2_id); |
||||
|
||||
xfixes_query_cookie = |
||||
xcb_xfixes_query_version(c->conn, |
||||
XCB_XFIXES_MAJOR_VERSION, |
||||
XCB_XFIXES_MINOR_VERSION); |
||||
|
||||
dri2_query_cookie = |
||||
xcb_dri2_query_version (c->conn, |
||||
XCB_DRI2_MAJOR_VERSION, |
||||
XCB_DRI2_MINOR_VERSION); |
||||
|
||||
connect_cookie = xcb_dri2_connect_unchecked (c->conn, |
||||
c->screen->root, |
||||
XCB_DRI2_DRIVER_TYPE_DRI); |
||||
|
||||
xfixes_query = |
||||
xcb_xfixes_query_version_reply (c->conn, |
||||
xfixes_query_cookie, &error); |
||||
if (xfixes_query == NULL || |
||||
error != NULL || xfixes_query->major_version < 2) { |
||||
free(error); |
||||
return -1; |
||||
} |
||||
free(xfixes_query); |
||||
|
||||
dri2_query = |
||||
xcb_dri2_query_version_reply (c->conn, |
||||
dri2_query_cookie, &error); |
||||
if (dri2_query == NULL || error != NULL) { |
||||
fprintf(stderr, "DRI2: failed to query version"); |
||||
free(error); |
||||
return EGL_FALSE; |
||||
} |
||||
c->dri2_major = dri2_query->major_version; |
||||
c->dri2_minor = dri2_query->minor_version; |
||||
free(dri2_query); |
||||
|
||||
connect = xcb_dri2_connect_reply (c->conn, |
||||
connect_cookie, NULL); |
||||
if (connect == NULL || |
||||
connect->driver_name_length + connect->device_name_length == 0) { |
||||
fprintf(stderr, "DRI2: failed to authenticate"); |
||||
return -1; |
||||
} |
||||
|
||||
c->base.base.device = |
||||
strndup(xcb_dri2_connect_device_name (connect), |
||||
xcb_dri2_connect_device_name_length (connect)); |
||||
|
||||
if (c->base.base.device == NULL) { |
||||
free(connect); |
||||
return -1; |
||||
} |
||||
free(connect); |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int |
||||
dri2_authenticate(struct x11_compositor *c) |
||||
{ |
||||
xcb_dri2_authenticate_reply_t *authenticate; |
||||
xcb_dri2_authenticate_cookie_t authenticate_cookie; |
||||
drm_magic_t magic; |
||||
|
||||
if (drmGetMagic(c->drm_fd, &magic)) { |
||||
fprintf(stderr, "DRI2: failed to get drm magic"); |
||||
return -1; |
||||
} |
||||
|
||||
authenticate_cookie = |
||||
xcb_dri2_authenticate_unchecked(c->conn, |
||||
c->screen->root, magic); |
||||
authenticate = |
||||
xcb_dri2_authenticate_reply(c->conn, |
||||
authenticate_cookie, NULL); |
||||
if (authenticate == NULL || !authenticate->authenticated) { |
||||
fprintf(stderr, "DRI2: failed to authenticate"); |
||||
free(authenticate); |
||||
return -1; |
||||
} |
||||
|
||||
free(authenticate); |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int |
||||
x11_compositor_init_egl(struct x11_compositor *c) |
||||
{ |
||||
PFNEGLGETTYPEDDISPLAYMESA get_typed_display_mesa; |
||||
EGLint major, minor, count; |
||||
EGLConfig config; |
||||
|
||||
static const EGLint config_attribs[] = { |
||||
EGL_SURFACE_TYPE, 0, |
||||
EGL_NO_SURFACE_CAPABLE_MESA, EGL_OPENGL_BIT, |
||||
EGL_RENDERABLE_TYPE, EGL_OPENGL_BIT, |
||||
EGL_NONE |
||||
}; |
||||
|
||||
if (dri2_connect(c) < 0) |
||||
return -1; |
||||
|
||||
c->drm_fd = open(c->base.base.device, O_RDWR); |
||||
if (c->drm_fd == -1) { |
||||
fprintf(stderr, |
||||
"DRI2: could not open %s (%s)", c->base.base.device, |
||||
strerror(errno)); |
||||
return -1; |
||||
} |
||||
|
||||
if (dri2_authenticate(c) < 0) |
||||
return -1; |
||||
|
||||
get_typed_display_mesa = |
||||
(PFNEGLGETTYPEDDISPLAYMESA) |
||||
eglGetProcAddress("eglGetTypedDisplayMESA"); |
||||
if (get_typed_display_mesa == NULL) { |
||||
fprintf(stderr, "eglGetTypedDisplayMESA() not found\n"); |
||||
return -1; |
||||
} |
||||
|
||||
c->base.display = get_typed_display_mesa(EGL_DRM_DISPLAY_TYPE_MESA, |
||||
(void *) c->drm_fd); |
||||
if (c->base.display == NULL) { |
||||
fprintf(stderr, "failed to create display\n"); |
||||
return -1; |
||||
} |
||||
|
||||
if (!eglInitialize(c->base.display, &major, &minor)) { |
||||
fprintf(stderr, "failed to initialize display\n"); |
||||
return -1; |
||||
} |
||||
|
||||
if (!eglChooseConfig(c->base.display, |
||||
config_attribs, &config, 1, &count) || |
||||
count == 0) { |
||||
fprintf(stderr, "eglChooseConfig() failed\n"); |
||||
return -1; |
||||
} |
||||
|
||||
eglBindAPI(EGL_OPENGL_API); |
||||
c->base.context = eglCreateContext(c->base.display, |
||||
config, EGL_NO_CONTEXT, NULL); |
||||
if (c->base.context == NULL) { |
||||
fprintf(stderr, "failed to create context\n"); |
||||
return -1; |
||||
} |
||||
|
||||
if (!eglMakeCurrent(c->base.display, EGL_NO_SURFACE, |
||||
EGL_NO_SURFACE, c->base.context)) { |
||||
fprintf(stderr, "failed to make context current\n"); |
||||
return -1; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static void |
||||
x11_compositor_present(struct wlsc_compositor *base) |
||||
{ |
||||
struct x11_compositor *c = (struct x11_compositor *) base; |
||||
struct x11_output *output; |
||||
xcb_dri2_copy_region_cookie_t cookie; |
||||
struct timeval tv; |
||||
uint32_t msec; |
||||
|
||||
glFlush(); |
||||
|
||||
wl_list_for_each(output, &c->base.output_list, base.link) { |
||||
cookie = xcb_dri2_copy_region_unchecked(c->conn, |
||||
output->window, |
||||
output->region, |
||||
XCB_DRI2_ATTACHMENT_BUFFER_FRONT_LEFT, |
||||
XCB_DRI2_ATTACHMENT_BUFFER_BACK_LEFT); |
||||
free(xcb_dri2_copy_region_reply(c->conn, cookie, NULL)); |
||||
}
|
||||
|
||||
gettimeofday(&tv, NULL); |
||||
msec = tv.tv_sec * 1000 + tv.tv_usec / 1000; |
||||
wlsc_compositor_finish_frame(&c->base, msec); |
||||
} |
||||
|
||||
static void |
||||
x11_output_set_wm_protocols(struct x11_output *output) |
||||
{ |
||||
xcb_atom_t list[1]; |
||||
struct x11_compositor *c = |
||||
(struct x11_compositor *) output->base.compositor; |
||||
|
||||
list[0] = c->atom.wm_delete_window; |
||||
xcb_change_property (c->conn,
|
||||
XCB_PROP_MODE_REPLACE, |
||||
output->window, |
||||
c->atom.wm_protocols, |
||||
XCB_ATOM_ATOM, |
||||
32, |
||||
ARRAY_SIZE(list), |
||||
list); |
||||
} |
||||
|
||||
struct wm_normal_hints { |
||||
uint32_t flags; |
||||
uint32_t pad[4]; |
||||
int32_t min_width, min_height; |
||||
int32_t max_width, max_height; |
||||
int32_t width_inc, height_inc; |
||||
int32_t min_aspect_x, min_aspect_y; |
||||
int32_t max_aspect_x, max_aspect_y; |
||||
int32_t base_width, base_height; |
||||
int32_t win_gravity; |
||||
}; |
||||
|
||||
#define WM_NORMAL_HINTS_MIN_SIZE 16 |
||||
#define WM_NORMAL_HINTS_MAX_SIZE 32 |
||||
|
||||
static int |
||||
x11_compositor_create_output(struct x11_compositor *c, int width, int height) |
||||
{ |
||||
static const char name[] = "Wayland Compositor"; |
||||
struct x11_output *output; |
||||
xcb_dri2_dri2_buffer_t *buffers; |
||||
xcb_dri2_get_buffers_reply_t *reply; |
||||
xcb_dri2_get_buffers_cookie_t cookie; |
||||
xcb_screen_iterator_t iter; |
||||
xcb_rectangle_t rectangle; |
||||
struct wm_normal_hints normal_hints; |
||||
unsigned int attachments[] = |
||||
{ XCB_DRI2_ATTACHMENT_BUFFER_BACK_LEFT}; |
||||
uint32_t mask = XCB_CW_EVENT_MASK | XCB_CW_CURSOR; |
||||
uint32_t values[2] = {
|
||||
XCB_EVENT_MASK_KEY_PRESS | |
||||
XCB_EVENT_MASK_KEY_RELEASE | |
||||
XCB_EVENT_MASK_BUTTON_PRESS | |
||||
XCB_EVENT_MASK_BUTTON_RELEASE | |
||||
XCB_EVENT_MASK_POINTER_MOTION | |
||||
XCB_EVENT_MASK_EXPOSURE | |
||||
XCB_EVENT_MASK_STRUCTURE_NOTIFY, |
||||
0 |
||||
}; |
||||
|
||||
EGLint attribs[] = { |
||||
EGL_WIDTH, 0, |
||||
EGL_HEIGHT, 0, |
||||
EGL_IMAGE_STRIDE_MESA, 0, |
||||
EGL_IMAGE_FORMAT_MESA, EGL_IMAGE_FORMAT_ARGB8888_MESA, |
||||
EGL_NONE |
||||
}; |
||||
|
||||
output = malloc(sizeof *output); |
||||
if (output == NULL) |
||||
return -1; |
||||
|
||||
memset(output, 0, sizeof *output); |
||||
wlsc_output_init(&output->base, &c->base, 0, 0, width, height); |
||||
|
||||
values[1] = c->null_cursor; |
||||
output->window = xcb_generate_id(c->conn); |
||||
iter = xcb_setup_roots_iterator(xcb_get_setup(c->conn)); |
||||
xcb_create_window(c->conn, |
||||
XCB_COPY_FROM_PARENT, |
||||
output->window, |
||||
iter.data->root, |
||||
0, 0, |
||||
width, height, |
||||
0, |
||||
XCB_WINDOW_CLASS_INPUT_OUTPUT, |
||||
iter.data->root_visual, |
||||
mask, values); |
||||
|
||||
/* Don't resize me. */ |
||||
memset(&normal_hints, 0, sizeof normal_hints); |
||||
normal_hints.flags = |
||||
WM_NORMAL_HINTS_MAX_SIZE | WM_NORMAL_HINTS_MIN_SIZE; |
||||
normal_hints.min_width = width; |
||||
normal_hints.min_height = height; |
||||
normal_hints.max_width = width; |
||||
normal_hints.max_height = height; |
||||
xcb_change_property (c->conn, XCB_PROP_MODE_REPLACE, output->window, |
||||
c->atom.wm_normal_hints, |
||||
c->atom.wm_size_hints, 32, |
||||
sizeof normal_hints / 4, |
||||
(uint8_t *) &normal_hints); |
||||
|
||||
xcb_map_window(c->conn, output->window); |
||||
|
||||
/* Set window name. Don't bother with non-EWMH WMs. */ |
||||
xcb_change_property(c->conn, XCB_PROP_MODE_REPLACE, output->window, |
||||
c->atom.net_wm_name, c->atom.utf8_string, 8, |
||||
strlen(name), name); |
||||
|
||||
rectangle.x = 0; |
||||
rectangle.y = 0; |
||||
rectangle.width = width; |
||||
rectangle.height = height; |
||||
output->region = xcb_generate_id(c->conn); |
||||
xcb_xfixes_create_region(c->conn, output->region, 1, &rectangle); |
||||
|
||||
xcb_dri2_create_drawable (c->conn, output->window); |
||||
|
||||
x11_output_set_wm_protocols(output); |
||||
|
||||
cookie = xcb_dri2_get_buffers_unchecked (c->conn, |
||||
output->window, |
||||
1, 1, attachments); |
||||
reply = xcb_dri2_get_buffers_reply (c->conn, cookie, NULL); |
||||
if (reply == NULL) |
||||
return -1; |
||||
buffers = xcb_dri2_get_buffers_buffers (reply); |
||||
if (buffers == NULL) |
||||
return -1; |
||||
|
||||
if (reply->count != 1) { |
||||
fprintf(stderr, |
||||
"got wrong number of buffers (%d)\n", reply->count); |
||||
return -1; |
||||
} |
||||
|
||||
attribs[1] = reply->width; |
||||
attribs[3] = reply->height; |
||||
attribs[5] = buffers[0].pitch / 4; |
||||
free(reply); |
||||
|
||||
output->image = |
||||
eglCreateImageKHR(c->base.display, c->base.context, |
||||
EGL_DRM_IMAGE_MESA, |
||||
(EGLClientBuffer) buffers[0].name, |
||||
attribs); |
||||
|
||||
glGenRenderbuffers(1, &output->rbo); |
||||
glBindRenderbuffer(GL_RENDERBUFFER, output->rbo); |
||||
|
||||
glEGLImageTargetRenderbufferStorageOES(GL_RENDERBUFFER, |
||||
output->image); |
||||
|
||||
glFramebufferRenderbuffer(GL_FRAMEBUFFER, |
||||
GL_COLOR_ATTACHMENT0, |
||||
GL_RENDERBUFFER, |
||||
output->rbo); |
||||
|
||||
wl_list_insert(c->base.output_list.prev, &output->base.link); |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static void |
||||
idle_repaint(void *data) |
||||
{ |
||||
struct x11_output *output = data; |
||||
struct x11_compositor *c = |
||||
(struct x11_compositor *) output->base.compositor; |
||||
xcb_xfixes_region_t region; |
||||
xcb_dri2_copy_region_cookie_t cookie; |
||||
|
||||
if (output->damage_count <= ARRAY_SIZE(output->damage)) { |
||||
region = xcb_generate_id(c->conn); |
||||
xcb_xfixes_create_region(c->conn, region, |
||||
output->damage_count, output->damage); |
||||
} else { |
||||
region = output->region; |
||||
} |
||||
|
||||
cookie = xcb_dri2_copy_region_unchecked(c->conn, |
||||
output->window, |
||||
region, |
||||
XCB_DRI2_ATTACHMENT_BUFFER_FRONT_LEFT, |
||||
XCB_DRI2_ATTACHMENT_BUFFER_BACK_LEFT); |
||||
|
||||
if (region != output->region) |
||||
xcb_xfixes_destroy_region(c->conn, region); |
||||
|
||||
free(xcb_dri2_copy_region_reply(c->conn, cookie, NULL)); |
||||
output->damage_count = 0; |
||||
} |
||||
|
||||
static struct x11_output * |
||||
x11_compositor_find_output(struct x11_compositor *c, xcb_window_t window) |
||||
{ |
||||
struct x11_output *output; |
||||
|
||||
wl_list_for_each(output, &c->base.output_list, base.link) { |
||||
if (output->window == window) |
||||
return output; |
||||
} |
||||
|
||||
return NULL; |
||||
} |
||||
|
||||
static void |
||||
x11_compositor_handle_event(int fd, uint32_t mask, void *data) |
||||
{ |
||||
struct x11_compositor *c = data; |
||||
struct x11_output *output; |
||||
xcb_generic_event_t *event; |
||||
struct wl_event_loop *loop; |
||||
xcb_client_message_event_t *client_message; |
||||
xcb_motion_notify_event_t *motion_notify; |
||||
xcb_key_press_event_t *key_press; |
||||
xcb_button_press_event_t *button_press; |
||||
xcb_expose_event_t *expose; |
||||
xcb_rectangle_t *r; |
||||
xcb_atom_t atom; |
||||
|
||||
loop = wl_display_get_event_loop(c->base.wl_display); |
||||
while (event = xcb_poll_for_event (c->conn), event != NULL) { |
||||
switch (event->response_type & ~0x80) { |
||||
|
||||
case XCB_KEY_PRESS: |
||||
key_press = (xcb_key_press_event_t *) event; |
||||
notify_key(c->base.input_device, |
||||
key_press->detail, 1); |
||||
|
||||
fprintf(stderr, "code %d, sequence %d\n", |
||||
key_press->detail, key_press->sequence); |
||||
|
||||
break; |
||||
case XCB_KEY_RELEASE: |
||||
key_press = (xcb_key_press_event_t *) event; |
||||
notify_key(c->base.input_device, |
||||
key_press->detail, 0); |
||||
break; |
||||
case XCB_BUTTON_PRESS: |
||||
button_press = (xcb_button_press_event_t *) event; |
||||
notify_button(c->base.input_device, |
||||
button_press->detail, 1); |
||||
break; |
||||
case XCB_BUTTON_RELEASE: |
||||
button_press = (xcb_button_press_event_t *) event; |
||||
notify_button(c->base.input_device, |
||||
button_press->detail, 0); |
||||
break; |
||||
|
||||
case XCB_MOTION_NOTIFY: |
||||
motion_notify = (xcb_motion_notify_event_t *) event; |
||||
notify_motion(c->base.input_device, |
||||
motion_notify->event_x, |
||||
motion_notify->event_y); |
||||
break; |
||||
|
||||
case XCB_EXPOSE: |
||||
expose = (xcb_expose_event_t *) event; |
||||
output = x11_compositor_find_output(c, expose->window); |
||||
if (output->damage_count == 0) |
||||
wl_event_loop_add_idle(loop, |
||||
idle_repaint, output); |
||||
|
||||
r = &output->damage[output->damage_count++]; |
||||
if (output->damage_count > 16) |
||||
break; |
||||
r->x = expose->x; |
||||
r->y = expose->y; |
||||
r->width = expose->width; |
||||
r->height = expose->height; |
||||
break; |
||||
case XCB_CLIENT_MESSAGE: |
||||
client_message = (xcb_client_message_event_t *) event; |
||||
atom = client_message->data.data32[0]; |
||||
if (atom == c->atom.wm_delete_window) |
||||
exit(1); |
||||
break; |
||||
default:
|
||||
|
||||
break; |
||||
} |
||||
|
||||
free (event); |
||||
} |
||||
} |
||||
|
||||
#define F(field) offsetof(struct x11_compositor, field) |
||||
|
||||
static void |
||||
x11_compositor_get_resources(struct x11_compositor *c) |
||||
{ |
||||
static const struct { const char *name; int offset; } atoms[] = { |
||||
{ "WM_PROTOCOLS", F(atom.wm_protocols) }, |
||||
{ "WM_NORMAL_HINTS", F(atom.wm_normal_hints) }, |
||||
{ "WM_SIZE_HINTS", F(atom.wm_size_hints) }, |
||||
{ "WM_DELETE_WINDOW", F(atom.wm_delete_window) }, |
||||
{ "_NET_WM_NAME", F(atom.net_wm_name) }, |
||||
{ "UTF8_STRING", F(atom.utf8_string) }, |
||||
}; |
||||
|
||||
xcb_intern_atom_cookie_t cookies[ARRAY_SIZE(atoms)]; |
||||
xcb_intern_atom_reply_t *reply; |
||||
xcb_pixmap_t pixmap; |
||||
xcb_gc_t gc; |
||||
int i; |
||||
uint8_t data[] = { 0, 0, 0, 0 }; |
||||
|
||||
for (i = 0; i < ARRAY_SIZE(atoms); i++) |
||||
cookies[i] = xcb_intern_atom (c->conn, 0, |
||||
strlen(atoms[i].name), |
||||
atoms[i].name); |
||||
|
||||
for (i = 0; i < ARRAY_SIZE(atoms); i++) { |
||||
reply = xcb_intern_atom_reply (c->conn, cookies[i], NULL); |
||||
*(xcb_atom_t *) ((char *) c + atoms[i].offset) = reply->atom; |
||||
free(reply); |
||||
} |
||||
|
||||
pixmap = xcb_generate_id(c->conn); |
||||
gc = xcb_generate_id(c->conn); |
||||
xcb_create_pixmap(c->conn, 1, pixmap, c->screen->root, 1, 1); |
||||
xcb_create_gc(c->conn, gc, pixmap, 0, NULL); |
||||
xcb_put_image(c->conn, XCB_IMAGE_FORMAT_XY_PIXMAP, |
||||
pixmap, gc, 1, 1, 0, 0, 0, 32, sizeof data, data); |
||||
c->null_cursor = xcb_generate_id(c->conn); |
||||
xcb_create_cursor (c->conn, c->null_cursor, |
||||
pixmap, pixmap, 0, 0, 0, 0, 0, 0, 1, 1); |
||||
xcb_free_gc(c->conn, gc); |
||||
xcb_free_pixmap(c->conn, pixmap); |
||||
} |
||||
|
||||
struct wlsc_compositor * |
||||
x11_compositor_create(struct wl_display *display) |
||||
{ |
||||
struct x11_compositor *c; |
||||
struct wl_event_loop *loop; |
||||
xcb_screen_iterator_t s; |
||||
|
||||
c = malloc(sizeof *c); |
||||
if (c == NULL) |
||||
return NULL; |
||||
|
||||
memset(c, 0, sizeof *c); |
||||
c->conn = xcb_connect(0, 0); |
||||
|
||||
if (xcb_connection_has_error(c->conn)) |
||||
return NULL; |
||||
|
||||
s = xcb_setup_roots_iterator(xcb_get_setup(c->conn)); |
||||
c->screen = s.data; |
||||
|
||||
x11_compositor_get_resources(c); |
||||
|
||||
x11_compositor_init_egl(c); |
||||
|
||||
/* Can't init base class until we have a current egl context */ |
||||
wlsc_compositor_init(&c->base, display); |
||||
|
||||
x11_compositor_create_output(c, 1024, 640); |
||||
|
||||
x11_input_create(c); |
||||
|
||||
loop = wl_display_get_event_loop(c->base.wl_display); |
||||
|
||||
c->xcb_source = |
||||
wl_event_loop_add_fd(loop, xcb_get_file_descriptor(c->conn), |
||||
WL_EVENT_READABLE, |
||||
x11_compositor_handle_event, c); |
||||
|
||||
c->base.present = x11_compositor_present; |
||||
|
||||
return &c->base; |
||||
} |
Loading…
Reference in new issue