|
- // Copyright 2014 Olivier Gillet.
- //
- // Author: Olivier Gillet (ol.gillet@gmail.com)
- //
- // 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.
- //
- // See http://creativecommons.org/licenses/MIT/ for more information.
- //
- // -----------------------------------------------------------------------------
- //
- // Single grain synthesis.
-
- #ifndef CLOUDS_DSP_GRAIN_H_
- #define CLOUDS_DSP_GRAIN_H_
-
- #include "stmlib/stmlib.h"
-
- #include "stmlib/dsp/dsp.h"
-
- #include "clouds/dsp/audio_buffer.h"
-
- #include "clouds/resources.h"
-
- namespace clouds {
-
- enum GrainQuality {
- GRAIN_QUALITY_LOW,
- GRAIN_QUALITY_MEDIUM,
- GRAIN_QUALITY_HIGH
- };
-
- class Grain {
- public:
- Grain() { }
- ~Grain() { }
-
- void Init() {
- active_ = false;
- envelope_phase_ = 2.0f;
- }
-
- void Start(
- int32_t pre_delay,
- int32_t buffer_size,
- int32_t start,
- int32_t width,
- int32_t phase_increment,
- float window_shape,
- float gain_l,
- float gain_r,
- GrainQuality recommended_quality) {
- pre_delay_ = pre_delay;
- width_ = width;
- first_sample_ = (start + buffer_size) % buffer_size;
- phase_increment_ = phase_increment;
- phase_ = 0;
- envelope_phase_ = 0.0f;
- envelope_phase_increment_ = 2.0f / static_cast<float>(width);
- if (window_shape >= 0.5f) {
- envelope_smoothness_ = (window_shape - 0.5f) * 2.0f;
- envelope_slope_ = 0.0f;
- } else {
- envelope_smoothness_ = 0.0f;
- envelope_slope_ = 0.5f / (window_shape + 0.01f);
- }
- active_ = true;
- gain_l_ = gain_l;
- gain_r_ = gain_r;
- recommended_quality_ = recommended_quality;
- }
-
- template<bool use_lut_for_envelope, GrainQuality quality>
- inline void RenderEnvelope(float* destination, size_t size) {
- const float increment = envelope_phase_increment_;
- const float smoothness = envelope_smoothness_;
- const float slope = envelope_slope_;
-
- float phase = envelope_phase_;
- while (size--) {
- float gain = phase;
- gain = gain >= 1.0f ? 2.0f - gain : gain;
- if (use_lut_for_envelope) {
- if (quality == GRAIN_QUALITY_HIGH) {
- float window = 0.0f;
- window = stmlib::Interpolate(lut_window, gain, 4096.0f);
- gain += smoothness * (window - gain);
- }
- } else {
- if (quality >= GRAIN_QUALITY_MEDIUM) {
- gain *= slope;
- if (gain >= 1.0f) gain = 1.0f;
- }
- }
- phase += increment;
- if (phase >= 2.0f) {
- *destination = -1.0f;
- break;
- }
- *destination++ = gain;
- }
- envelope_phase_ = phase;
- }
-
- template<int32_t num_channels, GrainQuality quality, Resolution resolution>
- inline void OverlapAdd(
- const AudioBuffer<resolution>* buffer,
- float* destination,
- float* envelope,
- size_t size) {
- if (!active_) {
- return;
- }
- // Rendering is done on 32-sample long blocks. The pre-delay allows grains
- // to start at arbitrary samples within a block, rather than at block
- // boundaries.
- while (pre_delay_ && size) {
- destination += 2;
- --size;
- --pre_delay_;
- }
-
- // Pre-render the envelope in one pass.
- if (envelope_smoothness_ == 0.0f) {
- RenderEnvelope<false, quality>(envelope, size);
- } else {
- RenderEnvelope<true, quality>(envelope, size);
- }
-
- const int32_t phase_increment = phase_increment_;
- const int32_t first_sample = first_sample_;
- const float gain_l = gain_l_;
- const float gain_r = gain_r_;
- int32_t phase = phase_;
- while (size--) {
- int32_t sample_index = first_sample + (phase >> 16);
-
- float gain = *envelope++;
- if (gain == -1.0f) {
- active_ = false;
- break;
- }
-
- float l = buffer[0].template Read<InterpolationMethod(quality)>(
- sample_index, phase & 65535) * gain;
- if (num_channels == 1) {
- *destination++ += l * gain_l;
- *destination++ += l * gain_r;
- } else if (num_channels == 2) {
- float r = buffer[1].template Read<InterpolationMethod(quality)>(
- sample_index, phase & 65535) * gain;
- *destination++ += l * gain_l + r * (1.0f - gain_r);
- *destination++ += r * gain_r + l * (1.0f - gain_l);
- }
- phase += phase_increment;
- }
- phase_ = phase;
- }
-
- inline bool active() { return active_; }
-
- inline GrainQuality recommended_quality() const {
- return recommended_quality_;
- }
-
- private:
- int32_t first_sample_;
- int32_t width_;
- int32_t phase_;
- int32_t phase_increment_;
- int32_t pre_delay_;
-
- float envelope_smoothness_;
- float envelope_slope_;
- float envelope_phase_;
- float envelope_phase_increment_;
-
- float gain_l_;
- float gain_r_;
-
- bool active_;
-
- GrainQuality recommended_quality_;
-
- DISALLOW_COPY_AND_ASSIGN(Grain);
- };
-
- } // namespace clouds
-
- #endif // CLOUDS_DSP_GRAIN_H_
|