Add wlcore/wl_shm (WIP)
This commit is contained in:
parent
f252c5a792
commit
2aafb5dd19
|
@ -61,6 +61,7 @@ add_subdirectory(backend)
|
||||||
add_subdirectory(types)
|
add_subdirectory(types)
|
||||||
add_subdirectory(session)
|
add_subdirectory(session)
|
||||||
add_subdirectory(render)
|
add_subdirectory(render)
|
||||||
|
add_subdirectory(wlcore)
|
||||||
add_subdirectory(util)
|
add_subdirectory(util)
|
||||||
|
|
||||||
add_subdirectory(example)
|
add_subdirectory(example)
|
||||||
|
|
|
@ -65,3 +65,16 @@ target_link_libraries(tablet
|
||||||
wlr-render
|
wlr-render
|
||||||
${XKBCOMMON_LIBRARIES}
|
${XKBCOMMON_LIBRARIES}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
add_executable(compositor
|
||||||
|
compositor
|
||||||
|
shared.c
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(compositor
|
||||||
|
wlr-backend
|
||||||
|
wlr-session
|
||||||
|
wlr-render
|
||||||
|
wlr-wlcore
|
||||||
|
${XKBCOMMON_LIBRARIES}
|
||||||
|
)
|
||||||
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
#define _POSIX_C_SOURCE 199309L
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <inttypes.h>
|
||||||
|
#include <wayland-server.h>
|
||||||
|
#include <wlr/backend.h>
|
||||||
|
#include <wlr/session.h>
|
||||||
|
#include <wlr/render.h>
|
||||||
|
#include <wlr/render/gles2.h>
|
||||||
|
#include <wlr/types/wlr_output.h>
|
||||||
|
#include <wlr/wlcore/wl_shm.h>
|
||||||
|
#include <xkbcommon/xkbcommon.h>
|
||||||
|
#include "shared.h"
|
||||||
|
|
||||||
|
struct sample_state {
|
||||||
|
struct wlr_renderer *renderer;
|
||||||
|
};
|
||||||
|
|
||||||
|
void handle_output_frame(struct output_state *output, struct timespec *ts) {
|
||||||
|
struct compositor_state *state = output->compositor;
|
||||||
|
struct sample_state *sample = state->data;
|
||||||
|
struct wlr_output *wlr_output = output->output;
|
||||||
|
|
||||||
|
wlr_renderer_begin(sample->renderer, wlr_output);
|
||||||
|
// TODO: render surfaces
|
||||||
|
wlr_renderer_end(sample->renderer);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
struct sample_state state = { 0 };
|
||||||
|
struct compositor_state compositor = { 0,
|
||||||
|
.data = &state,
|
||||||
|
.output_frame_cb = handle_output_frame,
|
||||||
|
};
|
||||||
|
compositor_init(&compositor);
|
||||||
|
|
||||||
|
state.renderer = wlr_gles2_renderer_init();
|
||||||
|
wlr_wl_shm_init(compositor.display);
|
||||||
|
|
||||||
|
compositor_run(&compositor);
|
||||||
|
}
|
|
@ -126,7 +126,6 @@ struct compositor_state {
|
||||||
struct wl_listener output_remove;
|
struct wl_listener output_remove;
|
||||||
struct wl_list outputs;
|
struct wl_list outputs;
|
||||||
|
|
||||||
bool exit;
|
|
||||||
void *data;
|
void *data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
#ifndef _WLR_ENDIAN_H
|
||||||
|
#define _WLR_ENDIAN_H
|
||||||
|
|
||||||
|
// https://stackoverflow.com/a/4240257
|
||||||
|
|
||||||
|
#define little_endian() (((union { unsigned x; unsigned char c; }){1}).c)
|
||||||
|
|
||||||
|
#endif
|
|
@ -38,6 +38,11 @@ void wlr_render_colored_quad(struct wlr_renderer *r,
|
||||||
*/
|
*/
|
||||||
void wlr_render_colored_ellipse(struct wlr_renderer *r,
|
void wlr_render_colored_ellipse(struct wlr_renderer *r,
|
||||||
const float (*color)[4], const float (*matrix)[16]);
|
const float (*color)[4], const float (*matrix)[16]);
|
||||||
|
/**
|
||||||
|
* Returns a list of pixel formats supported by this renderer.
|
||||||
|
*/
|
||||||
|
const enum wl_shm_format *wlr_renderer_get_formats(
|
||||||
|
struct wlr_renderer *r, size_t *len);
|
||||||
/**
|
/**
|
||||||
* Destroys this wlr_renderer. Surfaces must be destroyed separately.
|
* Destroys this wlr_renderer. Surfaces must be destroyed separately.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -20,9 +20,11 @@ struct wlr_renderer_impl {
|
||||||
bool (*render_with_matrix)(struct wlr_renderer_state *state,
|
bool (*render_with_matrix)(struct wlr_renderer_state *state,
|
||||||
struct wlr_surface *surface, const float (*matrix)[16]);
|
struct wlr_surface *surface, const float (*matrix)[16]);
|
||||||
void (*render_quad)(struct wlr_renderer_state *state,
|
void (*render_quad)(struct wlr_renderer_state *state,
|
||||||
const float (*color)[4], const float (*matrix)[16]);
|
const float (*color)[4], const float (*matrix)[16]);
|
||||||
void (*render_ellipse)(struct wlr_renderer_state *state,
|
void (*render_ellipse)(struct wlr_renderer_state *state,
|
||||||
const float (*color)[4], const float (*matrix)[16]);
|
const float (*color)[4], const float (*matrix)[16]);
|
||||||
|
const enum wl_shm_format *(*formats)(
|
||||||
|
struct wlr_renderer_state *state, size_t *len);
|
||||||
void (*destroy)(struct wlr_renderer_state *state);
|
void (*destroy)(struct wlr_renderer_state *state);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -36,7 +38,7 @@ struct wlr_surface_impl {
|
||||||
struct wl_shm_buffer *shm);
|
struct wl_shm_buffer *shm);
|
||||||
// TODO: egl
|
// TODO: egl
|
||||||
void (*get_matrix)(struct wlr_surface_state *state,
|
void (*get_matrix)(struct wlr_surface_state *state,
|
||||||
float (*matrix)[16], const float (*projection)[16], int x, int y);
|
float (*matrix)[16], const float (*projection)[16], int x, int y);
|
||||||
void (*bind)(struct wlr_surface_state *state);
|
void (*bind)(struct wlr_surface_state *state);
|
||||||
void (*destroy)(struct wlr_surface_state *state);
|
void (*destroy)(struct wlr_surface_state *state);
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
#ifndef _WLR_WLCORE_WL_SHM_H
|
||||||
|
#define _WLR_WLCORE_WL_SHM_H
|
||||||
|
#include <wayland-server-core.h>
|
||||||
|
#include <wlr/render.h>
|
||||||
|
|
||||||
|
struct wlr_wl_shm;
|
||||||
|
|
||||||
|
struct wlr_wl_shm *wlr_wl_shm_init(struct wl_display *display);
|
||||||
|
void wlr_wl_shm_add_format(struct wlr_wl_shm *shm, enum wl_shm_format format);
|
||||||
|
void wlr_wl_shm_add_renderer_formats(
|
||||||
|
struct wlr_wl_shm *shm, struct wlr_renderer *renderer);
|
||||||
|
|
||||||
|
#endif
|
|
@ -129,6 +129,7 @@ static void draw_quad() {
|
||||||
static bool wlr_gles2_render_surface(struct wlr_renderer_state *state,
|
static bool wlr_gles2_render_surface(struct wlr_renderer_state *state,
|
||||||
struct wlr_surface *surface, const float (*matrix)[16]) {
|
struct wlr_surface *surface, const float (*matrix)[16]) {
|
||||||
assert(surface && surface->valid);
|
assert(surface && surface->valid);
|
||||||
|
// TODO: Convert GL formats to WL_SHM formats
|
||||||
switch (surface->format) {
|
switch (surface->format) {
|
||||||
case GL_RGB:
|
case GL_RGB:
|
||||||
GL_CALL(glUseProgram(shaders.rgb));
|
GL_CALL(glUseProgram(shaders.rgb));
|
||||||
|
@ -163,6 +164,16 @@ static void wlr_gles2_render_ellipse(struct wlr_renderer_state *state,
|
||||||
draw_quad();
|
draw_quad();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const enum wl_shm_format *wlr_gles2_formats(
|
||||||
|
struct wlr_renderer_state *state, size_t *len) {
|
||||||
|
static enum wl_shm_format formats[] = {
|
||||||
|
WL_SHM_FORMAT_ARGB8888,
|
||||||
|
WL_SHM_FORMAT_XRGB8888,
|
||||||
|
};
|
||||||
|
*len = sizeof(formats) / sizeof(formats[0]);
|
||||||
|
return formats;
|
||||||
|
}
|
||||||
|
|
||||||
static void wlr_gles2_destroy(struct wlr_renderer_state *state) {
|
static void wlr_gles2_destroy(struct wlr_renderer_state *state) {
|
||||||
// no-op
|
// no-op
|
||||||
}
|
}
|
||||||
|
@ -174,6 +185,7 @@ static struct wlr_renderer_impl wlr_renderer_impl = {
|
||||||
.render_with_matrix = wlr_gles2_render_surface,
|
.render_with_matrix = wlr_gles2_render_surface,
|
||||||
.render_quad = wlr_gles2_render_quad,
|
.render_quad = wlr_gles2_render_quad,
|
||||||
.render_ellipse = wlr_gles2_render_ellipse,
|
.render_ellipse = wlr_gles2_render_ellipse,
|
||||||
|
.formats = wlr_gles2_formats,
|
||||||
.destroy = wlr_gles2_destroy
|
.destroy = wlr_gles2_destroy
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -41,3 +41,8 @@ void wlr_render_colored_ellipse(struct wlr_renderer *r,
|
||||||
const float (*color)[4], const float (*matrix)[16]) {
|
const float (*color)[4], const float (*matrix)[16]) {
|
||||||
r->impl->render_ellipse(r->state, color, matrix);
|
r->impl->render_ellipse(r->state, color, matrix);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const enum wl_shm_format *wlr_renderer_get_formats(
|
||||||
|
struct wlr_renderer *r, size_t *len) {
|
||||||
|
return r->impl->formats(r->state, len);
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
add_library(wlr-wlcore STATIC
|
||||||
|
wl_shm.c
|
||||||
|
)
|
|
@ -0,0 +1,78 @@
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <wayland-server.h>
|
||||||
|
#include <wlr/wlcore/wl_shm.h>
|
||||||
|
#include <wlr/util/list.h>
|
||||||
|
#include <wlr/util/log.h>
|
||||||
|
#include <wlr/render.h>
|
||||||
|
|
||||||
|
struct wlr_wl_shm {
|
||||||
|
struct wl_global *wl_global;
|
||||||
|
struct wl_list resources;
|
||||||
|
struct wl_list pools;
|
||||||
|
list_t *formats;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void wl_shm_destroy(struct wl_resource *resource) {
|
||||||
|
struct wlr_wl_shm *shm = wl_resource_get_user_data(resource);
|
||||||
|
struct wl_resource *_resource = NULL;
|
||||||
|
wl_resource_for_each(_resource, &shm->resources) {
|
||||||
|
if (_resource == resource) {
|
||||||
|
struct wl_list *link = wl_resource_get_link(_resource);
|
||||||
|
wl_list_remove(link);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wl_shm_interface wl_shm_impl = {
|
||||||
|
//.create_pool = wl_shm_create_pool
|
||||||
|
};
|
||||||
|
|
||||||
|
static void wl_shm_bind(struct wl_client *wl_client, void *_wlr_wl_shm,
|
||||||
|
uint32_t version, uint32_t id) {
|
||||||
|
struct wlr_wl_shm *wlr_shm = _wlr_wl_shm;
|
||||||
|
assert(wl_client && wlr_shm);
|
||||||
|
if (version > 1) {
|
||||||
|
wlr_log(L_ERROR, "Client requested unsupported wl_shm version, disconnecting");
|
||||||
|
wl_client_destroy(wl_client);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
struct wl_resource *wl_resource = wl_resource_create(
|
||||||
|
wl_client, &wl_shm_interface, version, id);
|
||||||
|
wl_resource_set_implementation(wl_resource, &wl_shm_impl,
|
||||||
|
wlr_shm, wl_shm_destroy);
|
||||||
|
wl_list_insert(&wlr_shm->resources, wl_resource_get_link(wl_resource));
|
||||||
|
for (size_t i = 0; i < wlr_shm->formats->length; ++i) {
|
||||||
|
uint32_t *f = wlr_shm->formats->items[i];
|
||||||
|
wl_shm_send_format(wl_resource, *f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wlr_wl_shm *wlr_wl_shm_init(struct wl_display *display) {
|
||||||
|
struct wlr_wl_shm *shm = calloc(1, sizeof(struct wlr_wl_shm));
|
||||||
|
wl_list_init(&shm->resources);
|
||||||
|
wl_list_init(&shm->pools);
|
||||||
|
shm->formats = list_create();
|
||||||
|
shm->wl_global = wl_global_create(display, &wl_shm_interface, 1,
|
||||||
|
shm, wl_shm_bind);
|
||||||
|
return shm;
|
||||||
|
}
|
||||||
|
|
||||||
|
void wlr_wl_shm_add_format(struct wlr_wl_shm *shm, enum wl_shm_format format) {
|
||||||
|
assert(shm);
|
||||||
|
uint32_t *f = calloc(1, sizeof(uint32_t));
|
||||||
|
*f = format;
|
||||||
|
list_add(shm->formats, f);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wlr_wl_shm_add_renderer_formats(struct wlr_wl_shm *shm,
|
||||||
|
struct wlr_renderer *renderer) {
|
||||||
|
assert(shm && renderer);
|
||||||
|
size_t len;
|
||||||
|
const enum wl_shm_format *formats = wlr_renderer_get_formats(renderer, &len);
|
||||||
|
for (size_t i = 0; i < len; ++i) {
|
||||||
|
wlr_wl_shm_add_format(shm, formats[i]);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue