obs-source: Add audio capture and selectively listen to events

Audio capture previously needed an additional class, however there is no need for this if this can be done together with the actual obs::source object. So obs::source now has a new audio_data event that can be listened to, which handles capturing audio data if requested.

Additionally all other events are now selectively listened to, instead of permanently listening to them. This should drastically reduce the overhead of using obs::source.
This commit is contained in:
Michael Fabian 'Xaymar' Dirks 2019-04-02 19:12:44 +02:00
parent 1b7de8b6b1
commit 8fb7d193f5
2 changed files with 94 additions and 85 deletions

View File

@ -286,6 +286,16 @@ void obs::source::handle_audio_mixers(void* p, calldata_t* calldata)
calldata_set_int(calldata, "mixers", mixers); calldata_set_int(calldata, "mixers", mixers);
} }
void obs::source::handle_audio_data(void* p, obs_source_t* source, const audio_data* audio, bool muted)
{
obs::source* self = reinterpret_cast<obs::source*>(p);
if (!self->events.audio_data) {
return;
}
self->events.audio_data(self, audio, muted);
}
void obs::source::handle_filter_add(void* p, calldata_t* calldata) void obs::source::handle_filter_add(void* p, calldata_t* calldata)
{ {
obs::source* self = reinterpret_cast<obs::source*>(p); obs::source* self = reinterpret_cast<obs::source*>(p);
@ -352,11 +362,34 @@ void obs::source::handle_transition_stop(void* p, calldata_t*)
self->events.transition_stop(self); self->events.transition_stop(self);
} }
void obs::source::connect_signals() obs::source::~source()
{ {
auto sh = obs_source_get_signal_handler(this->self); if (this->track_ownership && this->self) {
if (sh) { obs_source_release(this->self);
#define auto_signal_c(SIGNAL) signal_handler_connect(sh, "" #SIGNAL, obs::source::handle_##SIGNAL, this); }
this->self = nullptr;
}
obs::source::source()
{
#ifdef auto_signal_c
#undef auto_signal_c
#endif
#define auto_signal_c(SIGNAL) \
{ \
this->events.##SIGNAL.set_listen_callback([this] { \
auto sh = obs_source_get_signal_handler(this->self); \
if (sh) { \
signal_handler_connect(sh, "" #SIGNAL, obs::source::handle_##SIGNAL, this); \
} \
}); \
this->events.##SIGNAL.set_silence_callback([this] { \
auto sh = obs_source_get_signal_handler(this->self); \
if (sh) { \
signal_handler_disconnect(sh, "" #SIGNAL, obs::source::handle_##SIGNAL, this); \
} \
}); \
}
auto_signal_c(destroy); auto_signal_c(destroy);
auto_signal_c(remove); auto_signal_c(remove);
auto_signal_c(save); auto_signal_c(save);
@ -384,51 +417,18 @@ void obs::source::connect_signals()
auto_signal_c(transition_video_stop); auto_signal_c(transition_video_stop);
auto_signal_c(transition_stop); auto_signal_c(transition_stop);
#undef auto_signal_c #undef auto_signal_c
}
}
obs::source::~source() // libOBS unfortunately does not use the event system for audio data callbacks, which is kind of odd as most other
// things do. So instead we'll have to manually deal with it for now.
{ {
if (this->self) { this->events.audio_data.set_listen_callback(
auto sh = obs_source_get_signal_handler(this->self); [this] { obs_source_add_audio_capture_callback(this->self, obs::source::handle_audio_data, this); });
if (sh) { this->events.audio_data.set_silence_callback(
#define auto_signal_dc(SIGNAL) signal_handler_disconnect(sh, "" #SIGNAL, obs::source::handle_##SIGNAL, this); [this] { obs_source_remove_audio_capture_callback(this->self, obs::source::handle_audio_data, this); });
auto_signal_dc(destroy);
auto_signal_dc(remove);
auto_signal_dc(save);
auto_signal_dc(load);
auto_signal_dc(activate);
auto_signal_dc(deactivate);
auto_signal_dc(show);
auto_signal_dc(hide);
auto_signal_dc(mute);
auto_signal_dc(push_to_mute_changed);
auto_signal_dc(push_to_mute_delay);
auto_signal_dc(push_to_talk_changed);
auto_signal_dc(push_to_talk_delay);
auto_signal_dc(enable);
auto_signal_dc(rename);
auto_signal_dc(volume);
auto_signal_dc(update_properties);
auto_signal_dc(update_flags);
auto_signal_dc(audio_sync);
auto_signal_dc(audio_mixers);
auto_signal_dc(filter_add);
auto_signal_dc(filter_remove);
auto_signal_dc(reorder_filters);
auto_signal_dc(transition_start);
auto_signal_dc(transition_video_stop);
auto_signal_dc(transition_stop);
#undef auto_signal_dc
} }
} }
if (this->track_ownership && this->self) {
obs_source_release(this->self);
}
this->self = nullptr;
}
obs::source::source(std::string name, bool track_ownership, bool add_reference) obs::source::source(std::string name, bool track_ownership, bool add_reference) : source()
{ {
this->self = obs_get_source_by_name(name.c_str()); this->self = obs_get_source_by_name(name.c_str());
if (!this->self) { if (!this->self) {
@ -439,10 +439,9 @@ obs::source::source(std::string name, bool track_ownership, bool add_reference)
if (!add_reference) { if (!add_reference) {
obs_source_release(this->self); obs_source_release(this->self);
} }
connect_signals();
} }
obs::source::source(obs_source_t* source, bool track_ownership, bool add_reference) obs::source::source(obs_source_t* source, bool track_ownership, bool add_reference) : source()
{ {
this->self = source; this->self = source;
if (!this->self) { if (!this->self) {
@ -453,7 +452,6 @@ obs::source::source(obs_source_t* source, bool track_ownership, bool add_referen
if (add_reference) { if (add_reference) {
obs_source_addref(this->self); obs_source_addref(this->self);
} }
connect_signals();
} }
obs::source::source(source const& other) obs::source::source(source const& other)

View File

@ -60,6 +60,7 @@ namespace obs {
static void handle_volume(void* p, calldata_t* calldata); static void handle_volume(void* p, calldata_t* calldata);
static void handle_audio_sync(void* p, calldata_t* calldata); static void handle_audio_sync(void* p, calldata_t* calldata);
static void handle_audio_mixers(void* p, calldata_t* calldata); static void handle_audio_mixers(void* p, calldata_t* calldata);
static void handle_audio_data(void* p, obs_source_t* source, const audio_data* audio, bool muted);
static void handle_filter_add(void* p, calldata_t* calldata); static void handle_filter_add(void* p, calldata_t* calldata);
static void handle_filter_remove(void* p, calldata_t* calldata); static void handle_filter_remove(void* p, calldata_t* calldata);
static void handle_reorder_filters(void* p, calldata_t* calldata); static void handle_reorder_filters(void* p, calldata_t* calldata);
@ -67,12 +68,11 @@ namespace obs {
static void handle_transition_video_stop(void* p, calldata_t* calldata); static void handle_transition_video_stop(void* p, calldata_t* calldata);
static void handle_transition_stop(void* p, calldata_t* calldata); static void handle_transition_stop(void* p, calldata_t* calldata);
private:
void connect_signals();
public: public:
virtual ~source(); virtual ~source();
source();
source(std::string name, bool track_ownership = true, bool add_reference = true); source(std::string name, bool track_ownership = true, bool add_reference = true);
source(obs_source_t* source, bool track_ownership = true, bool add_reference = false); source(obs_source_t* source, bool track_ownership = true, bool add_reference = false);
@ -102,36 +102,47 @@ namespace obs {
public: // Events public: // Events
struct { struct {
// Destroy and Remove
util::event<obs::source*> destroy; util::event<obs::source*> destroy;
util::event<obs::source*> remove; util::event<obs::source*> remove;
// Saving, Loading and Update
util::event<obs::source*> save; util::event<obs::source*> save;
util::event<obs::source*> load; util::event<obs::source*> load;
util::event<obs::source*> update_properties;
// Activate, Deactivate
util::event<obs::source*> activate; util::event<obs::source*> activate;
util::event<obs::source*> deactivate; util::event<obs::source*> deactivate;
// Show Hide
util::event<obs::source*> show; util::event<obs::source*> show;
util::event<obs::source*> hide; util::event<obs::source*> hide;
// Other
util::event<obs::source*, bool> enable; util::event<obs::source*, bool> enable;
util::event<obs::source*, std::string, std::string> rename;
util::event<obs::source*, long long> update_flags;
// Hotkeys (PtM, PtT)
util::event<obs::source*, bool> push_to_mute_changed; util::event<obs::source*, bool> push_to_mute_changed;
util::event<obs::source*, long long> push_to_mute_delay; util::event<obs::source*, long long> push_to_mute_delay;
util::event<obs::source*, bool> push_to_talk_changed; util::event<obs::source*, bool> push_to_talk_changed;
util::event<obs::source*, long long> push_to_talk_delay; util::event<obs::source*, long long> push_to_talk_delay;
util::event<obs::source*, std::string, std::string> rename; // Audio
util::event<obs::source*> update_properties;
util::event<obs::source*, long long> update_flags;
util::event<obs::source*, bool> mute; util::event<obs::source*, bool> mute;
util::event<obs::source*, double&> volume; util::event<obs::source*, double&> volume;
util::event<obs::source*, long long&> audio_sync; util::event<obs::source*, long long&> audio_sync;
util::event<obs::source*, long long&> audio_mixers; util::event<obs::source*, long long&> audio_mixers;
util::event<obs::source*, const audio_data*, bool> audio_data;
// Filters
util::event<obs::source*, obs_source_t*> filter_add; util::event<obs::source*, obs_source_t*> filter_add;
util::event<obs::source*, obs_source_t*> filter_remove; util::event<obs::source*, obs_source_t*> filter_remove;
util::event<obs::source*> reorder_filters; util::event<obs::source*> reorder_filters;
// Transition
util::event<obs::source*> transition_start; util::event<obs::source*> transition_start;
util::event<obs::source*> transition_video_stop; util::event<obs::source*> transition_video_stop;
util::event<obs::source*> transition_stop; util::event<obs::source*> transition_stop;