From 05000b6e378e6aaacbb452c3e4b08c37a4760830 Mon Sep 17 00:00:00 2001 From: columbarius Date: Thu, 27 May 2021 01:48:44 +0200 Subject: [PATCH 1/5] screencast: announce supported buffer type --- src/screencast/pipewire_screencast.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/screencast/pipewire_screencast.c b/src/screencast/pipewire_screencast.c index b7f8162d..000184bc 100644 --- a/src/screencast/pipewire_screencast.c +++ b/src/screencast/pipewire_screencast.c @@ -153,7 +153,8 @@ static void pwr_handle_stream_param_changed(void *data, uint32_t id, SPA_PARAM_BUFFERS_blocks, SPA_POD_Int(1), SPA_PARAM_BUFFERS_size, SPA_POD_Int(cast->simple_frame.size), SPA_PARAM_BUFFERS_stride, SPA_POD_Int(cast->simple_frame.stride), - SPA_PARAM_BUFFERS_align, SPA_POD_Int(XDPW_PWR_ALIGN)); + SPA_PARAM_BUFFERS_align, SPA_POD_Int(XDPW_PWR_ALIGN), + SPA_PARAM_BUFFERS_dataType,SPA_POD_CHOICE_FLAGS_Int(1< Date: Wed, 21 Oct 2020 19:41:26 +0200 Subject: [PATCH 2/5] screencast: pipewire add {add,remove}_buffer handler --- src/screencast/pipewire_screencast.c | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/src/screencast/pipewire_screencast.c b/src/screencast/pipewire_screencast.c index 000184bc..ba3b0e55 100644 --- a/src/screencast/pipewire_screencast.c +++ b/src/screencast/pipewire_screencast.c @@ -164,10 +164,33 @@ static void pwr_handle_stream_param_changed(void *data, uint32_t id, pw_stream_update_params(stream, params, 2); } +static void pwr_handle_stream_add_buffer(void *data, struct pw_buffer *buffer) { + struct xdpw_screencast_instance *cast = data; + struct spa_data *d; + + logprint(TRACE, "pipewire: add buffer event handle"); + + d = buffer->buffer->datas; + + // Select buffer type from negotiation result + if ((d[0].type & (1u << SPA_DATA_MemPtr)) > 0) { + d[0].type = SPA_DATA_MemPtr; + } else { + logprint(ERROR, "pipewire: unsupported buffer type"); + cast->err = 1; + } +} + +static void pwr_handle_stream_remove_buffer(void *data, struct pw_buffer *buffer) { + logprint(TRACE, "pipewire: remove buffer event handle"); +} + static const struct pw_stream_events pwr_stream_events = { PW_VERSION_STREAM_EVENTS, .state_changed = pwr_handle_stream_state_changed, .param_changed = pwr_handle_stream_param_changed, + .add_buffer = pwr_handle_stream_add_buffer, + .remove_buffer = pwr_handle_stream_remove_buffer, }; void pwr_update_stream_param(struct xdpw_screencast_instance *cast) { From 07ded597a2b1543051c398691e3f380d81ab6597 Mon Sep 17 00:00:00 2001 From: columbarius Date: Mon, 26 Oct 2020 03:22:06 +0100 Subject: [PATCH 3/5] screencast: pipewire move buffer information from pwr_on_event to pwr_handle_stream_add_buffer --- src/screencast/pipewire_screencast.c | 21 ++++++++++++--------- 1 file changed, 12 insertions(+), 9 deletions(-) diff --git a/src/screencast/pipewire_screencast.c b/src/screencast/pipewire_screencast.c index ba3b0e55..906dcdfa 100644 --- a/src/screencast/pipewire_screencast.c +++ b/src/screencast/pipewire_screencast.c @@ -85,15 +85,6 @@ static void pwr_on_event(void *data, uint64_t expirations) { h->dts_offset = 0; } - d[0].type = SPA_DATA_MemPtr; - d[0].maxsize = cast->simple_frame.size; - d[0].mapoffset = 0; - d[0].chunk->size = cast->simple_frame.size; - d[0].chunk->stride = cast->simple_frame.stride; - d[0].chunk->offset = 0; - d[0].flags = 0; - d[0].fd = -1; - writeFrameData(d[0].data, cast->simple_frame.data, cast->simple_frame.height, cast->simple_frame.stride, cast->simple_frame.y_invert); @@ -178,6 +169,18 @@ static void pwr_handle_stream_add_buffer(void *data, struct pw_buffer *buffer) { } else { logprint(ERROR, "pipewire: unsupported buffer type"); cast->err = 1; + return; + } + + // Prepare buffer for choosen type + if (d[0].type == SPA_DATA_MemPtr) { + d[0].maxsize = cast->simple_frame.size; + d[0].mapoffset = 0; + d[0].chunk->size = cast->simple_frame.size; + d[0].chunk->stride = cast->simple_frame.stride; + d[0].chunk->offset = 0; + d[0].flags = 0; + d[0].fd = -1; } } From cbaad394f4b7ec9db405c66c58f3043da8242a69 Mon Sep 17 00:00:00 2001 From: columbarius Date: Fri, 23 Oct 2020 01:55:12 +0200 Subject: [PATCH 4/5] screencast: pipewire remove uneeded PW_STREAM_FLAG_MAP_BUFFERS flag --- src/screencast/pipewire_screencast.c | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/screencast/pipewire_screencast.c b/src/screencast/pipewire_screencast.c index 906dcdfa..babca821 100644 --- a/src/screencast/pipewire_screencast.c +++ b/src/screencast/pipewire_screencast.c @@ -250,8 +250,7 @@ void xdpw_pwr_stream_create(struct xdpw_screencast_instance *cast) { pw_stream_connect(cast->stream, PW_DIRECTION_OUTPUT, PW_ID_ANY, - (PW_STREAM_FLAG_DRIVER | - PW_STREAM_FLAG_MAP_BUFFERS), + PW_STREAM_FLAG_DRIVER, ¶m, 1); } From b7ee68a601d750888fe94f39d79c201cb2781df5 Mon Sep 17 00:00:00 2001 From: columbarius Date: Mon, 26 Oct 2020 16:22:08 +0100 Subject: [PATCH 5/5] screencast: pipewire change to self allocation of shm buffers --- include/screencast_common.h | 1 + src/screencast/pipewire_screencast.c | 48 ++++++++++++++++++++++++---- src/screencast/screencast_common.c | 24 ++++++++++++++ src/screencast/wlr_screencast.c | 19 ----------- 4 files changed, 67 insertions(+), 25 deletions(-) diff --git a/include/screencast_common.h b/include/screencast_common.h index e35fc2d1..4f7b4144 100644 --- a/include/screencast_common.h +++ b/include/screencast_common.h @@ -122,6 +122,7 @@ struct xdpw_wlr_output { }; void randname(char *buf); +int anonymous_shm_open(void); enum spa_video_format xdpw_format_pw_from_wl_shm(enum wl_shm_format format); enum spa_video_format xdpw_format_pw_strip_alpha(enum spa_video_format format); diff --git a/src/screencast/pipewire_screencast.c b/src/screencast/pipewire_screencast.c index babca821..d8efb108 100644 --- a/src/screencast/pipewire_screencast.c +++ b/src/screencast/pipewire_screencast.c @@ -5,6 +5,8 @@ #include #include #include +#include +#include #include "wlr_screencast.h" #include "xdpw.h" @@ -145,7 +147,7 @@ static void pwr_handle_stream_param_changed(void *data, uint32_t id, SPA_PARAM_BUFFERS_size, SPA_POD_Int(cast->simple_frame.size), SPA_PARAM_BUFFERS_stride, SPA_POD_Int(cast->simple_frame.stride), SPA_PARAM_BUFFERS_align, SPA_POD_Int(XDPW_PWR_ALIGN), - SPA_PARAM_BUFFERS_dataType,SPA_POD_CHOICE_FLAGS_Int(1<buffer->datas; // Select buffer type from negotiation result - if ((d[0].type & (1u << SPA_DATA_MemPtr)) > 0) { - d[0].type = SPA_DATA_MemPtr; + if ((d[0].type & (1u << SPA_DATA_MemFd)) > 0) { + d[0].type = SPA_DATA_MemFd; } else { logprint(ERROR, "pipewire: unsupported buffer type"); cast->err = 1; return; } + logprint(TRACE, "pipewire: selected buffertype %u", d[0].type); // Prepare buffer for choosen type - if (d[0].type == SPA_DATA_MemPtr) { + if (d[0].type == SPA_DATA_MemFd) { d[0].maxsize = cast->simple_frame.size; d[0].mapoffset = 0; d[0].chunk->size = cast->simple_frame.size; d[0].chunk->stride = cast->simple_frame.stride; d[0].chunk->offset = 0; d[0].flags = 0; - d[0].fd = -1; + d[0].fd = anonymous_shm_open(); + + if (d[0].fd == -1) { + logprint(ERROR, "pipewire: unable to create anonymous filedescriptor"); + cast->err = 1; + return; + } + + if (ftruncate(d[0].fd, d[0].maxsize) < 0) { + logprint(ERROR, "pipewire: unable to truncate filedescriptor"); + close(d[0].fd); + d[0].fd = -1; + cast->err = 1; + return; + } + + // mmap buffer, so we can use the data_ptr in on_process + d[0].data = mmap(NULL, d[0].maxsize, PROT_READ | PROT_WRITE, MAP_SHARED, d[0].fd, d[0].mapoffset); + if (d[0].data == MAP_FAILED) { + logprint(ERROR, "pipewire: unable to mmap memory"); + cast->err = 1; + return; + } } } static void pwr_handle_stream_remove_buffer(void *data, struct pw_buffer *buffer) { logprint(TRACE, "pipewire: remove buffer event handle"); + + struct spa_data *d = buffer->buffer->datas; + switch (d[0].type) { + case SPA_DATA_MemFd: + munmap(d[0].data, d[0].maxsize); + close(d[0].fd); + break; + default: + break; + } } static const struct pw_stream_events pwr_stream_events = { @@ -250,7 +285,8 @@ void xdpw_pwr_stream_create(struct xdpw_screencast_instance *cast) { pw_stream_connect(cast->stream, PW_DIRECTION_OUTPUT, PW_ID_ANY, - PW_STREAM_FLAG_DRIVER, + (PW_STREAM_FLAG_DRIVER | + PW_STREAM_FLAG_ALLOC_BUFFERS), ¶m, 1); } diff --git a/src/screencast/screencast_common.c b/src/screencast/screencast_common.c index 63d8d9a7..915a5856 100644 --- a/src/screencast/screencast_common.c +++ b/src/screencast/screencast_common.c @@ -1,5 +1,10 @@ #include "screencast_common.h" #include +#include +#include +#include +#include +#include void randname(char *buf) { struct timespec ts; @@ -12,6 +17,25 @@ void randname(char *buf) { } } +int anonymous_shm_open(void) { + char name[] = "/xdpw-shm-XXXXXX"; + int retries = 100; + + do { + randname(name + strlen(name) - 6); + + --retries; + // shm_open guarantees that O_CLOEXEC is set + int fd = shm_open(name, O_RDWR | O_CREAT | O_EXCL, S_IRUSR | S_IWUSR); + if (fd >= 0) { + shm_unlink(name); + return fd; + } + } while (retries > 0 && errno == EEXIST); + + return -1; +} + enum spa_video_format xdpw_format_pw_from_wl_shm(enum wl_shm_format format) { switch (format) { case WL_SHM_FORMAT_ARGB8888: diff --git a/src/screencast/wlr_screencast.c b/src/screencast/wlr_screencast.c index ca8ef933..21164a20 100644 --- a/src/screencast/wlr_screencast.c +++ b/src/screencast/wlr_screencast.c @@ -61,25 +61,6 @@ void xdpw_wlr_frame_free(struct xdpw_screencast_instance *cast) { } } -static int anonymous_shm_open(void) { - char name[] = "/xdpw-shm-XXXXXX"; - int retries = 100; - - do { - randname(name + strlen(name) - 6); - - --retries; - // shm_open guarantees that O_CLOEXEC is set - int fd = shm_open(name, O_RDWR | O_CREAT | O_EXCL, S_IRUSR | S_IWUSR); - if (fd >= 0) { - shm_unlink(name); - return fd; - } - } while (retries > 0 && errno == EEXIST); - - return -1; -} - static struct wl_buffer *create_shm_buffer(struct xdpw_screencast_instance *cast, enum wl_shm_format fmt, int width, int height, int stride, void **data_out) {