Skip to content
Snippets Groups Projects
Commit d4ce2c0d authored by moon2's avatar moon2 :speech_balloon:
Browse files

bl00mbox: envelope applies square slopes

parent 8ae0a0b2
No related branches found
No related tags found
1 merge request!609bl00mbox: envelope applies square slopes
Pipeline #9393 failed
......@@ -127,11 +127,11 @@ the sample buffer size does **not** change size dynamically. when loading a file
### env\_adsr <a name="env_adsr"></a>
generates a linear ADSR envelope and optionally applies it directly to a signal.
generates a linear ADSR envelope and optionally applies a version with "squared" slopes to a signal.
###### signals
- `output [output]`: provides a version of `input` with `env_output` applied. lerps between `env_output` cornerpoints to minimize zipper noise. constant buffer if `input` is a constant buffer or if the envelope is idling.
- `output [output]`: provides a version of `input` with `env_output` applied, however the transition phases are modified with a square function to achieve a CPU-cheap log-ish behavior. lerps between `env_output` cornerpoints to minimize zipper noise. constant buffer if `input` is a constant buffer or if the envelope is idling.
- `input [input]`: signal to be sent to `output` with `env_output` applied. source render is only requested if envelope is not idling and `gain` is not 0.
- `env_output [output/lazy/gain]`: provides a constant buffer with the current gain value of the envelope scaled by last event volume received by `trigger` as well as `gain`.
- `trigger [input/lazy/trigger]`: processes start, stop and restart events as expected and honors event volume.
......
......@@ -51,9 +51,20 @@ static inline void update_attack_coeffs(radspa_t * env_adsr, uint16_t num_sample
static inline void update_release_coeffs(radspa_t * env_adsr, uint16_t num_samples, uint32_t render_pass_id){
env_adsr_data_t * data = env_adsr->plugin_data;
int16_t release = radspa_signal_get_value(&env_adsr->signals[ENV_ADSR_RELEASE], 0, render_pass_id);
if((data->release_prev_ms != release) || (data->num_samples_prev != num_samples)){
if((data->release_prev_ms != release) || (data->num_samples_prev != num_samples) || data->release_init_val_prev != data->release_init_val){
data->release_raw = env_adsr_time_ms_to_val_rise(release, data->release_init_val, num_samples);
data->release_prev_ms = release;
if(data->release_init_val_prev != data->release_init_val){;
uint8_t phase = ENV_ADSR_PHASE_RELEASE;
data->square_min[phase] = 0;
uint32_t delta = data->release_init_val >> 17;;
if(delta){
data->square_mult[phase] = (1UL<<31) / delta;
} else {
data->square_mult[phase] = 0;
}
data->release_init_val_prev = data->release_init_val;
}
}
}
......@@ -71,7 +82,17 @@ static inline void update_decay_coeffs(radspa_t * env_adsr, uint16_t num_samples
if((data->decay_prev_ms != decay) || (data->sustain_prev != data->sustain) || (data->num_samples_prev != num_samples)){
data->decay_raw = env_adsr_time_ms_to_val_rise(decay, UINT32_MAX - data->sustain, num_samples);
data->decay_prev_ms = decay;
data->sustain_prev = data->sustain;
if(data->sustain_prev != data->sustain){
uint8_t phase = ENV_ADSR_PHASE_DECAY;
data->square_min[phase] = data->sustain >> 17;
uint32_t delta = 32767 - data->square_min[phase];
if(delta){
data->square_mult[phase] = (1UL<<31) / delta;
} else {
data->square_mult[phase] = 0 ;
}
data->sustain_prev = data->sustain;
}
}
}
......@@ -151,6 +172,17 @@ void env_adsr_run(radspa_t * env_adsr, uint16_t num_samples, uint32_t render_pas
env = (env * gain) >> 15;
radspa_signal_set_const_value(&env_adsr->signals[ENV_ADSR_ENV_OUTPUT], env);
if(data->square_mult[data->env_phase]){
uint32_t env_sq = (data->env_counter >> 17) - data->square_min[data->env_phase];
env_sq *= env_sq << 1;
env_sq = ((uint64_t) data->square_mult[data->env_phase] * env_sq) >> 32;;
env_sq += data->square_min[data->env_phase];
env = env_sq;
env = (env * data->velocity) >> 15;
env = (env * gain) >> 15;
}
int16_t ret = radspa_signal_get_const_value(&env_adsr->signals[ENV_ADSR_INPUT], render_pass_id);
if(ret == RADSPA_SIGNAL_NONCONST){
int32_t env_slope = ((env - data->env_prev) << 15) / num_samples;
......@@ -192,6 +224,10 @@ radspa_t * env_adsr_create(uint32_t init_var){
data->sustain_prev = -1;
data->decay_prev_ms = -1;
data->env_prev = 0;
data->square_min[ENV_ADSR_PHASE_ATTACK] = 0;
data->square_mult[ENV_ADSR_PHASE_ATTACK] = (1UL<<31) / 32767;
data->square_mult[ENV_ADSR_PHASE_OFF] = 0;
data->square_mult[ENV_ADSR_PHASE_SUSTAIN] = 0;
return env_adsr;
}
......
......@@ -5,6 +5,8 @@
typedef struct {
uint32_t env_counter;
uint32_t velocity;
uint32_t square_mult[5];
uint32_t square_min[5];
int16_t trigger_prev;
uint16_t num_samples_prev;
int32_t env_prev;
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment