#include #include #include #include #ifdef G_LOG_DOMAIN #undef G_LOG_DOMAIN #endif #define G_LOG_DOMAIN "PA-Mock" G_DEFINE_QUARK("pa-mock-state-cb-list", state_cb); G_DEFINE_QUARK("pa-mock-subscribe-callback", subscribe_cb); G_DEFINE_QUARK("pa-mock-subscribe-mask", subscribe_mask); /* ******************************* * context.h * *******************************/ typedef struct { pa_context_notify_cb_t cb; gpointer userdata; } state_cb_t; static void context_weak_cb (gpointer user_data, GObject * oldobj) { g_debug("Finalizing context: %p", oldobj); } pa_context * pa_context_new_with_proplist (pa_mainloop_api *mainloop, const char *name, pa_proplist *proplist) { GObject * gctx = g_object_new(G_TYPE_OBJECT, NULL); pa_context * ctx = (pa_context *)gctx; g_debug("Creating new context: %p", ctx); g_object_weak_ref(gctx, context_weak_cb, NULL); return ctx; } void pa_context_unref (pa_context *c) { g_return_if_fail(G_IS_OBJECT(c)); g_object_unref(G_OBJECT(c)); } pa_context * pa_context_ref (pa_context *c) { g_return_if_fail(G_IS_OBJECT(c)); g_object_ref(G_OBJECT(c)); return c; } int pa_context_connect (pa_context *c, const char *server, pa_context_flags_t flags, const pa_spawn_api *api) { g_return_if_fail(G_IS_OBJECT(c)); g_debug("Context Connect"); return 0; } void pa_context_disconnect (pa_context *c) { g_return_if_fail(G_IS_OBJECT(c)); g_debug("Context Disconnect"); } int pa_context_errno (pa_context *c) { g_return_val_if_fail(G_IS_OBJECT(c), -1); return 0; } static void state_cb_list_destroy (gpointer data) { GList * statelist = (GList *)data; g_list_free_full(statelist, g_free); } void pa_context_set_state_callback (pa_context *c, pa_context_notify_cb_t cb, void *userdata) { g_return_if_fail(G_IS_OBJECT(c)); g_return_if_fail(cb != NULL); state_cb_t * state_cb = g_new0(state_cb_t, 1); state_cb->cb = cb; state_cb->userdata = userdata; GList * statelist = g_object_get_qdata(G_OBJECT(c), state_cb_quark()); statelist = g_list_append(statelist, state_cb); g_object_set_qdata_full(G_OBJECT(c), state_cb_quark(), state_cb, state_cb_list_destroy); } pa_context_state_t pa_context_get_state (pa_context *c) { g_return_if_fail(G_IS_OBJECT(c)); return PA_CONTEXT_READY; } /* ******************************* * introspect.h * *******************************/ typedef struct { pa_server_info_cb_t cb; gpointer userdata; pa_context * context; } get_server_info_t; static void get_server_info_free (gpointer data) { get_server_info_t * info = (get_server_info_t *)data; g_object_unref(info->context); g_free(info); } static gboolean get_server_info_cb (gpointer data) { pa_server_info server = { .user_name = "user", .host_name = "host", .server_version = "1.2.3", .server_name = "server", .default_sink_name = "default-sink", .default_source_name = "default-source", .cookie = 1234, .channel_map = { .channels = 0 } }; get_server_info_t * info = (get_server_info_t *)data; info->cb(info->context, &server, info->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_get_server_info (pa_context *c, pa_server_info_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); get_server_info_t * info = g_new(get_server_info_t, 1); info->cb = cb; info->userdata = userdata; info->context = g_object_ref(c); g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, get_server_info_cb, info, get_server_info_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_sink_info_cb_t cb; gpointer userdata; pa_context * context; uint32_t index; } get_sink_info_t; static void get_sink_info_free (gpointer data) { get_sink_info_t * info = (get_sink_info_t *)data; g_object_unref(info->context); g_free(info); } static gboolean get_sink_info_cb (gpointer data) { pa_sink_info sink = { .name = "default-sink", .index = 0, .description = "Default Sink", .channel_map = { .channels = 0 } }; get_sink_info_t * info = (get_sink_info_t *)data; info->cb(info->context, &sink, 1, info->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_get_sink_info_by_name (pa_context *c, const gchar * name, pa_sink_info_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(g_strcmp0(name, "default-sink") == 0, NULL); g_return_val_if_fail(cb != NULL, NULL); get_sink_info_t * info = g_new(get_sink_info_t, 1); info->cb = cb; info->userdata = userdata; info->context = g_object_ref(c); g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, get_sink_info_cb, info, get_sink_info_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } pa_operation* pa_context_get_sink_info_list (pa_context *c, pa_sink_info_cb_t cb, void *userdata) { /* Only have one today, so this is the same */ return pa_context_get_sink_info_by_name(c, "default-sink", cb, userdata); } typedef struct { pa_sink_input_info_cb_t cb; gpointer userdata; pa_context * context; } get_sink_input_info_t; static void get_sink_input_info_free (gpointer data) { get_sink_input_info_t * info = (get_sink_input_info_t *)data; pa_context_unref(info->context); g_free(info); } static gboolean get_sink_input_info_cb (gpointer data) { pa_sink_input_info sink = { .name = "default-sink" }; get_sink_input_info_t * info = (get_sink_input_info_t *)data; info->cb(info->context, &sink, 0, info->userdata); return G_SOURCE_REMOVE; } pa_operation * pa_context_get_sink_input_info (pa_context *c, uint32_t idx, pa_sink_input_info_cb_t cb, void * userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); get_sink_input_info_t * info = g_new(get_sink_input_info_t, 1); info->cb = cb; info->userdata = userdata; info->context = g_object_ref(c); g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, get_sink_input_info_cb, info, get_sink_input_info_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_source_info_cb_t cb; gpointer userdata; pa_context * context; } get_source_info_t; static void get_source_info_free (gpointer data) { get_source_info_t * info = (get_source_info_t *)data; g_object_unref(info->context); g_free(info); } static gboolean get_source_info_cb (gpointer data) { pa_source_info source = { .name = "default-source" }; get_source_info_t * info = (get_source_info_t *)data; info->cb(info->context, &source, 0, info->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_get_source_info_by_name (pa_context *c, const char * name, pa_source_info_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); get_source_info_t * info = g_new(get_source_info_t, 1); info->cb = cb; info->userdata = userdata; info->context = g_object_ref(c); g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, get_source_info_cb, info, get_source_info_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_source_output_info_cb_t cb; gpointer userdata; pa_context * context; } get_source_output_t; static void get_source_output_free (gpointer data) { get_source_output_t * info = (get_source_output_t *)data; g_object_unref(info->context); g_free(info); } static gboolean get_source_output_cb (gpointer data) { pa_source_output_info source = { .name = "default-source" }; get_source_output_t * info = (get_source_output_t *)data; info->cb(info->context, &source, 0, info->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_get_source_output_info (pa_context *c, uint32_t idx, pa_source_output_info_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); get_source_output_t * info = g_new(get_source_output_t, 1); info->cb = cb; info->userdata = userdata; info->context = g_object_ref(c); g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, get_source_output_cb, info, get_source_output_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_context_success_cb_t cb; gpointer userdata; pa_context * context; int mute; } set_sink_mute_t; static void set_sink_mute_free (gpointer data) { set_sink_mute_t * mute = (set_sink_mute_t *)data; g_object_unref(mute->context); g_free(mute); } static gboolean set_sink_mute_cb (gpointer data) { set_sink_mute_t * mute = (set_sink_mute_t *)data; mute->cb(mute->context, 1, mute->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_set_sink_mute_by_index (pa_context *c, uint32_t idx, int mute, pa_context_success_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); set_sink_mute_t * data = g_new(set_sink_mute_t, 1); data->cb = cb; data->userdata = userdata; data->context = g_object_ref(c); data->mute = mute; g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, set_sink_mute_cb, data, set_sink_mute_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_context_success_cb_t cb; gpointer userdata; pa_context * context; pa_cvolume cvol; } set_sink_volume_t; static void set_sink_volume_free (gpointer data) { set_sink_volume_t * vol = (set_sink_volume_t *)data; g_object_unref(vol->context); g_free(vol); } static gboolean set_sink_volume_cb (gpointer data) { set_sink_volume_t * vol = (set_sink_volume_t *)data; vol->cb(vol->context, 1, vol->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_set_sink_volume_by_index (pa_context *c, uint32_t idx, const pa_cvolume * cvol, pa_context_success_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); set_sink_volume_t * data = g_new(set_sink_volume_t, 1); data->cb = cb; data->userdata = userdata; data->context = g_object_ref(c); data->cvol.channels = cvol->channels; int i; for (i = 0; i < cvol->channels; i++) data->cvol.values[i] = cvol->values[i]; g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, set_sink_volume_cb, data, set_sink_volume_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_context_success_cb_t cb; gpointer userdata; pa_context * context; pa_cvolume cvol; } set_source_volume_t; static void set_source_volume_free (gpointer data) { set_source_volume_t * vol = (set_source_volume_t *)data; g_object_unref(vol->context); g_free(vol); } static gboolean set_source_volume_cb (gpointer data) { set_source_volume_t * vol = (set_source_volume_t *)data; vol->cb(vol->context, 1, vol->userdata); return G_SOURCE_REMOVE; } pa_operation* pa_context_set_source_volume_by_name (pa_context *c, const char * name, const pa_cvolume * cvol, pa_context_success_cb_t cb, void *userdata) { g_return_val_if_fail(G_IS_OBJECT(c), NULL); g_return_val_if_fail(cb != NULL, NULL); set_source_volume_t * data = g_new(set_source_volume_t, 1); data->cb = cb; data->userdata = userdata; data->context = g_object_ref(c); data->cvol.channels = cvol->channels; int i; for (i = 0; i < cvol->channels; i++) data->cvol.values[i] = cvol->values[i]; g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, set_source_volume_cb, data, set_source_volume_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } /* ******************************* * subscribe.h * *******************************/ typedef struct { pa_context_success_cb_t cb; gpointer userdata; pa_context * context; pa_subscription_mask_t mask; } subscribe_mask_t; static void subscribe_mask_free (gpointer data) { subscribe_mask_t * mask_data = (subscribe_mask_t *)data; g_object_unref(mask_data->context); g_free(mask_data); } static gboolean subscribe_mask_cb (gpointer data) { subscribe_mask_t * mask_data = (subscribe_mask_t *)data; g_object_set_qdata(G_OBJECT(mask_data->context), subscribe_mask_quark(), GINT_TO_POINTER(mask_data->mask)); mask_data->cb(mask_data->context, 1, mask_data->userdata); return G_SOURCE_REMOVE; } pa_operation * pa_context_subscribe (pa_context * c, pa_subscription_mask_t mask, pa_context_success_cb_t callback, void * userdata) { g_return_if_fail(G_IS_OBJECT(c)); subscribe_mask_t * data = g_new0(subscribe_mask_t, 1); data->cb = callback; data->userdata = userdata; data->context = g_object_ref(G_OBJECT(c)); data->mask = mask; g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, subscribe_mask_cb, data, subscribe_mask_free); GObject * goper = g_object_new(G_TYPE_OBJECT, NULL); pa_operation * oper = (pa_operation *)goper; return oper; } typedef struct { pa_context_subscribe_cb_t cb; gpointer userdata; } subscribe_cb_t; void pa_context_set_subscribe_callback (pa_context * c, pa_context_subscribe_cb_t callback, void * userdata) { g_return_if_fail(G_IS_OBJECT(c)); subscribe_cb_t * sub = g_new0(subscribe_cb_t, 1); sub->cb = callback; sub->userdata = userdata; g_object_set_qdata_full(G_OBJECT(c), subscribe_cb_quark(), sub, g_free); } /* ******************************* * glib-mainloop.h * *******************************/ struct pa_glib_mainloop { GMainContext * context; }; struct pa_mainloop_api mock_mainloop = { 0 }; pa_mainloop_api * pa_glib_mainloop_get_api (pa_glib_mainloop * g) { return &mock_mainloop; } pa_glib_mainloop * pa_glib_mainloop_new (GMainContext * c) { pa_glib_mainloop * loop = g_new0(pa_glib_mainloop, 1); if (c == NULL) loop->context = g_main_context_default(); else loop->context = c; g_main_context_ref(loop->context); return loop; } void pa_glib_mainloop_free (pa_glib_mainloop * g) { g_main_context_unref(g->context); g_free(g); } /* ******************************* * operation.h * *******************************/ void pa_operation_unref (pa_operation * operation) { g_return_if_fail(G_IS_OBJECT(operation)); g_object_unref(G_OBJECT(operation)); } /* ******************************* * proplist.h * *******************************/ pa_proplist * pa_proplist_new (void) { return (pa_proplist *)g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free); } void pa_proplist_free (pa_proplist * p) { g_return_if_fail(p != NULL); g_hash_table_destroy((GHashTable *)p); } const char * pa_proplist_gets (pa_proplist * p, const char * key) { g_return_val_if_fail(p != NULL, NULL); g_return_val_if_fail(key != NULL, NULL); return g_hash_table_lookup((GHashTable *)p, key); } int pa_proplist_sets (pa_proplist *p, const char * key, const char * value) { g_return_val_if_fail(p != NULL, -1); g_return_val_if_fail(key != NULL, -1); g_return_val_if_fail(value != NULL, -1); g_hash_table_insert((GHashTable *)p, g_strdup(key), g_strdup(value)); return 0; } /* ******************************* * error.h * *******************************/ const char * pa_strerror (int error) { return "This is error text"; } /* ******************************* * volume.h * *******************************/ pa_volume_t pa_sw_volume_from_dB (double f) { double linear = pow(10.0, f / 20.0); pa_volume_t calculated = lround(cbrt(linear) * PA_VOLUME_NORM); if (G_UNLIKELY(calculated > PA_VOLUME_MAX)) { return PA_VOLUME_MAX; } else if (G_UNLIKELY(calculated < PA_VOLUME_MUTED)) { return PA_VOLUME_MUTED; } else { return calculated; } } pa_cvolume * pa_cvolume_init (pa_cvolume * cvol) { g_return_val_if_fail(cvol != NULL, NULL); cvol->channels = 0; unsigned int i; for (i = 0; i < PA_CHANNELS_MAX; i++) cvol->values[i] = PA_VOLUME_INVALID; return cvol; } pa_cvolume * pa_cvolume_set (pa_cvolume * cvol, unsigned channels, pa_volume_t volume) { g_return_val_if_fail(cvol != NULL, NULL); g_return_val_if_fail(channels > 0, NULL); g_return_val_if_fail(channels <= PA_CHANNELS_MAX, NULL); cvol->channels = channels; unsigned int i; for (i = 0; i < channels; i++) { if (G_UNLIKELY(volume > PA_VOLUME_MAX)) { cvol->values[i] = PA_VOLUME_MAX; } else if (G_UNLIKELY(volume < PA_VOLUME_MUTED)) { cvol->values[i] = PA_VOLUME_MUTED; } else { cvol->values[i] = volume; } } return cvol; } pa_volume_t pa_cvolume_max (const pa_cvolume * cvol) { g_return_val_if_fail(cvol != NULL, NULL); pa_volume_t max = PA_VOLUME_MUTED; unsigned int i; for (i = 0; i < cvol->channels; i++) max = MAX(max, cvol->values[i]); return max; } pa_cvolume * pa_cvolume_scale (pa_cvolume * cvol, pa_volume_t max) { g_return_val_if_fail(cvol != NULL, NULL); pa_volume_t originalmax = pa_cvolume_max(cvol); if (originalmax <= PA_VOLUME_MUTED) return pa_cvolume_set(cvol, cvol->channels, max); unsigned int i; for (i = 0; i < cvol->channels; i++) { pa_volume_t calculated = (cvol->values[i] * max) / originalmax; if (G_UNLIKELY(calculated > PA_VOLUME_MAX)) { cvol->values[i] = PA_VOLUME_MAX; } else if (G_UNLIKELY(calculated < PA_VOLUME_MUTED)) { cvol->values[i] = PA_VOLUME_MUTED; } else { cvol->values[i] = calculated; } } return cvol; }