mirror of
https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
synced 2025-11-07 13:30:03 -05:00
Since 14.0, filter streams would be killed if the master sink or source disappeared. This patch changes the behavior to allow rescuing of filter streams.
301 lines
11 KiB
C
301 lines
11 KiB
C
#ifndef foocorehfoo
|
|
#define foocorehfoo
|
|
|
|
/***
|
|
This file is part of PulseAudio.
|
|
|
|
Copyright 2004-2006 Lennart Poettering
|
|
|
|
PulseAudio 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.1 of the License,
|
|
or (at your option) any later version.
|
|
|
|
PulseAudio 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
|
|
General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
|
|
***/
|
|
|
|
#include <pulsecore/typedefs.h>
|
|
#include <pulse/mainloop-api.h>
|
|
#include <pulse/sample.h>
|
|
#include <pulsecore/cpu.h>
|
|
|
|
/* This is a bitmask that encodes the cause why a sink/source is
|
|
* suspended.
|
|
*
|
|
* When adding new causes, remember to update pa_suspend_cause_to_string() and
|
|
* PA_SUSPEND_CAUSE_TO_STRING_BUF_SIZE! */
|
|
typedef enum pa_suspend_cause {
|
|
PA_SUSPEND_USER = 1, /* Exposed to the user via some protocol */
|
|
PA_SUSPEND_APPLICATION = 2, /* Used by the device reservation logic */
|
|
PA_SUSPEND_IDLE = 4, /* Used by module-suspend-on-idle */
|
|
PA_SUSPEND_SESSION = 8, /* Used by module-hal for mark inactive sessions */
|
|
PA_SUSPEND_PASSTHROUGH = 16, /* Used to suspend monitor sources when the sink is in passthrough mode */
|
|
PA_SUSPEND_INTERNAL = 32, /* This is used for short period server-internal suspends, such as for sample rate updates */
|
|
PA_SUSPEND_UNAVAILABLE = 64, /* Used by device implementations that have to suspend when the device is unavailable */
|
|
PA_SUSPEND_ALL = 0xFFFF /* Magic cause that can be used to resume forcibly */
|
|
} pa_suspend_cause_t;
|
|
|
|
#include <pulsecore/idxset.h>
|
|
#include <pulsecore/hashmap.h>
|
|
#include <pulsecore/memblock.h>
|
|
#include <pulsecore/resampler.h>
|
|
#include <pulsecore/llist.h>
|
|
#include <pulsecore/hook-list.h>
|
|
#include <pulsecore/asyncmsgq.h>
|
|
#include <pulsecore/sample-util.h>
|
|
#include <pulsecore/sink.h>
|
|
#include <pulsecore/source.h>
|
|
#include <pulsecore/core-subscribe.h>
|
|
#include <pulsecore/msgobject.h>
|
|
|
|
typedef enum pa_server_type {
|
|
PA_SERVER_TYPE_UNSET,
|
|
PA_SERVER_TYPE_USER,
|
|
PA_SERVER_TYPE_SYSTEM,
|
|
PA_SERVER_TYPE_NONE
|
|
} pa_server_type_t;
|
|
|
|
typedef enum pa_core_state {
|
|
PA_CORE_STARTUP,
|
|
PA_CORE_RUNNING,
|
|
PA_CORE_SHUTDOWN
|
|
} pa_core_state_t;
|
|
|
|
typedef enum pa_core_hook {
|
|
PA_CORE_HOOK_SINK_NEW,
|
|
PA_CORE_HOOK_SINK_FIXATE,
|
|
PA_CORE_HOOK_SINK_PUT,
|
|
PA_CORE_HOOK_SINK_UNLINK,
|
|
PA_CORE_HOOK_SINK_UNLINK_POST,
|
|
PA_CORE_HOOK_SINK_STATE_CHANGED,
|
|
PA_CORE_HOOK_SINK_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_SINK_PORT_CHANGED,
|
|
PA_CORE_HOOK_SINK_FLAGS_CHANGED,
|
|
PA_CORE_HOOK_SINK_VOLUME_CHANGED,
|
|
PA_CORE_HOOK_SINK_MUTE_CHANGED,
|
|
PA_CORE_HOOK_SINK_PORT_LATENCY_OFFSET_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_NEW,
|
|
PA_CORE_HOOK_SOURCE_FIXATE,
|
|
PA_CORE_HOOK_SOURCE_PUT,
|
|
PA_CORE_HOOK_SOURCE_UNLINK,
|
|
PA_CORE_HOOK_SOURCE_UNLINK_POST,
|
|
PA_CORE_HOOK_SOURCE_STATE_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_PORT_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_FLAGS_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_VOLUME_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_MUTE_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_PORT_LATENCY_OFFSET_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_NEW,
|
|
PA_CORE_HOOK_SINK_INPUT_FIXATE,
|
|
PA_CORE_HOOK_SINK_INPUT_PUT,
|
|
PA_CORE_HOOK_SINK_INPUT_UNLINK,
|
|
PA_CORE_HOOK_SINK_INPUT_UNLINK_POST,
|
|
PA_CORE_HOOK_SINK_INPUT_MOVE_START,
|
|
PA_CORE_HOOK_SINK_INPUT_MOVE_FINISH,
|
|
PA_CORE_HOOK_SINK_INPUT_MOVE_FAIL,
|
|
PA_CORE_HOOK_SINK_INPUT_STATE_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_VOLUME_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_MUTE_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_PREFERRED_SINK_CHANGED,
|
|
PA_CORE_HOOK_SINK_INPUT_SEND_EVENT,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_NEW,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_FIXATE,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_PUT,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_UNLINK,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_UNLINK_POST,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_MOVE_START,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_MOVE_FINISH,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_MOVE_FAIL,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_STATE_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_VOLUME_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_MUTE_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_PREFERRED_SOURCE_CHANGED,
|
|
PA_CORE_HOOK_SOURCE_OUTPUT_SEND_EVENT,
|
|
PA_CORE_HOOK_CLIENT_NEW,
|
|
PA_CORE_HOOK_CLIENT_PUT,
|
|
PA_CORE_HOOK_CLIENT_UNLINK,
|
|
PA_CORE_HOOK_CLIENT_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_CLIENT_SEND_EVENT,
|
|
PA_CORE_HOOK_CARD_NEW,
|
|
PA_CORE_HOOK_CARD_CHOOSE_INITIAL_PROFILE,
|
|
PA_CORE_HOOK_CARD_PUT,
|
|
PA_CORE_HOOK_CARD_UNLINK,
|
|
PA_CORE_HOOK_CARD_PREFERRED_PORT_CHANGED,
|
|
PA_CORE_HOOK_CARD_PROFILE_CHANGED,
|
|
PA_CORE_HOOK_CARD_PROFILE_ADDED,
|
|
PA_CORE_HOOK_CARD_PROFILE_AVAILABLE_CHANGED,
|
|
PA_CORE_HOOK_CARD_SUSPEND_CHANGED,
|
|
PA_CORE_HOOK_PORT_AVAILABLE_CHANGED,
|
|
PA_CORE_HOOK_PORT_LATENCY_OFFSET_CHANGED,
|
|
PA_CORE_HOOK_DEFAULT_SINK_CHANGED,
|
|
PA_CORE_HOOK_DEFAULT_SOURCE_CHANGED,
|
|
PA_CORE_HOOK_MODULE_NEW,
|
|
PA_CORE_HOOK_MODULE_PROPLIST_CHANGED,
|
|
PA_CORE_HOOK_MODULE_UNLINK,
|
|
PA_CORE_HOOK_SAMPLE_CACHE_NEW,
|
|
PA_CORE_HOOK_SAMPLE_CACHE_CHANGED,
|
|
PA_CORE_HOOK_SAMPLE_CACHE_UNLINK,
|
|
PA_CORE_HOOK_MAX
|
|
} pa_core_hook_t;
|
|
|
|
/* The core structure of PulseAudio. Every PulseAudio daemon contains
|
|
* exactly one of these. It is used for storing kind of global
|
|
* variables for the daemon. */
|
|
|
|
struct pa_core {
|
|
pa_msgobject parent;
|
|
|
|
pa_core_state_t state;
|
|
|
|
/* A random value which may be used to identify this instance of
|
|
* PulseAudio. Not cryptographically secure in any way. */
|
|
uint32_t cookie;
|
|
|
|
pa_mainloop_api *mainloop;
|
|
|
|
/* idxset of all kinds of entities */
|
|
pa_idxset *clients, *cards, *sinks, *sources, *sink_inputs, *source_outputs, *modules, *scache;
|
|
|
|
/* Some hashmaps for all sorts of entities */
|
|
pa_hashmap *namereg, *shared, *message_handlers;
|
|
|
|
/* The default sink/source as configured by the user. If the user hasn't
|
|
* explicitly configured anything, these are set to NULL. These are strings
|
|
* instead of sink/source pointers, because that allows us to reference
|
|
* devices that don't currently exist. That's useful for remembering that
|
|
* a hotplugged USB sink was previously set as the default sink. */
|
|
char *configured_default_sink;
|
|
char *configured_default_source;
|
|
|
|
/* The default sink/source set by some policy module. This will override
|
|
* the user configured default sink/source, so that the default will
|
|
* return to the user configured sink/source once the sink/source set by
|
|
* the policy module is no longer available. */
|
|
char *policy_default_sink;
|
|
char *policy_default_source;
|
|
|
|
/* The effective default sink/source. If no sink or source is explicitly
|
|
* configured as the default, we pick the device that ranks highest
|
|
* according to the compare_sinks() and compare_sources() functions in
|
|
* core.c. pa_core_update_default_sink/source() has to be called whenever
|
|
* anything changes that might change the comparison results. */
|
|
pa_sink *default_sink;
|
|
pa_source *default_source;
|
|
|
|
pa_channel_map default_channel_map;
|
|
pa_sample_spec default_sample_spec;
|
|
uint32_t alternate_sample_rate;
|
|
unsigned default_n_fragments, default_fragment_size_msec;
|
|
unsigned deferred_volume_safety_margin_usec;
|
|
int deferred_volume_extra_delay_usec;
|
|
unsigned lfe_crossover_freq;
|
|
|
|
pa_defer_event *module_defer_unload_event;
|
|
pa_hashmap *modules_pending_unload; /* pa_module -> pa_module (hashmap-as-a-set) */
|
|
|
|
pa_defer_event *subscription_defer_event;
|
|
PA_LLIST_HEAD(pa_subscription, subscriptions);
|
|
PA_LLIST_HEAD(pa_subscription_event, subscription_event_queue);
|
|
pa_subscription_event *subscription_event_last;
|
|
|
|
/* The mempool is used for data we write to, it's readonly for the client. */
|
|
pa_mempool *mempool;
|
|
|
|
/* Shared memory size, as specified either by daemon configuration
|
|
* or PA daemon defaults (~ 64 MiB). */
|
|
size_t shm_size;
|
|
|
|
pa_silence_cache silence_cache;
|
|
|
|
pa_time_event *exit_event;
|
|
pa_time_event *scache_auto_unload_event;
|
|
|
|
int exit_idle_time, scache_idle_time;
|
|
|
|
bool flat_volumes:1;
|
|
bool rescue_streams:1;
|
|
bool disallow_module_loading:1;
|
|
bool disallow_exit:1;
|
|
bool running_as_daemon:1;
|
|
bool realtime_scheduling:1;
|
|
bool avoid_resampling:1;
|
|
bool disable_remixing:1;
|
|
bool remixing_use_all_sink_channels:1;
|
|
bool remixing_produce_lfe:1;
|
|
bool remixing_consume_lfe:1;
|
|
bool deferred_volume:1;
|
|
|
|
pa_resample_method_t resample_method;
|
|
int realtime_priority;
|
|
|
|
pa_server_type_t server_type;
|
|
pa_cpu_info cpu_info;
|
|
|
|
/* hooks */
|
|
pa_hook hooks[PA_CORE_HOOK_MAX];
|
|
};
|
|
|
|
PA_DECLARE_PUBLIC_CLASS(pa_core);
|
|
#define PA_CORE(o) pa_core_cast(o)
|
|
|
|
enum {
|
|
PA_CORE_MESSAGE_UNLOAD_MODULE,
|
|
PA_CORE_MESSAGE_MAX
|
|
};
|
|
|
|
pa_core* pa_core_new(pa_mainloop_api *m, bool shared, bool enable_memfd, size_t shm_size);
|
|
|
|
void pa_core_set_configured_default_sink(pa_core *core, const char *sink);
|
|
void pa_core_set_configured_default_source(pa_core *core, const char *source);
|
|
void pa_core_set_policy_default_sink(pa_core *core, const char *sink);
|
|
void pa_core_set_policy_default_source(pa_core *core, const char *source);
|
|
|
|
/* These should be called whenever something changes that may affect the
|
|
* default sink or source choice.
|
|
*
|
|
* If the default source choice happens between two monitor sources, the
|
|
* monitored sinks are compared, so if the default sink changes, the default
|
|
* source may change too. However, pa_core_update_default_sink() calls
|
|
* pa_core_update_default_source() internally, so it's sufficient to only call
|
|
* pa_core_update_default_sink() when something happens that affects the sink
|
|
* ordering. */
|
|
void pa_core_update_default_sink(pa_core *core);
|
|
void pa_core_update_default_source(pa_core *core);
|
|
|
|
pa_sink *pa_core_find_best_sink(pa_core *core, bool ignore_configured_virtual_default);
|
|
pa_source *pa_core_find_best_source(pa_core *core, bool ignore_configured_virtual_default);
|
|
|
|
void pa_core_set_exit_idle_time(pa_core *core, int time);
|
|
|
|
/* Check whether no one is connected to this core */
|
|
void pa_core_check_idle(pa_core *c);
|
|
|
|
int pa_core_exit(pa_core *c, bool force, int retval);
|
|
|
|
void pa_core_maybe_vacuum(pa_core *c);
|
|
|
|
/* wrapper for c->mainloop->time_*() RT time events */
|
|
pa_time_event* pa_core_rttime_new(pa_core *c, pa_usec_t usec, pa_time_event_cb_t cb, void *userdata);
|
|
void pa_core_rttime_restart(pa_core *c, pa_time_event *e, pa_usec_t usec);
|
|
|
|
static const size_t PA_SUSPEND_CAUSE_TO_STRING_BUF_SIZE =
|
|
sizeof("USER|APPLICATION|IDLE|SESSION|PASSTHROUGH|INTERNAL|UNAVAILABLE");
|
|
|
|
/* Converts the given suspend cause to a string. The string is written to the
|
|
* provided buffer. The same buffer is the return value of this function. */
|
|
const char *pa_suspend_cause_to_string(pa_suspend_cause_t cause, char buf[PA_SUSPEND_CAUSE_TO_STRING_BUF_SIZE]);
|
|
|
|
void pa_core_move_streams_to_newly_available_preferred_sink(pa_core *c, pa_sink *s);
|
|
|
|
void pa_core_move_streams_to_newly_available_preferred_source(pa_core *c, pa_source *s);
|
|
|
|
#endif
|