2712014744
Happy new year to the wonderful Godot community!
964 lines
25 KiB
C++
964 lines
25 KiB
C++
/*************************************************************************/
|
|
/* audio_server_sw.cpp */
|
|
/*************************************************************************/
|
|
/* This file is part of: */
|
|
/* GODOT ENGINE */
|
|
/* https://godotengine.org */
|
|
/*************************************************************************/
|
|
/* Copyright (c) 2007-2018 Juan Linietsky, Ariel Manzur. */
|
|
/* Copyright (c) 2014-2018 Godot Engine contributors (cf. AUTHORS.md) */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
/* a copy of this software and associated documentation files (the */
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
/* the following conditions: */
|
|
/* */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/*************************************************************************/
|
|
#include "audio_server_sw.h"
|
|
#include "globals.h"
|
|
#include "os/os.h"
|
|
|
|
struct _AudioDriverLock {
|
|
|
|
_AudioDriverLock() {
|
|
if (AudioDriverSW::get_singleton()) AudioDriverSW::get_singleton()->lock();
|
|
}
|
|
~_AudioDriverLock() {
|
|
if (AudioDriverSW::get_singleton()) AudioDriverSW::get_singleton()->unlock();
|
|
}
|
|
};
|
|
|
|
#define AUDIO_LOCK _AudioDriverLock _adlock;
|
|
|
|
AudioMixer *AudioServerSW::get_mixer() {
|
|
|
|
return mixer;
|
|
}
|
|
|
|
/* CALLBACKS */
|
|
|
|
void AudioServerSW::audio_mixer_chunk_callback(int p_frames) {
|
|
/*
|
|
for(List<Stream*>::Element *E=event_streams.front();E;E=E->next()) {
|
|
|
|
if (E->get()->active)
|
|
E->get()->audio_stream->mix(NULL,p_frames);
|
|
}
|
|
*/
|
|
}
|
|
|
|
void AudioServerSW::_mixer_callback(void *p_udata) {
|
|
|
|
AudioServerSW *self = (AudioServerSW *)p_udata;
|
|
for (List<Stream *>::Element *E = self->active_audio_streams.front(); E; E = E->next()) {
|
|
|
|
if (!E->get()->active)
|
|
continue;
|
|
|
|
EventStream *es = E->get()->event_stream;
|
|
if (!es)
|
|
continue;
|
|
|
|
es->update(self->mixer_step_usecs);
|
|
}
|
|
}
|
|
|
|
void AudioServerSW::driver_process_chunk(int p_frames, int32_t *p_buffer) {
|
|
|
|
int samples = p_frames * internal_buffer_channels;
|
|
|
|
for (int i = 0; i < samples; i++) {
|
|
internal_buffer[i] = 0;
|
|
}
|
|
|
|
while (voice_rb.commands_left()) {
|
|
|
|
VoiceRBSW::Command cmd = voice_rb.pop_command();
|
|
|
|
if (cmd.type == VoiceRBSW::Command::CMD_CHANGE_ALL_FX_VOLUMES) {
|
|
|
|
SelfList<Voice> *al = active_list.first();
|
|
while (al) {
|
|
|
|
Voice *v = al->self();
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL) {
|
|
mixer->channel_set_volume(v->channel, v->volume * fx_volume_scale);
|
|
}
|
|
al = al->next();
|
|
}
|
|
|
|
continue;
|
|
}
|
|
if (!voice_owner.owns(cmd.voice))
|
|
continue;
|
|
|
|
Voice *v = voice_owner.get(cmd.voice);
|
|
|
|
switch (cmd.type) {
|
|
case VoiceRBSW::Command::CMD_NONE: {
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_PLAY: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_free(v->channel);
|
|
|
|
RID sample = cmd.play.sample;
|
|
if (!sample_manager->is_sample(sample))
|
|
continue;
|
|
|
|
v->channel = mixer->channel_alloc(sample);
|
|
v->volume = 1.0;
|
|
mixer->channel_set_volume(v->channel, fx_volume_scale);
|
|
if (v->channel == AudioMixer::INVALID_CHANNEL) {
|
|
#ifdef AUDIO_DEBUG
|
|
WARN_PRINT("AUDIO: all channels used, failed to allocate voice");
|
|
#endif
|
|
v->active = false;
|
|
break; // no voices left?
|
|
}
|
|
|
|
v->active = true; // this kind of ensures it works
|
|
if (!v->active_item.in_list())
|
|
active_list.add(&v->active_item);
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_STOP: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL) {
|
|
mixer->channel_free(v->channel);
|
|
if (v->active_item.in_list()) {
|
|
active_list.remove(&v->active_item);
|
|
}
|
|
}
|
|
v->active = false;
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_VOLUME: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL) {
|
|
v->volume = cmd.volume.volume;
|
|
mixer->channel_set_volume(v->channel, cmd.volume.volume * fx_volume_scale);
|
|
}
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_PAN: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_pan(v->channel, cmd.pan.pan, cmd.pan.depth, cmd.pan.height);
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_FILTER: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_filter(v->channel, (AudioMixer::FilterType)cmd.filter.type, cmd.filter.cutoff, cmd.filter.resonance, cmd.filter.gain);
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_CHORUS: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_chorus(v->channel, cmd.chorus.send);
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_REVERB: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_reverb(v->channel, (AudioMixer::ReverbRoomType)cmd.reverb.room, cmd.reverb.send);
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_MIX_RATE: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_mix_rate(v->channel, cmd.mix_rate.mix_rate);
|
|
|
|
} break;
|
|
case VoiceRBSW::Command::CMD_SET_POSITIONAL: {
|
|
|
|
if (v->channel != AudioMixer::INVALID_CHANNEL)
|
|
mixer->channel_set_positional(v->channel, cmd.positional.positional);
|
|
|
|
} break;
|
|
default: {}
|
|
}
|
|
}
|
|
|
|
mixer->mix(internal_buffer, p_frames);
|
|
//uint64_t stepsize=mixer->get_step_usecs();
|
|
|
|
for (List<Stream *>::Element *E = active_audio_streams.front(); E; E = E->next()) {
|
|
|
|
ERR_CONTINUE(!E->get()->active); // bug?
|
|
|
|
AudioStream *as = E->get()->audio_stream;
|
|
if (!as)
|
|
continue;
|
|
|
|
int channels = as->get_channel_count();
|
|
if (channels == 0)
|
|
continue; // does not want mix
|
|
if (!as->mix(stream_buffer, p_frames))
|
|
continue; //nothing was mixed!!
|
|
|
|
int32_t stream_vol_scale = (stream_volume * stream_volume_scale * E->get()->volume_scale) * (1 << STREAM_SCALE_BITS);
|
|
|
|
#define STRSCALE(m_val) (((m_val >> STREAM_SCALE_BITS) * stream_vol_scale) >> 8)
|
|
switch (internal_buffer_channels) {
|
|
|
|
case 2: {
|
|
|
|
switch (channels) {
|
|
case 1: {
|
|
|
|
for (int i = 0; i < p_frames; i++) {
|
|
|
|
internal_buffer[(i << 1) + 0] += STRSCALE(stream_buffer[i]);
|
|
internal_buffer[(i << 1) + 1] += STRSCALE(stream_buffer[i]);
|
|
}
|
|
} break;
|
|
case 2: {
|
|
|
|
for (int i = 0; i < p_frames * 2; i++) {
|
|
|
|
internal_buffer[i] += STRSCALE(stream_buffer[i]);
|
|
}
|
|
} break;
|
|
case 4: {
|
|
|
|
for (int i = 0; i < p_frames; i++) {
|
|
|
|
internal_buffer[(i << 2) + 0] += STRSCALE((stream_buffer[(i << 2) + 0] + stream_buffer[(i << 2) + 2]) >> 1);
|
|
internal_buffer[(i << 2) + 1] += STRSCALE((stream_buffer[(i << 2) + 1] + stream_buffer[(i << 2) + 3]) >> 1);
|
|
}
|
|
} break;
|
|
}
|
|
break;
|
|
|
|
} break;
|
|
case 4: {
|
|
|
|
switch (channels) {
|
|
case 1: {
|
|
|
|
for (int i = 0; i < p_frames; i++) {
|
|
|
|
internal_buffer[(i << 2) + 0] += STRSCALE(stream_buffer[i]);
|
|
internal_buffer[(i << 2) + 1] += STRSCALE(stream_buffer[i]);
|
|
internal_buffer[(i << 2) + 2] += STRSCALE(stream_buffer[i]);
|
|
internal_buffer[(i << 2) + 3] += STRSCALE(stream_buffer[i]);
|
|
}
|
|
} break;
|
|
case 2: {
|
|
|
|
for (int i = 0; i < p_frames * 2; i++) {
|
|
|
|
internal_buffer[(i << 2) + 0] += STRSCALE(stream_buffer[(i << 1) + 0]);
|
|
internal_buffer[(i << 2) + 1] += STRSCALE(stream_buffer[(i << 1) + 1]);
|
|
internal_buffer[(i << 2) + 2] += STRSCALE(stream_buffer[(i << 1) + 0]);
|
|
internal_buffer[(i << 2) + 3] += STRSCALE(stream_buffer[(i << 1) + 1]);
|
|
}
|
|
} break;
|
|
case 4: {
|
|
|
|
for (int i = 0; i < p_frames * 4; i++) {
|
|
internal_buffer[i] += STRSCALE(stream_buffer[i]);
|
|
}
|
|
} break;
|
|
}
|
|
break;
|
|
|
|
} break;
|
|
case 6: {
|
|
|
|
} break;
|
|
}
|
|
|
|
#undef STRSCALE
|
|
}
|
|
|
|
SelfList<Voice> *activeE = active_list.first();
|
|
while (activeE) {
|
|
|
|
SelfList<Voice> *activeN = activeE->next();
|
|
if (activeE->self()->channel == AudioMixer::INVALID_CHANNEL || !mixer->channel_is_valid(activeE->self()->channel)) {
|
|
|
|
active_list.remove(activeE);
|
|
activeE->self()->active = false;
|
|
}
|
|
activeE = activeN;
|
|
}
|
|
|
|
uint32_t peak = 0;
|
|
for (int i = 0; i < samples; i++) {
|
|
//clamp to (1<<24) using branchless code
|
|
int32_t in = internal_buffer[i];
|
|
#ifdef DEBUG_ENABLED
|
|
{
|
|
int mask = (in >> (32 - 1));
|
|
uint32_t p = (in + mask) ^ mask;
|
|
if (p > peak)
|
|
peak = p;
|
|
}
|
|
#endif
|
|
int32_t lo = -0x800000, hi = 0x7FFFFF;
|
|
lo -= in;
|
|
hi -= in;
|
|
in += (lo & ((lo < 0) - 1)) + (hi & ((hi > 0) - 1));
|
|
p_buffer[i] = in << 8;
|
|
}
|
|
|
|
if (peak > max_peak)
|
|
max_peak = peak;
|
|
}
|
|
|
|
void AudioServerSW::driver_process(int p_frames, int32_t *p_buffer) {
|
|
|
|
_output_delay = p_frames / double(AudioDriverSW::get_singleton()->get_mix_rate());
|
|
//process in chunks to make sure to never process more than INTERNAL_BUFFER_SIZE
|
|
int todo = p_frames;
|
|
while (todo) {
|
|
|
|
int tomix = MIN(todo, INTERNAL_BUFFER_SIZE);
|
|
driver_process_chunk(tomix, p_buffer);
|
|
p_buffer += tomix;
|
|
todo -= tomix;
|
|
}
|
|
}
|
|
|
|
/* SAMPLE API */
|
|
|
|
RID AudioServerSW::sample_create(SampleFormat p_format, bool p_stereo, int p_length) {
|
|
|
|
AUDIO_LOCK
|
|
|
|
return sample_manager->sample_create(p_format, p_stereo, p_length);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_description(RID p_sample, const String &p_description) {
|
|
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_description(p_sample, p_description);
|
|
}
|
|
String AudioServerSW::sample_get_description(RID p_sample) const {
|
|
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_description(p_sample);
|
|
}
|
|
|
|
AS::SampleFormat AudioServerSW::sample_get_format(RID p_sample) const {
|
|
//AUDIO_LOCK
|
|
return sample_manager->sample_get_format(p_sample);
|
|
}
|
|
bool AudioServerSW::sample_is_stereo(RID p_sample) const {
|
|
//AUDIO_LOCK
|
|
return sample_manager->sample_is_stereo(p_sample);
|
|
}
|
|
int AudioServerSW::sample_get_length(RID p_sample) const {
|
|
///AUDIO_LOCK
|
|
return sample_manager->sample_get_length(p_sample);
|
|
}
|
|
|
|
const void *AudioServerSW::sample_get_data_ptr(RID p_sample) const {
|
|
///AUDIO_LOCK
|
|
return sample_manager->sample_get_data_ptr(p_sample);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_data(RID p_sample, const DVector<uint8_t> &p_buffer) {
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_data(p_sample, p_buffer);
|
|
}
|
|
DVector<uint8_t> AudioServerSW::sample_get_data(RID p_sample) const {
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_data(p_sample);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_mix_rate(RID p_sample, int p_rate) {
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_mix_rate(p_sample, p_rate);
|
|
}
|
|
int AudioServerSW::sample_get_mix_rate(RID p_sample) const {
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_mix_rate(p_sample);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_loop_format(RID p_sample, SampleLoopFormat p_format) {
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_loop_format(p_sample, p_format);
|
|
}
|
|
AS::SampleLoopFormat AudioServerSW::sample_get_loop_format(RID p_sample) const {
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_loop_format(p_sample);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_loop_begin(RID p_sample, int p_pos) {
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_loop_begin(p_sample, p_pos);
|
|
}
|
|
int AudioServerSW::sample_get_loop_begin(RID p_sample) const {
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_loop_begin(p_sample);
|
|
}
|
|
|
|
void AudioServerSW::sample_set_loop_end(RID p_sample, int p_pos) {
|
|
AUDIO_LOCK
|
|
sample_manager->sample_set_loop_end(p_sample, p_pos);
|
|
}
|
|
int AudioServerSW::sample_get_loop_end(RID p_sample) const {
|
|
AUDIO_LOCK
|
|
return sample_manager->sample_get_loop_end(p_sample);
|
|
}
|
|
|
|
/* VOICE API */
|
|
|
|
RID AudioServerSW::voice_create() {
|
|
|
|
Voice *v = memnew(Voice);
|
|
v->channel = AudioMixer::INVALID_CHANNEL;
|
|
|
|
AUDIO_LOCK
|
|
return voice_owner.make_rid(v);
|
|
}
|
|
void AudioServerSW::voice_play(RID p_voice, RID p_sample) {
|
|
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND(!v);
|
|
v->active = true; // force actvive (will be disabled later i gues..)
|
|
|
|
//stop old, start new
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_PLAY;
|
|
cmd.voice = p_voice;
|
|
cmd.play.sample = p_sample;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
|
|
void AudioServerSW::voice_set_volume(RID p_voice, float p_volume) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_VOLUME;
|
|
cmd.voice = p_voice;
|
|
cmd.volume.volume = p_volume;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_pan(RID p_voice, float p_pan, float p_depth, float p_height) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_PAN;
|
|
cmd.voice = p_voice;
|
|
cmd.pan.pan = p_pan;
|
|
cmd.pan.depth = p_depth;
|
|
cmd.pan.height = p_height;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_filter(RID p_voice, FilterType p_type, float p_cutoff, float p_resonance, float p_gain) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_FILTER;
|
|
cmd.voice = p_voice;
|
|
cmd.filter.type = p_type;
|
|
cmd.filter.cutoff = p_cutoff;
|
|
cmd.filter.resonance = p_resonance;
|
|
cmd.filter.gain = p_gain;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_chorus(RID p_voice, float p_chorus) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_CHORUS;
|
|
cmd.voice = p_voice;
|
|
cmd.chorus.send = p_chorus;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_reverb(RID p_voice, ReverbRoomType p_room_type, float p_reverb) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_REVERB;
|
|
cmd.voice = p_voice;
|
|
cmd.reverb.room = p_room_type;
|
|
cmd.reverb.send = p_reverb;
|
|
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_mix_rate(RID p_voice, int p_mix_rate) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_MIX_RATE;
|
|
cmd.voice = p_voice;
|
|
cmd.mix_rate.mix_rate = p_mix_rate;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
void AudioServerSW::voice_set_positional(RID p_voice, bool p_positional) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_SET_POSITIONAL;
|
|
cmd.voice = p_voice;
|
|
cmd.positional.positional = p_positional;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
|
|
float AudioServerSW::voice_get_volume(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_volume(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_pan(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_pan(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_pan_depth(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_pan_depth(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_pan_height(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_pan_height(v->channel);
|
|
}
|
|
AS::FilterType AudioServerSW::voice_get_filter_type(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, AS::FILTER_NONE);
|
|
|
|
return (AS::FilterType)mixer->channel_get_filter_type(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_filter_cutoff(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_filter_cutoff(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_filter_resonance(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_filter_resonance(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_chorus(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_chorus(v->channel);
|
|
}
|
|
AS::ReverbRoomType AudioServerSW::voice_get_reverb_type(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, REVERB_SMALL);
|
|
|
|
return (AS::ReverbRoomType)mixer->channel_get_reverb_type(v->channel);
|
|
}
|
|
float AudioServerSW::voice_get_reverb(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_reverb(v->channel);
|
|
}
|
|
|
|
int AudioServerSW::voice_get_mix_rate(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_get_mix_rate(v->channel);
|
|
}
|
|
bool AudioServerSW::voice_is_positional(RID p_voice) const {
|
|
|
|
AUDIO_LOCK
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, 0);
|
|
|
|
return mixer->channel_is_positional(v->channel);
|
|
}
|
|
|
|
void AudioServerSW::voice_stop(RID p_voice) {
|
|
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_STOP;
|
|
cmd.voice = p_voice;
|
|
voice_rb.push_command(cmd);
|
|
|
|
//return mixer->channel_free( v->channel );
|
|
}
|
|
|
|
bool AudioServerSW::voice_is_active(RID p_voice) const {
|
|
|
|
Voice *v = voice_owner.get(p_voice);
|
|
ERR_FAIL_COND_V(!v, false);
|
|
return v->active;
|
|
}
|
|
|
|
/* STREAM API */
|
|
|
|
RID AudioServerSW::audio_stream_create(AudioStream *p_stream) {
|
|
|
|
AUDIO_LOCK
|
|
Stream *s = memnew(Stream);
|
|
s->audio_stream = p_stream;
|
|
s->event_stream = NULL;
|
|
s->active = false;
|
|
s->E = NULL;
|
|
s->volume_scale = 1.0;
|
|
p_stream->set_mix_rate(AudioDriverSW::get_singleton()->get_mix_rate());
|
|
|
|
return stream_owner.make_rid(s);
|
|
}
|
|
|
|
RID AudioServerSW::event_stream_create(EventStream *p_stream) {
|
|
|
|
AUDIO_LOCK
|
|
Stream *s = memnew(Stream);
|
|
s->audio_stream = NULL;
|
|
s->event_stream = p_stream;
|
|
s->active = false;
|
|
s->E = NULL;
|
|
s->volume_scale = 1.0;
|
|
//p_stream->set_mix_rate(AudioDriverSW::get_singleton()->get_mix_rate());
|
|
|
|
return stream_owner.make_rid(s);
|
|
}
|
|
|
|
void AudioServerSW::stream_set_active(RID p_stream, bool p_active) {
|
|
|
|
Stream *s = stream_owner.get(p_stream);
|
|
ERR_FAIL_COND(!s);
|
|
_THREAD_SAFE_METHOD_
|
|
|
|
if (s->active == p_active)
|
|
return;
|
|
AUDIO_LOCK;
|
|
s->active = p_active;
|
|
if (p_active)
|
|
s->E = active_audio_streams.push_back(s);
|
|
else {
|
|
active_audio_streams.erase(s->E);
|
|
s->E = NULL;
|
|
}
|
|
}
|
|
|
|
bool AudioServerSW::stream_is_active(RID p_stream) const {
|
|
|
|
Stream *s = stream_owner.get(p_stream);
|
|
ERR_FAIL_COND_V(!s, false);
|
|
return s->active;
|
|
}
|
|
|
|
void AudioServerSW::stream_set_volume_scale(RID p_stream, float p_scale) {
|
|
|
|
Stream *s = stream_owner.get(p_stream);
|
|
ERR_FAIL_COND(!s);
|
|
s->volume_scale = p_scale;
|
|
}
|
|
|
|
float AudioServerSW::stream_set_volume_scale(RID p_stream) const {
|
|
|
|
Stream *s = stream_owner.get(p_stream);
|
|
ERR_FAIL_COND_V(!s, 0);
|
|
return s->volume_scale;
|
|
}
|
|
|
|
void AudioServerSW::free(RID p_id) {
|
|
|
|
if (voice_owner.owns(p_id)) {
|
|
|
|
Voice *v = voice_owner.get(p_id);
|
|
AUDIO_LOCK
|
|
mixer->channel_free(v->channel);
|
|
voice_owner.free(p_id);
|
|
memdelete(v);
|
|
|
|
} else if (stream_owner.owns(p_id)) {
|
|
|
|
Stream *s = stream_owner.get(p_id);
|
|
|
|
if (s->active) {
|
|
stream_set_active(p_id, false);
|
|
}
|
|
|
|
memdelete(s);
|
|
stream_owner.free(p_id);
|
|
|
|
} else if (sample_manager->is_sample(p_id)) {
|
|
|
|
AUDIO_LOCK
|
|
sample_manager->free(p_id);
|
|
}
|
|
}
|
|
|
|
void AudioServerSW::_thread_func(void *self) {
|
|
|
|
Thread::set_name("AudioServerSW");
|
|
|
|
AudioServerSW *as = (AudioServerSW *)self;
|
|
|
|
while (!as->exit_update_thread) {
|
|
as->_update_streams(true);
|
|
if (OS::get_singleton())
|
|
OS::get_singleton()->delay_usec(5000);
|
|
}
|
|
}
|
|
|
|
void AudioServerSW::init() {
|
|
|
|
int latency = GLOBAL_DEF("audio/mixer_latency", 10);
|
|
internal_buffer_channels = 2; // read from driver
|
|
internal_buffer = memnew_arr(int32_t, INTERNAL_BUFFER_SIZE * internal_buffer_channels);
|
|
stream_buffer = memnew_arr(int32_t, INTERNAL_BUFFER_SIZE * 4); //max 4 channels
|
|
AudioMixerSW::MixChannels mix_chans = AudioMixerSW::MIX_STEREO;
|
|
|
|
switch (AudioDriverSW::get_singleton()->get_output_format()) {
|
|
|
|
case AudioDriverSW::OUTPUT_MONO:
|
|
case AudioDriverSW::OUTPUT_STEREO:
|
|
mix_chans = AudioMixerSW::MIX_STEREO;
|
|
break;
|
|
case AudioDriverSW::OUTPUT_QUAD:
|
|
case AudioDriverSW::OUTPUT_5_1:
|
|
mix_chans = AudioMixerSW::MIX_QUAD;
|
|
break;
|
|
}
|
|
|
|
mixer = memnew(AudioMixerSW(sample_manager, latency, AudioDriverSW::get_singleton()->get_mix_rate(), mix_chans, mixer_use_fx, mixer_interp, _mixer_callback, this));
|
|
mixer_step_usecs = mixer->get_step_usecs();
|
|
|
|
_output_delay = 0;
|
|
|
|
stream_volume = 0.3;
|
|
// start the audio driver
|
|
if (AudioDriverSW::get_singleton())
|
|
AudioDriverSW::get_singleton()->start();
|
|
|
|
#ifndef NO_THREADS
|
|
exit_update_thread = false;
|
|
thread = Thread::create(_thread_func, this);
|
|
#endif
|
|
}
|
|
|
|
void AudioServerSW::finish() {
|
|
|
|
#ifndef NO_THREADS
|
|
exit_update_thread = true;
|
|
Thread::wait_to_finish(thread);
|
|
memdelete(thread);
|
|
#endif
|
|
|
|
if (AudioDriverSW::get_singleton())
|
|
AudioDriverSW::get_singleton()->finish();
|
|
|
|
memdelete_arr(internal_buffer);
|
|
memdelete_arr(stream_buffer);
|
|
memdelete(mixer);
|
|
}
|
|
|
|
void AudioServerSW::_update_streams(bool p_thread) {
|
|
|
|
_THREAD_SAFE_METHOD_
|
|
for (List<Stream *>::Element *E = active_audio_streams.front(); E;) { //stream might be removed durnig this callback
|
|
|
|
List<Stream *>::Element *N = E->next();
|
|
|
|
if (E->get()->audio_stream && p_thread == E->get()->audio_stream->can_update_mt())
|
|
E->get()->audio_stream->update();
|
|
|
|
E = N;
|
|
}
|
|
}
|
|
|
|
void AudioServerSW::update() {
|
|
|
|
_update_streams(false);
|
|
#ifdef NO_THREADS
|
|
|
|
_update_streams(true);
|
|
#endif
|
|
}
|
|
|
|
void AudioServerSW::lock() {
|
|
|
|
AudioDriverSW::get_singleton()->lock();
|
|
}
|
|
|
|
void AudioServerSW::unlock() {
|
|
AudioDriverSW::get_singleton()->unlock();
|
|
}
|
|
|
|
int AudioServerSW::get_default_mix_rate() const {
|
|
|
|
return AudioDriverSW::get_singleton()->get_mix_rate();
|
|
}
|
|
int AudioServerSW::get_default_channel_count() const {
|
|
return internal_buffer_channels;
|
|
}
|
|
|
|
void AudioServerSW::set_mixer_params(AudioMixerSW::InterpolationType p_interp, bool p_use_fx) {
|
|
|
|
mixer_interp = p_interp;
|
|
mixer_use_fx = p_use_fx;
|
|
}
|
|
|
|
void AudioServerSW::set_stream_global_volume_scale(float p_volume) {
|
|
|
|
stream_volume_scale = p_volume;
|
|
}
|
|
|
|
float AudioServerSW::get_stream_global_volume_scale() const {
|
|
|
|
return stream_volume_scale;
|
|
}
|
|
|
|
void AudioServerSW::set_fx_global_volume_scale(float p_volume) {
|
|
|
|
fx_volume_scale = p_volume;
|
|
//mixer->set_mixer_volume(fx_volume_scale);
|
|
VoiceRBSW::Command cmd;
|
|
cmd.type = VoiceRBSW::Command::CMD_CHANGE_ALL_FX_VOLUMES;
|
|
cmd.voice = RID();
|
|
cmd.volume.volume = p_volume;
|
|
voice_rb.push_command(cmd);
|
|
}
|
|
|
|
float AudioServerSW::get_fx_global_volume_scale() const {
|
|
|
|
return fx_volume_scale;
|
|
}
|
|
|
|
void AudioServerSW::set_event_voice_global_volume_scale(float p_volume) {
|
|
|
|
event_voice_volume_scale = p_volume;
|
|
//mixer->set_mixer_volume(event_voice_volume_scale);
|
|
}
|
|
|
|
float AudioServerSW::get_event_voice_global_volume_scale() const {
|
|
|
|
return event_voice_volume_scale;
|
|
}
|
|
|
|
double AudioServerSW::get_output_delay() const {
|
|
|
|
return _output_delay + AudioDriverSW::get_singleton()->get_latency();
|
|
}
|
|
|
|
double AudioServerSW::get_mix_time() const {
|
|
|
|
return AudioDriverSW::get_singleton()->get_mix_time();
|
|
}
|
|
|
|
uint32_t AudioServerSW::read_output_peak() const {
|
|
|
|
uint32_t val = max_peak;
|
|
uint32_t *p = (uint32_t *)&max_peak;
|
|
*p = 0;
|
|
return val;
|
|
}
|
|
|
|
AudioServerSW::AudioServerSW(SampleManagerSW *p_sample_manager) {
|
|
|
|
sample_manager = p_sample_manager;
|
|
String interp = GLOBAL_DEF("audio/mixer_interp", "linear");
|
|
Globals::get_singleton()->set_custom_property_info("audio/mixer_interp", PropertyInfo(Variant::STRING, "audio/mixer_interp", PROPERTY_HINT_ENUM, "raw,linear,cubic"));
|
|
if (interp == "raw")
|
|
mixer_interp = AudioMixerSW::INTERPOLATION_RAW;
|
|
else if (interp == "cubic")
|
|
mixer_interp = AudioMixerSW::INTERPOLATION_CUBIC;
|
|
else
|
|
mixer_interp = AudioMixerSW::INTERPOLATION_LINEAR;
|
|
mixer_use_fx = GLOBAL_DEF("audio/use_chorus_reverb", true);
|
|
stream_volume_scale = GLOBAL_DEF("audio/stream_volume_scale", 1.0);
|
|
fx_volume_scale = GLOBAL_DEF("audio/fx_volume_scale", 1.0);
|
|
event_voice_volume_scale = GLOBAL_DEF("audio/event_voice_volume_scale", 0.5);
|
|
max_peak = 0;
|
|
}
|
|
|
|
AudioServerSW::~AudioServerSW() {
|
|
}
|
|
|
|
AudioDriverSW *AudioDriverSW::singleton = NULL;
|
|
AudioDriverSW *AudioDriverSW::get_singleton() {
|
|
|
|
return singleton;
|
|
}
|
|
|
|
void AudioDriverSW::set_singleton() {
|
|
|
|
singleton = this;
|
|
}
|
|
|
|
void AudioDriverSW::audio_server_process(int p_frames, int32_t *p_buffer, bool p_update_mix_time) {
|
|
|
|
AudioServerSW *audio_server = static_cast<AudioServerSW *>(AudioServer::get_singleton());
|
|
if (p_update_mix_time)
|
|
update_mix_time(p_frames);
|
|
audio_server->driver_process(p_frames, p_buffer);
|
|
}
|
|
|
|
void AudioDriverSW::update_mix_time(int p_frames) {
|
|
|
|
_mix_amount += p_frames;
|
|
_last_mix_time = OS::get_singleton()->get_ticks_usec();
|
|
}
|
|
|
|
double AudioDriverSW::get_mix_time() const {
|
|
|
|
double total = (OS::get_singleton()->get_ticks_usec() - _last_mix_time) / 1000000.0;
|
|
total += _mix_amount / (double)get_mix_rate();
|
|
return total;
|
|
}
|
|
|
|
AudioDriverSW::AudioDriverSW() {
|
|
|
|
_last_mix_time = 0;
|
|
_mix_amount = 0;
|
|
}
|
|
|
|
AudioDriverSW *AudioDriverManagerSW::drivers[MAX_DRIVERS];
|
|
int AudioDriverManagerSW::driver_count = 0;
|
|
|
|
void AudioDriverManagerSW::add_driver(AudioDriverSW *p_driver) {
|
|
|
|
ERR_FAIL_COND(driver_count >= MAX_DRIVERS);
|
|
drivers[driver_count++] = p_driver;
|
|
}
|
|
|
|
int AudioDriverManagerSW::get_driver_count() {
|
|
|
|
return driver_count;
|
|
}
|
|
AudioDriverSW *AudioDriverManagerSW::get_driver(int p_driver) {
|
|
|
|
ERR_FAIL_INDEX_V(p_driver, driver_count, NULL);
|
|
return drivers[p_driver];
|
|
}
|