libs-back/Source/cairo/WaylandCairoSurface.m

236 lines
6.2 KiB
Mathematica
Raw Normal View History

/* -*- mode:ObjC -*-
WaylandCairoSurface - Draw with Cairo onto Wayland surfaces
Copyright (C) 2020 Free Software Foundation, Inc.
Author: Sergio L. Pascual <slp@sinrega.org>
Date: February 2016
This file is part of the GNU Objective C Backend Library.
This library 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 of the License, or (at your option) any later version.
This library 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 this library; see the file COPYING.LIB.
If not, see <http://www.gnu.org/licenses/> or write to the
Free Software Foundation, 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
2016-02-16 22:30:32 +00:00
*/
2019-10-04 20:16:54 +00:00
#define _GNU_SOURCE
2016-02-16 22:30:32 +00:00
#include "wayland/WaylandServer.h"
#include "cairo/WaylandCairoSurface.h"
#include <cairo/cairo.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/mman.h>
#define GSWINDEVICE ((struct window *)gsDevice)
2019-10-04 20:16:54 +00:00
/* Linux specific version */
2016-02-16 22:30:32 +00:00
static int
os_create_anonymous_file(off_t size)
{
static const char template[] = "/weston-shared-XXXXXX";
const char *path;
char *name;
int fd;
path = getenv("XDG_RUNTIME_DIR");
if (!path) {
errno = ENOENT;
return -1;
}
name = malloc(strlen(path) + sizeof(template));
if (!name)
return -1;
strcpy(name, path);
strcat(name, template);
2019-10-04 20:16:54 +00:00
fd = memfd_create(name, MFD_CLOEXEC);
2016-02-16 22:30:32 +00:00
free(name);
if (fd < 0)
return -1;
2019-10-04 20:16:54 +00:00
if (ftruncate(fd, size) != 0) {
2016-02-16 22:30:32 +00:00
close(fd);
return -1;
}
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
return fd;
}
static inline size_t
shm_buffer_stride(struct window *window)
{
return window->width * 4;
}
static inline size_t
shm_buffer_size(struct window *window)
{
return shm_buffer_stride(window) * window->height;
}
2019-12-29 17:30:48 +00:00
static cairo_surface_t *
2016-02-16 22:30:32 +00:00
create_shm_buffer(struct window *window)
{
struct wl_shm_pool *pool;
cairo_surface_t *surface;
int fd, size, stride;
stride = shm_buffer_stride(window);
size = shm_buffer_size(window);
2016-02-16 22:30:32 +00:00
2019-12-29 17:30:48 +00:00
NSDebugLog(@"WaylandCairoSurface: creating shm buffer of %d bytes", size);
2016-02-16 22:30:32 +00:00
fd = os_create_anonymous_file(size);
if (fd < 0) {
NSLog(@"creating a buffer file for surface failed");
return NULL;
}
window->data =
mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if (window->data == MAP_FAILED) {
NSLog(@"error mapping anonymous file");
close(fd);
return NULL;
}
pool = wl_shm_create_pool(window->wlconfig->shm, fd, size);
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
surface = cairo_image_surface_create_for_data(window->data,
CAIRO_FORMAT_ARGB32,
window->width,
window->height,
stride);
window->buffer =
wl_shm_pool_create_buffer(pool, 0,
window->width, window->height, stride,
WL_SHM_FORMAT_ARGB8888);
wl_shm_pool_destroy(pool);
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
close(fd);
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
return surface;
}
@implementation WaylandCairoSurface
- (id) initWithDevice: (void*)device
{
struct window *window = (struct window *) device;
NSDebugLog(@"WaylandCairoSurface: initWithDevice win=%d", window->window_id);
gsDevice = device;
_surface = create_shm_buffer(window);
if (_surface == NULL) {
NSDebugLog(@"can't create cairo surface");
return 0;
}
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
wl_surface_attach(window->surface, window->buffer, 0, 0);
window->wcs = self;
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
return self;
}
- (void) dealloc
{
struct window *window = (struct window*) gsDevice;
2019-10-04 20:04:58 +00:00
NSDebugLog(@"WaylandCairoSurface: dealloc win=%d", window->window_id);
// FIXME: This is leaking memory. We need to *correctly* implement
// the counterpart to create_shm_buffer.
//
// For instance, this is the wrong place to destroy the cairo surface:
// cairo_surface_destroy(window->surface);
// window->surface = NULL;
// and likely to unmap the data, and destroy/release the buffer.
// "Destroying the wl_buffer after wl_buffer.release does not change
// the surface contents. However, if the client destroys the wl_buffer
// before receiving the wl_buffer.release event, the surface contents
// become undefined immediately."
// Hence also skipping:
// munmap(window->data, shm_buffer_size(window));
2016-02-16 22:30:32 +00:00
[super dealloc];
}
- (NSSize) size
{
NSDebugLog(@"WaylandCairoSurface: size");
struct window *window = (struct window*) gsDevice;
return NSMakeSize(window->width, window->height);
}
- (void) setSurface: (cairo_surface_t*)surface
{
NSDebugLog(@"WaylandCairoSurface: setSurface");
_surface = surface;
}
- (void) handleExposeRect: (NSRect)rect
{
NSDebugLog(@"handleExposeRect");
struct window *window = (struct window*) gsDevice;
2019-12-29 17:30:48 +00:00
cairo_surface_t *cairo_surface = _surface;
2016-02-16 22:30:32 +00:00
double backupOffsetX = 0;
double backupOffsetY = 0;
int x = NSMinX(rect);
int y = NSMinY(rect);
int width = NSWidth(rect);
int height = NSHeight(rect);
NSDebugLog(@"updating region: %dx%d %dx%d", x, y, width, height);
2019-12-29 17:30:48 +00:00
if (cairo_surface_status(cairo_surface) != CAIRO_STATUS_SUCCESS)
2016-02-16 22:30:32 +00:00
{
NSWarnMLog(@"cairo initial window error status: %s\n",
cairo_status_to_string(cairo_surface_status(_surface)));
}
cairo_surface_get_device_offset(cairo_surface, &backupOffsetX, &backupOffsetY);
cairo_surface_set_device_offset(cairo_surface, 0, 0);
cairo_t *cr = cairo_create(cairo_surface);
cairo_rectangle(cr, x, y, width, height);
cairo_clip(cr);
cairo_set_source_surface(cr, cairo_surface, 0, 0);
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
cairo_paint(cr);
2019-10-04 20:04:58 +00:00
2016-02-16 22:30:32 +00:00
cairo_destroy(cr);
NSDebugLog(@"trying to commit cairo surface for window %d", window->window_id);
if (window->configured)
wl_surface_commit(window->surface);
NSDebugLog(@"done trying to commit cairo surface for window %d", window->window_id);
2016-02-16 22:30:32 +00:00
wl_display_dispatch_pending(window->wlconfig->display);
wl_display_flush(window->wlconfig->display);
2019-12-29 17:30:48 +00:00
cairo_surface_set_device_offset(_surface, backupOffsetX, backupOffsetY);
2016-02-16 22:30:32 +00:00
NSDebugLog(@"handleExposeRect exit");
}
@end