-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFlanger.m
297 lines (269 loc) · 12.1 KB
/
Flanger.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
classdef (StrictDefaults)Flanger < matlab.System & matlab.system.mixin.Propagates
%Flanger Add flanging effect to audio signal.
%
% FLANGER = audioexample.Flanger returns an flanger System object,
% FLANGER, that adds flanging effect to the audio signal.
%
% FLANGER = audioexample.Flanger('Name', Value, ...) returns a flanger
% System object, FLANGER, with each specified property name set to the
% specified value. You can specify additional name-value pair arguments
% in any order as (Name1,Value1,...,NameN, ValueN).
%
% Step method syntax:
%
% Y = step(FLANGER, X) adds flanging effect for the audio input X based
% on the properties specified in the object FLANGER and returns it as
% audio output Y. Each column of X is treated as individual channel of
% input.
%
% System objects may be called directly like a function instead of using
% the step method. For example, y = step(obj, x) and y = obj(x) are
% equivalent.
%
% Flanger methods:
%
% step - See above for the description of the method
% reset - Resets the internal state to initial conditions
% clone - Create Flanger system object with similar property values
% isLocked - Locked status (logical)
%
% Flanger properties:
%
% Delay - Base delay in seconds
% Depth - Amplitude of modulator
% Rate - Frequency of modulator
% FeedbackLevel - Feedback gain
% WetDryMix - Wet to dry signal ratio
% SampleRate - Sample rate of the input audio signal
%
% % Example: Add chorus effect to an audio signal.
%
% reader = dsp.AudioFileReader('SamplesPerFrame', 1024,...
% 'PlayCount', 3);
%
% player = audioDeviceWriter('SampleRate', reader.SampleRate);
%
% flanger = audioexample.Flanger;
%
% while ~isDone(reader)
% Input = reader();
% Output = flanger(Input);
% player(Output);
% end
%
% release(reader)
% release(player)
% Copyright 2015-2016 The MathWorks, Inc.
%#codegen
%----------------------------------------------------------------------
% Public, tunable properties.
%----------------------------------------------------------------------
properties %Some value maxs have been changed
%Delay Base delay
% Specify the base delay for flanger effect as positive scalar
% value in seconds. Base delay value must be in the range between
% 0 and 0.1 seconds. The default value of this property is
% 0.001.
Delay = 0.001
%Depth Amplitude of modulator
% Specify the amplitude of modulating sine wave as a positive
% scalar value. This sinewave is added to the base delay value to
% make the delay sinusoidally modulating. The value must range
% from 0 to 50. The default value of this property is 30.
Depth = 30
%Rate Frequency of modulator
% Specify the frequency of the sine wave as a positive scalar
% value in Hz. This property controls the flanging rate. The
% value must range from 0 to 0.5 Hz. The default value of this
% property is 0.25.
Rate = 0.25
%FeedbackLevel Feedback gain value.
% Specify the feedback gain value as a positive scalar. This
% value must range from 0 to 1. Setting FeedbackLevel to 0 turns
% off the feedback path. The default value of this property is
% 0.4.
FeedbackLevel = 0.4
%WetDryMix Wet/dry mix
% Specify the wet/dry mix ratio as a positive scalar. This value
% ranges from 0 to 1. For example, for a value of 0.6, the ratio
% will be 60% wet to 40% dry signal (Wet - Signal that has effect
% in it. Dry - Unaffected signal). The default value of this
% property is 0.5.
WetDryMix = 0.5
end
%----------------------------------------------------------------------
% Non-tunable properties.
%----------------------------------------------------------------------
properties (Nontunable)
%SampleRate Sampling rate of input audio signal
% Specify the sampling rate of the audio signal as a positive
% scalar value. The default value of this property is 44100 Hz.
SampleRate = 44100
end
%----------------------------------------------------------------------
% Private, non-tunable properties.
%----------------------------------------------------------------------
properties (Access = private, Nontunable)
%pDataType is the data type of input signal. To maintain similar
% data type throughout the process, this property is used to type
% cast the variables.
pDataType
end
%----------------------------------------------------------------------
% Private properties.
%----------------------------------------------------------------------
properties (Access = private)
%pDelayInSamples is the number of samples required to delay the
% input signal.
pDelayInSamples
%pDelay is the object for fractional delay with linear
%interpolation and feedback.
pDelay
%pSineWave is the oscillator for generating sine wave.
pSineWave
%pWetDryMix WetDryMix casted to input data type
pWetDryMix
end
%----------------------------------------------------------------------
% Public properties.
%----------------------------------------------------------------------
methods
% Constructor for Flanger system object.
function obj = Flanger(varargin)
% Set properties according to name-value pairs
setProperties(obj,nargin,varargin{:});
end
%------------------------------------------------------------------
% These set functions validate the attributes and limits of the
% properties of this system object.
function set.Delay(obj,Delay)
validateattributes(Delay,{'numeric'},{'scalar','real','>=',0,'<=',0.1},'Flanger','Delay');
obj.Delay = Delay;
end
function set.Depth(obj,Depth)
validateattributes(Depth,{'numeric'},{'scalar','real','>=',0,'<=',80},'Flanger','Depth');
obj.Depth = Depth;
end
function set.Rate(obj,Rate)
validateattributes(Rate,{'numeric'},{'scalar','real','>=',0,'<=',1},'Flanger','Rate');
obj.Rate = Rate;
end
function set.WetDryMix(obj,WetDryMix)
validateattributes(WetDryMix,{'numeric'},{'scalar','real','>=',0,'<=',1},'Flanger','WetDryMix');
obj.WetDryMix = WetDryMix;
end
function set.FeedbackLevel(obj,FeedbackLevel)
validateattributes(FeedbackLevel,{'numeric'},{'scalar','real','>=',0,'<=',1},'Flanger','FeedbackLevel');
obj.FeedbackLevel = FeedbackLevel;
end
function set.SampleRate(obj,SampleRate)
validateattributes(SampleRate,{'numeric'},{'scalar','real','>',0},'Flanger','SampleRate');
obj.SampleRate = SampleRate;
end
end
%----------------------------------------------------------------------
% Protected methods
%----------------------------------------------------------------------
methods (Access = protected)
function setupImpl(obj,Input)
obj.pDataType = class(Input);
% Create the tunable sine wave oscillator
obj.pSineWave = audioOscillator('Frequency',obj.Rate,...
'Amplitude',obj.Depth,...
'SampleRate',obj.SampleRate, ...
'OutputDataType', obj.pDataType);
% Create the fractional delay object
obj.pDelay = audioexample.DelayFilter(...
'SampleRate',obj.SampleRate,...
'FeedbackLevel',obj.FeedbackLevel);
% Set the tunable properties
processTunedPropertiesImpl(obj)
% Setup the oscillator and DelayFilter object
setup(obj.pSineWave);
setup(obj.pDelay,obj.pDelayInSamples,Input)
end
%------------------------------------------------------------------
function resetImpl(obj)
% Reset the delay and sine wave objects
reset(obj.pDelay)
reset(obj.pSineWave)
end
%------------------------------------------------------------------
function Output = stepImpl(obj,Input)
% Create the delay vector. Delay in samples are added to the
% sine wave here.
obj.pSineWave.SamplesPerFrame = size(Input, 1);
DelayVector = obj.pDelayInSamples + obj.pSineWave();
% Calculate delayed output
Output = obj.pDelay(DelayVector,Input);
% Calculate output by adding wet and dry signal in appropriate
% ratio.
mix = obj.pWetDryMix;
Output = (1-mix)*Input + (mix)*Output;
end
%------------------------------------------------------------------
% When tunable property changes, this function will be called.
function processTunedPropertiesImpl(obj)
% When Delay property changes, we have recalculate
% pDelayInSamples property. Note that to maintain similar data
% types throughout the process, cast function is used.
obj.pDelayInSamples = cast(obj.Delay*obj.SampleRate, obj.pDataType);
% Set the amplitude of sine wave object when Depth property
% changes.
obj.pSineWave.Amplitude = obj.Depth;
% Set the frequency of sine wave when Rate property changes.
obj.pSineWave.Frequency = obj.Rate;
% Setting feedback level of delay object when FeedbackLevel
% property changes.
obj.pDelay.FeedbackLevel = obj.FeedbackLevel;
% Cast obj.WetDryMix to correct data type
obj.pWetDryMix = cast(obj.WetDryMix, obj.pDataType);
end
%------------------------------------------------------------------
function validateInputsImpl(~,Input)
% Validate inputs to the step method at initialization.
validateattributes(Input,{'single','double'},{'nonempty'},'Flanger','Input');
end
%------------------------------------------------------------------
function s = saveObjectImpl(obj)
s = [email protected](obj);
if isLocked(obj)
s.pDelay = matlab.System.saveObject(obj.pDelay);
s.pSineWave = matlab.System.saveObject(obj.pSineWave);
s.pDelayInSamples = obj.pDelayInSamples;
s.pWetDryMix = obj.pWetDryMix;
s.pDataType = obj.pDataType;
end
end
%------------------------------------------------------------------
function loadObjectImpl(obj,s,wasLocked)
if wasLocked
obj.pDelay = matlab.System.loadObject(s.pDelay);
obj.pSineWave = matlab.System.loadObject(s.pSineWave);
obj.pDelayInSamples = s.pDelayInSamples;
obj.pWetDryMix = s.pWetDryMix;
obj.pDataType = s.pDataType;
end
[email protected](obj,s,wasLocked);
end
%------------------------------------------------------------------
function releaseImpl(obj)
release(obj.pDelay)
release(obj.pSineWave)
end
%------------------------------------------------------------------
% Propagators for MATLAB System block
function flag = IsOutputComplexImpl(~)
flag = false;
end
function flag = getOutputSizeImpl(obj)
flag = propagatedInputSize(obj, 1);
end
function flag = getOutputDataTypeImpl(obj)
flag = propagatedInputDataType(obj, 1);
end
function flag = isOutputFixedSizeImpl(obj)
flag = propagatedInputFixedSize(obj,1);
end
end
end