2015-06-26 03:10:39 -07:00
|
|
|
#include <media-io/audio-math.h>
|
2015-06-19 21:51:40 -07:00
|
|
|
#include <obs-module.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#define do_log(level, format, ...) \
|
|
|
|
blog(level, "[noise gate: '%s'] " format, \
|
|
|
|
obs_source_get_name(ng->context), ##__VA_ARGS__)
|
|
|
|
|
|
|
|
#define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__)
|
|
|
|
#define info(format, ...) do_log(LOG_INFO, format, ##__VA_ARGS__)
|
2019-06-22 22:13:45 -07:00
|
|
|
|
|
|
|
/* clang-format off */
|
2015-06-19 21:51:40 -07:00
|
|
|
|
|
|
|
#define S_OPEN_THRESHOLD "open_threshold"
|
|
|
|
#define S_CLOSE_THRESHOLD "close_threshold"
|
|
|
|
#define S_ATTACK_TIME "attack_time"
|
|
|
|
#define S_HOLD_TIME "hold_time"
|
|
|
|
#define S_RELEASE_TIME "release_time"
|
|
|
|
|
|
|
|
#define MT_ obs_module_text
|
|
|
|
#define TEXT_OPEN_THRESHOLD MT_("NoiseGate.OpenThreshold")
|
|
|
|
#define TEXT_CLOSE_THRESHOLD MT_("NoiseGate.CloseThreshold")
|
|
|
|
#define TEXT_ATTACK_TIME MT_("NoiseGate.AttackTime")
|
|
|
|
#define TEXT_HOLD_TIME MT_("NoiseGate.HoldTime")
|
|
|
|
#define TEXT_RELEASE_TIME MT_("NoiseGate.ReleaseTime")
|
|
|
|
|
2019-06-22 22:13:45 -07:00
|
|
|
/* clang-format on */
|
|
|
|
|
2015-06-19 21:51:40 -07:00
|
|
|
struct noise_gate_data {
|
|
|
|
obs_source_t *context;
|
|
|
|
|
|
|
|
float sample_rate_i;
|
|
|
|
size_t channels;
|
|
|
|
|
|
|
|
float open_threshold;
|
|
|
|
float close_threshold;
|
|
|
|
float decay_rate;
|
|
|
|
float attack_rate;
|
|
|
|
float release_rate;
|
|
|
|
float hold_time;
|
|
|
|
|
|
|
|
bool is_open;
|
|
|
|
float attenuation;
|
|
|
|
float level;
|
|
|
|
float held_time;
|
|
|
|
};
|
|
|
|
|
2017-09-25 14:22:55 -07:00
|
|
|
#define VOL_MIN -96.0
|
|
|
|
#define VOL_MAX 0.0
|
2015-06-19 21:51:40 -07:00
|
|
|
|
2015-09-16 01:30:51 -07:00
|
|
|
static const char *noise_gate_name(void *unused)
|
2015-06-19 21:51:40 -07:00
|
|
|
{
|
2015-09-16 01:30:51 -07:00
|
|
|
UNUSED_PARAMETER(unused);
|
2015-06-19 21:51:40 -07:00
|
|
|
return obs_module_text("NoiseGate");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void noise_gate_destroy(void *data)
|
|
|
|
{
|
|
|
|
struct noise_gate_data *ng = data;
|
|
|
|
bfree(ng);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline float ms_to_secf(int ms)
|
|
|
|
{
|
|
|
|
return (float)ms / 1000.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void noise_gate_update(void *data, obs_data_t *s)
|
|
|
|
{
|
|
|
|
struct noise_gate_data *ng = data;
|
|
|
|
float open_threshold_db;
|
|
|
|
float close_threshold_db;
|
|
|
|
float sample_rate;
|
|
|
|
int attack_time_ms;
|
|
|
|
int hold_time_ms;
|
|
|
|
int release_time_ms;
|
|
|
|
|
|
|
|
open_threshold_db = (float)obs_data_get_double(s, S_OPEN_THRESHOLD);
|
|
|
|
close_threshold_db = (float)obs_data_get_double(s, S_CLOSE_THRESHOLD);
|
|
|
|
attack_time_ms = (int)obs_data_get_int(s, S_ATTACK_TIME);
|
|
|
|
hold_time_ms = (int)obs_data_get_int(s, S_HOLD_TIME);
|
|
|
|
release_time_ms = (int)obs_data_get_int(s, S_RELEASE_TIME);
|
|
|
|
sample_rate = (float)audio_output_get_sample_rate(obs_get_audio());
|
|
|
|
|
|
|
|
ng->sample_rate_i = 1.0f / sample_rate;
|
|
|
|
ng->channels = audio_output_get_channels(obs_get_audio());
|
2015-06-26 03:10:39 -07:00
|
|
|
ng->open_threshold = db_to_mul(open_threshold_db);
|
|
|
|
ng->close_threshold = db_to_mul(close_threshold_db);
|
2015-06-19 21:51:40 -07:00
|
|
|
ng->attack_rate = 1.0f / (ms_to_secf(attack_time_ms) * sample_rate);
|
|
|
|
ng->release_rate = 1.0f / (ms_to_secf(release_time_ms) * sample_rate);
|
|
|
|
|
|
|
|
const float threshold_diff = ng->open_threshold - ng->close_threshold;
|
|
|
|
const float min_decay_period = (1.0f / 75.0f) * sample_rate;
|
|
|
|
|
|
|
|
ng->decay_rate = threshold_diff / min_decay_period;
|
|
|
|
ng->hold_time = ms_to_secf(hold_time_ms);
|
|
|
|
ng->is_open = false;
|
|
|
|
ng->attenuation = 0.0f;
|
|
|
|
ng->level = 0.0f;
|
|
|
|
ng->held_time = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *noise_gate_create(obs_data_t *settings, obs_source_t *filter)
|
|
|
|
{
|
|
|
|
struct noise_gate_data *ng = bzalloc(sizeof(*ng));
|
|
|
|
ng->context = filter;
|
|
|
|
noise_gate_update(ng, settings);
|
|
|
|
return ng;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct obs_audio_data *
|
|
|
|
noise_gate_filter_audio(void *data, struct obs_audio_data *audio)
|
|
|
|
{
|
|
|
|
struct noise_gate_data *ng = data;
|
|
|
|
|
libobs: Add surround sound audio support
(This commit also modifies the following modules: UI,
deps/media-playback, coreaudio-encoder, decklink, linux-alsa,
linux-pulseaudio, mac-capture, obs-ffmpeg, obs-filters, obs-libfdk,
obs-outputs, win-dshow, and win-wasapi)
Adds surround sound audio support to the core, core plugins, and user
interface.
Compatible streaming services: Twitch, FB 360 live
Compatible protocols: rtmp / mpeg-ts tcp udp
Compatible file formats: mkv mp4 ts (others untested)
Compatible codecs: ffmpeg aac, fdk_aac, CoreAudio aac,
opus, vorbis, pcm (others untested).
Tested streaming servers: wowza, nginx
HLS, mpeg-dash : surround passthrough
Html5 players tested with live surround:
videojs, mediaelement, viblast (hls+dash), hls.js
Decklink: on win32, swap channels order for 5.1 7.1
(due to different channel mapping on wav, mpeg, ffmpeg)
Audio filters: surround working.
Monitoring: surround working (win macOs linux (pulse-audio)).
VST: stereo plugins keep in general only the first two channels.
surround plugins should work (e.g. mcfx does).
OS: win, macOs, linux (alsa, pulse-audio).
Misc: larger audio bitrates unlocked to accommodate more channels
NB: mf-aac only supports mono and stereo + 5.1 on win 10
(not implemented due to lack of usefulness)
Closes jp9000/obs-studio#968
2017-05-27 02:15:54 +02:00
|
|
|
float **adata = (float **)audio->data;
|
2015-06-19 21:51:40 -07:00
|
|
|
const float close_threshold = ng->close_threshold;
|
|
|
|
const float open_threshold = ng->open_threshold;
|
|
|
|
const float sample_rate_i = ng->sample_rate_i;
|
|
|
|
const float release_rate = ng->release_rate;
|
|
|
|
const float attack_rate = ng->attack_rate;
|
|
|
|
const float decay_rate = ng->decay_rate;
|
|
|
|
const float hold_time = ng->hold_time;
|
|
|
|
const size_t channels = ng->channels;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < audio->frames; i++) {
|
libobs: Add surround sound audio support
(This commit also modifies the following modules: UI,
deps/media-playback, coreaudio-encoder, decklink, linux-alsa,
linux-pulseaudio, mac-capture, obs-ffmpeg, obs-filters, obs-libfdk,
obs-outputs, win-dshow, and win-wasapi)
Adds surround sound audio support to the core, core plugins, and user
interface.
Compatible streaming services: Twitch, FB 360 live
Compatible protocols: rtmp / mpeg-ts tcp udp
Compatible file formats: mkv mp4 ts (others untested)
Compatible codecs: ffmpeg aac, fdk_aac, CoreAudio aac,
opus, vorbis, pcm (others untested).
Tested streaming servers: wowza, nginx
HLS, mpeg-dash : surround passthrough
Html5 players tested with live surround:
videojs, mediaelement, viblast (hls+dash), hls.js
Decklink: on win32, swap channels order for 5.1 7.1
(due to different channel mapping on wav, mpeg, ffmpeg)
Audio filters: surround working.
Monitoring: surround working (win macOs linux (pulse-audio)).
VST: stereo plugins keep in general only the first two channels.
surround plugins should work (e.g. mcfx does).
OS: win, macOs, linux (alsa, pulse-audio).
Misc: larger audio bitrates unlocked to accommodate more channels
NB: mf-aac only supports mono and stereo + 5.1 on win 10
(not implemented due to lack of usefulness)
Closes jp9000/obs-studio#968
2017-05-27 02:15:54 +02:00
|
|
|
float cur_level = fabsf(adata[0][i]);
|
|
|
|
for (size_t j = 0; j < channels; j++) {
|
|
|
|
cur_level = fmaxf(cur_level, fabsf(adata[j][i]));
|
|
|
|
}
|
2015-06-19 21:51:40 -07:00
|
|
|
|
|
|
|
if (cur_level > open_threshold && !ng->is_open) {
|
|
|
|
ng->is_open = true;
|
|
|
|
}
|
|
|
|
if (ng->level < close_threshold && ng->is_open) {
|
|
|
|
ng->held_time = 0.0f;
|
|
|
|
ng->is_open = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ng->level = fmaxf(ng->level, cur_level) - decay_rate;
|
|
|
|
|
|
|
|
if (ng->is_open) {
|
|
|
|
ng->attenuation =
|
|
|
|
fminf(1.0f, ng->attenuation + attack_rate);
|
|
|
|
} else {
|
|
|
|
ng->held_time += sample_rate_i;
|
|
|
|
if (ng->held_time > hold_time) {
|
|
|
|
ng->attenuation = fmaxf(
|
|
|
|
0.0f, ng->attenuation - release_rate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t c = 0; c < channels; c++)
|
|
|
|
adata[c][i] *= ng->attenuation;
|
|
|
|
}
|
|
|
|
|
|
|
|
return audio;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void noise_gate_defaults(obs_data_t *s)
|
|
|
|
{
|
2017-09-25 14:22:55 -07:00
|
|
|
obs_data_set_default_double(s, S_OPEN_THRESHOLD, -26.0);
|
|
|
|
obs_data_set_default_double(s, S_CLOSE_THRESHOLD, -32.0);
|
2015-06-19 21:51:40 -07:00
|
|
|
obs_data_set_default_int(s, S_ATTACK_TIME, 25);
|
|
|
|
obs_data_set_default_int(s, S_HOLD_TIME, 200);
|
|
|
|
obs_data_set_default_int(s, S_RELEASE_TIME, 150);
|
|
|
|
}
|
|
|
|
|
|
|
|
static obs_properties_t *noise_gate_properties(void *data)
|
|
|
|
{
|
|
|
|
obs_properties_t *ppts = obs_properties_create();
|
2019-07-18 02:38:50 -05:00
|
|
|
obs_property_t *p;
|
|
|
|
|
|
|
|
p = obs_properties_add_float_slider(ppts, S_CLOSE_THRESHOLD,
|
|
|
|
TEXT_CLOSE_THRESHOLD, VOL_MIN,
|
|
|
|
VOL_MAX, 1.0);
|
|
|
|
obs_property_float_set_suffix(p, " dB");
|
|
|
|
p = obs_properties_add_float_slider(ppts, S_OPEN_THRESHOLD,
|
|
|
|
TEXT_OPEN_THRESHOLD, VOL_MIN,
|
|
|
|
VOL_MAX, 1.0);
|
|
|
|
obs_property_float_set_suffix(p, " dB");
|
|
|
|
p = obs_properties_add_int(ppts, S_ATTACK_TIME, TEXT_ATTACK_TIME, 0,
|
|
|
|
10000, 1);
|
|
|
|
obs_property_int_set_suffix(p, " ms");
|
|
|
|
p = obs_properties_add_int(ppts, S_HOLD_TIME, TEXT_HOLD_TIME, 0, 10000,
|
|
|
|
1);
|
|
|
|
obs_property_int_set_suffix(p, " ms");
|
|
|
|
p = obs_properties_add_int(ppts, S_RELEASE_TIME, TEXT_RELEASE_TIME, 0,
|
|
|
|
10000, 1);
|
|
|
|
obs_property_int_set_suffix(p, " ms");
|
2015-06-19 21:51:40 -07:00
|
|
|
|
|
|
|
UNUSED_PARAMETER(data);
|
|
|
|
return ppts;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct obs_source_info noise_gate_filter = {
|
|
|
|
.id = "noise_gate_filter",
|
|
|
|
.type = OBS_SOURCE_TYPE_FILTER,
|
|
|
|
.output_flags = OBS_SOURCE_AUDIO,
|
|
|
|
.get_name = noise_gate_name,
|
|
|
|
.create = noise_gate_create,
|
|
|
|
.destroy = noise_gate_destroy,
|
|
|
|
.update = noise_gate_update,
|
|
|
|
.filter_audio = noise_gate_filter_audio,
|
|
|
|
.get_defaults = noise_gate_defaults,
|
|
|
|
.get_properties = noise_gate_properties,
|
|
|
|
};
|