Skip to content

Commit

Permalink
PipeWire improvements (libretro#17305)
Browse files Browse the repository at this point in the history
* Fix freeze when pipewire service is stopped/restarted
* Fix `device_list `memleak
* Refactor pipewire drivers
  • Loading branch information
viachaslavic authored Dec 29, 2024
1 parent 945d3eb commit 4124ca4
Show file tree
Hide file tree
Showing 4 changed files with 334 additions and 381 deletions.
102 changes: 80 additions & 22 deletions audio/common/pipewire.c
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,46 @@

#include "pipewire.h"

#include <spa/utils/result.h>

#include <pipewire/pipewire.h>

#include <retro_assert.h>

#include "verbosity.h"


static void core_error_cb(void *data, uint32_t id, int seq, int res, const char *message)
{
pipewire_core_t *pw = (pipewire_core_t*)data;

RARCH_ERR("[PipeWire]: error id:%u seq:%d res:%d (%s): %s\n",
id, seq, res, spa_strerror(res), message);

/* stop and exit the thread loop */
pw_thread_loop_stop(pw->thread_loop);
}

static void core_done_cb(void *data, uint32_t id, int seq)
{
pipewire_core_t *pw = (pipewire_core_t*)data;

retro_assert(id == PW_ID_CORE);

pw->last_seq = seq;
if (pw->pending_seq == seq)
{
/* stop and exit the thread loop */
pw_thread_loop_signal(pw->thread_loop, false);
}
}

static const struct pw_core_events core_events = {
PW_VERSION_CORE_EVENTS,
.done = core_done_cb,
.error = core_error_cb,
};

size_t calc_frame_size(enum spa_audio_format fmt, uint32_t nchannels)
{
uint32_t sample_size = 1;
Expand Down Expand Up @@ -80,37 +114,61 @@ void set_position(uint32_t channels, uint32_t position[SPA_AUDIO_MAX_CHANNELS])
}
}

int pipewire_wait_resync(pipewire_core_t *pipewire)
void pipewire_wait_resync(pipewire_core_t *pw)
{
int res;
retro_assert(pipewire != NULL);

pipewire->pending_seq = pw_core_sync(pipewire->core, PW_ID_CORE, pipewire->pending_seq);
retro_assert(pw);
pw->pending_seq = pw_core_sync(pw->core, PW_ID_CORE, pw->pending_seq);

for (;;)
{
pw_thread_loop_wait(pipewire->thread_loop);

res = pipewire->error;
if (res < 0)
{
pipewire->error = 0;
return res;
}
if (pipewire->pending_seq == pipewire->last_seq)
pw_thread_loop_wait(pw->thread_loop);
if (pw->pending_seq == pw->last_seq)
break;
}
return 0;
}

bool pipewire_set_active(pipewire_core_t *pipewire, pipewire_device_handle_t *device, bool active)
bool pipewire_set_active(struct pw_thread_loop *loop, struct pw_stream *stream, bool active)
{
RARCH_LOG("[PipeWire]: %s.\n", active? "Unpausing": "Pausing");
enum pw_stream_state st;
const char *error;

retro_assert(loop);
retro_assert(stream);

pw_thread_loop_lock(loop);
pw_stream_set_active(stream, active);
pw_thread_loop_wait(loop);
pw_thread_loop_unlock(loop);

st = pw_stream_get_state(stream, &error);
return active ? st == PW_STREAM_STATE_STREAMING : st == PW_STREAM_STATE_PAUSED;
}

bool pipewire_core_init(pipewire_core_t *pw, const char *loop_name)
{
retro_assert(pw);

pw->thread_loop = pw_thread_loop_new(loop_name, NULL);
if (!pw->thread_loop)
return false;

pw->ctx = pw_context_new(pw_thread_loop_get_loop(pw->thread_loop), NULL, 0);
if (!pw->ctx)
return false;

if (pw_thread_loop_start(pw->thread_loop) < 0)
return false;

pw_thread_loop_lock(pw->thread_loop);

pw->core = pw_context_connect(pw->ctx, NULL, 0);
if(!pw->core)
return false;

pw_thread_loop_lock(pipewire->thread_loop);
pw_stream_set_active(device->stream, active);
pw_thread_loop_wait(pipewire->thread_loop);
pw_thread_loop_unlock(pipewire->thread_loop);
if (pw_core_add_listener(pw->core,
&pw->core_listener,
&core_events, pw) < 0)
return false;

return device->is_paused != active;
return true;
}
40 changes: 17 additions & 23 deletions audio/common/pipewire.h
Original file line number Diff line number Diff line change
Expand Up @@ -22,17 +22,25 @@
#include <spa/param/audio/format-utils.h>
#include <spa/utils/ringbuffer.h>

#include <pipewire/pipewire.h>

#define RINGBUFFER_SIZE (1u << 22)
#define RINGBUFFER_MASK (RINGBUFFER_SIZE - 1)
#define PW_RARCH_APPNAME "RetroArch"

typedef struct pipewire
/* String literals are part of the PipeWire specification */
#define PW_RARCH_MEDIA_TYPE_AUDIO "Audio"
#define PW_RARCH_MEDIA_TYPE_VIDEO "Video"
#define PW_RARCH_MEDIA_TYPE_MIDI "Midi"
#define PW_RARCH_MEDIA_CATEGORY_PLAYBACK "Playback"
#define PW_RARCH_MEDIA_CATEGORY_RECORD "Capture"
#define PW_RARCH_MEDIA_ROLE "Game"

typedef struct pipewire_core
{
struct pw_thread_loop *thread_loop;
struct pw_context *context;
struct pw_context *ctx;
struct pw_core *core;
struct spa_hook core_listener;
int last_seq, pending_seq, error;
int last_seq, pending_seq;

struct pw_registry *registry;
struct spa_hook registry_listener;
Expand All @@ -43,28 +51,14 @@ typedef struct pipewire
struct string_list *devicelist;
} pipewire_core_t;

typedef struct pipewire_device_handle
{
pipewire_core_t *pw;

struct pw_stream *stream;
struct spa_hook stream_listener;
struct spa_audio_info_raw info;
uint32_t highwater_mark;
uint32_t frame_size;
uint32_t req;
struct spa_ringbuffer ring;
uint8_t buffer[RINGBUFFER_SIZE];

bool is_paused;
} pipewire_device_handle_t;

size_t calc_frame_size(enum spa_audio_format fmt, uint32_t nchannels);

void set_position(uint32_t channels, uint32_t position[SPA_AUDIO_MAX_CHANNELS]);

int pipewire_wait_resync(pipewire_core_t *pipewire);
void pipewire_wait_resync(pipewire_core_t *pipewire);

bool pipewire_set_active(struct pw_thread_loop *loop, struct pw_stream *stream, bool active);

bool pipewire_set_active(pipewire_core_t *pipewire, pipewire_device_handle_t *device, bool active);
bool pipewire_core_init(pipewire_core_t *pipewire, const char *loop_name);

#endif /* _RETROARCH_PIPEWIRE */
Loading

0 comments on commit 4124ca4

Please sign in to comment.