1 |
$VERSION = '1.2'; |
2 |
|
3 |
pp_setversion $VERSION; |
4 |
pp_beginwrap (); # force error with older PPs |
5 |
|
6 |
pp_addpm {At => Top}, <<'EOD'; |
7 |
=head1 NAME |
8 |
|
9 |
PDL::Audio - Some PDL functions intended for audio processing. |
10 |
|
11 |
=head1 SYNOPSIS |
12 |
|
13 |
use PDL; |
14 |
use PDL::Audio; |
15 |
|
16 |
=head1 DESCRIPTION |
17 |
|
18 |
Oh well ;) Not much "introductory documentation" has been written yet :( |
19 |
|
20 |
Installing this distribution also installs F<pdlaudio-demo>, which |
21 |
showcases some of the oeprators, and C<pdlaudio-birds>, which imites some |
22 |
bird calls with PDL::Audio. You should study them to get the hang of it. |
23 |
|
24 |
=head2 NOTATION |
25 |
|
26 |
Brackets around parameters indicate that the respective parameter is |
27 |
optional and will be replaced with some default value when absent (or |
28 |
C<undef>, which might be different in other packages). |
29 |
|
30 |
The sampling frequency and duration are by default (see individual |
31 |
descriptions) given in cycles/sample (or samples in case of a |
32 |
duration). That means if you want to specify a duration of two seconds, |
33 |
you have to multiply by the sampling frequency in HZ, and if you want |
34 |
to specify a frequency of 440 Hz, you have to divide by the sampling |
35 |
frequency: |
36 |
|
37 |
# Syntax: gen_oscil duration*, frequency/ |
38 |
$signal = gen_oscil 2*HZ, 440/HZ; |
39 |
# with a sampling frequency of 44100 Hertz: |
40 |
$signal = gen_oscil 2*44100, 440/44100; |
41 |
|
42 |
print describe_audio $signal, "\n"; |
43 |
playaudio $signal->scale2short; |
44 |
|
45 |
To help you, the required unit is given as a type suffix in the parameter |
46 |
name. A "/" means that you have to divide by the sampling frequency (to |
47 |
convert from Hertz) and a suffix of "*" indicates that a multiplication is |
48 |
required. |
49 |
|
50 |
Most parameters named "size", "duration" (or marked with "*") can |
51 |
be replaced by a piddle, which is then used to give length and from |
52 |
(mono/stereo). |
53 |
|
54 |
=head2 HEADER ATTRIBUTES |
55 |
|
56 |
The following header attributes are stored and evaluated by most |
57 |
functions. PDL::Audio provides mutator methods for all them (e.g. |
58 |
|
59 |
print "samplerate is ", $pdl->rate; |
60 |
$pdl->comment("set the comment to this string"); |
61 |
|
62 |
=over 4 |
63 |
|
64 |
=item rate |
65 |
|
66 |
The sampling rate in hz. |
67 |
|
68 |
=item filetype |
69 |
|
70 |
The filetype (wav, au etc..). Must be one of: |
71 |
|
72 |
FILE_NEXT FILE_AIFC FILE_RIFF FILE_BICSF FILE_NIST FILE_INRS FILE_ESPS |
73 |
FILE_SVX FILE_VOC FILE_SNDT FILE_RAW FILE_SMP FILE_SD2 FILE_AVR |
74 |
FILE_IRCAM FILE_SD1 FILE_SPPACK FILE_MUS10 FILE_HCOM FILE_PSION |
75 |
FILE_MAUD FILE_IEEE FILE_DESKMATE FILE_DESKMATE_2500 FILE_MATLAB |
76 |
FILE_ADC FILE_SOUNDEDIT FILE_SOUNDEDIT_16 FILE_DVSM FILE_MIDI |
77 |
FILE_ESIGNAL FILE_SOUNDFONT FILE_GRAVIS FILE_COMDISCO FILE_GOLDWAVE |
78 |
FILE_SRFS FILE_MIDI_SAMPLE_DUMP FILE_DIAMONDWARE FILE_REALAUDIO |
79 |
FILE_ADF FILE_SBSTUDIOII FILE_DELUSION FILE_FARANDOLE FILE_SAMPLE_DUMP |
80 |
FILE_ULTRATRACKER FILE_YAMAHA_SY85 FILE_YAMAHA_TX16 FILE_DIGIPLAYER |
81 |
FILE_COVOX FILE_SPL FILE_AVI FILE_OMF FILE_QUICKTIME FILE_ASF |
82 |
FILE_YAMAHA_SY99 FILE_KURZWEIL_2000 FILE_AIFF FILE_AU |
83 |
|
84 |
=item path |
85 |
|
86 |
The filename (or file specification) used to load or save a file. |
87 |
|
88 |
=item format |
89 |
|
90 |
Specifies the type the underlying file format uses. The samples will always be in short or long signed format. |
91 |
|
92 |
Must be one of |
93 |
|
94 |
FORMAT_NO_SND FORMAT_16_LINEAR FORMAT_8_MULAW FORMAT_8_LINEAR |
95 |
FORMAT_32_FLOAT FORMAT_32_LINEAR FORMAT_8_ALAW FORMAT_8_UNSIGNED |
96 |
FORMAT_24_LINEAR FORMAT_64_DOUBLE FORMAT_16_LINEAR_LITTLE_ENDIAN |
97 |
FORMAT_32_LINEAR_LITTLE_ENDIAN FORMAT_32_FLOAT_LITTLE_ENDIAN |
98 |
FORMAT_64_DOUBLE_LITTLE_ENDIAN FORMAT_16_UNSIGNED |
99 |
FORMAT_16_UNSIGNED_LITTLE_ENDIAN FORMAT_24_LINEAR_LITTLE_ENDIAN |
100 |
FORMAT_32_VAX_FLOAT FORMAT_12_LINEAR FORMAT_12_LINEAR_LITTLE_ENDIAN |
101 |
FORMAT_12_UNSIGNED FORMAT_12_UNSIGNED_LITTLE_ENDIAN COMPATIBLE_FORMAT |
102 |
|
103 |
PDL::Audio conviniently defines the following aliases for the following |
104 |
constants, that are already correct for the host byteorder: |
105 |
|
106 |
FORMAT_ULAW_BYTE FORMAT_ALAW_BYTE FORMAT_LINEAR_BYTE |
107 |
FORMAT_LINEAR_SHORT FORMAT_LINEAR_USHORT FORMAT_LINEAR_LONG |
108 |
FORMAT_LINEAR_FLOAT FORMAT_LINEAR_DOUBLE |
109 |
|
110 |
=item comment |
111 |
|
112 |
The file comment (if any). |
113 |
|
114 |
=item device |
115 |
|
116 |
The device to output audio. One of: |
117 |
|
118 |
DEV_DEFAULT DEV_READ_WRITE DEV_ADAT_IN DEV_AES_IN DEV_LINE_OUT |
119 |
DEV_LINE_IN DEV_MICROPHONE DEV_SPEAKERS DEV_DIGITAL_IN DEV_DIGITAL_OUT |
120 |
DEV_DAC_OUT DEV_ADAT_OUT DEV_AES_OUT DEV_DAC_FILTER DEV_MIXER |
121 |
DEV_LINE1 DEV_LINE2 DEV_LINE3 DEV_AUX_INPUT DEV_CD_IN DEV_AUX_OUTPUT |
122 |
DEV_SPDIF_IN DEV_SPDIF_OUT |
123 |
|
124 |
=back |
125 |
|
126 |
=head2 EXPORTED CONSTANTS |
127 |
|
128 |
In addition to the exported constants described above (and later in |
129 |
the function descriptions), this module also exports the mathematical |
130 |
constants M_PI and M_2PI, so watch out for clashes! |
131 |
|
132 |
=cut |
133 |
EOD |
134 |
|
135 |
@cconsts = qw( |
136 |
SNDLIB_DEFAULT_DEVICE SNDLIB_READ_WRITE_DEVICE |
137 |
SNDLIB_ADAT_IN_DEVICE SNDLIB_AES_IN_DEVICE |
138 |
SNDLIB_LINE_OUT_DEVICE SNDLIB_LINE_IN_DEVICE |
139 |
SNDLIB_MICROPHONE_DEVICE SNDLIB_SPEAKERS_DEVICE |
140 |
SNDLIB_DIGITAL_IN_DEVICE SNDLIB_DIGITAL_OUT_DEVICE |
141 |
SNDLIB_DAC_OUT_DEVICE SNDLIB_ADAT_OUT_DEVICE |
142 |
SNDLIB_AES_OUT_DEVICE SNDLIB_DAC_FILTER_DEVICE |
143 |
SNDLIB_MIXER_DEVICE SNDLIB_LINE1_DEVICE |
144 |
SNDLIB_LINE2_DEVICE SNDLIB_LINE3_DEVICE |
145 |
SNDLIB_AUX_INPUT_DEVICE SNDLIB_CD_IN_DEVICE |
146 |
SNDLIB_AUX_OUTPUT_DEVICE SNDLIB_SPDIF_IN_DEVICE |
147 |
SNDLIB_SPDIF_OUT_DEVICE |
148 |
|
149 |
SNDLIB_NO_SND SNDLIB_16_LINEAR SNDLIB_8_MULAW SNDLIB_8_LINEAR |
150 |
SNDLIB_32_FLOAT SNDLIB_32_LINEAR SNDLIB_8_ALAW SNDLIB_8_UNSIGNED |
151 |
SNDLIB_24_LINEAR SNDLIB_64_DOUBLE SNDLIB_16_LINEAR_LITTLE_ENDIAN |
152 |
SNDLIB_32_LINEAR_LITTLE_ENDIAN SNDLIB_32_FLOAT_LITTLE_ENDIAN |
153 |
SNDLIB_64_DOUBLE_LITTLE_ENDIAN SNDLIB_16_UNSIGNED |
154 |
SNDLIB_16_UNSIGNED_LITTLE_ENDIAN SNDLIB_24_LINEAR_LITTLE_ENDIAN |
155 |
SNDLIB_32_VAX_FLOAT SNDLIB_12_LINEAR SNDLIB_12_LINEAR_LITTLE_ENDIAN |
156 |
SNDLIB_12_UNSIGNED SNDLIB_12_UNSIGNED_LITTLE_ENDIAN |
157 |
SNDLIB_COMPATIBLE_FORMAT |
158 |
|
159 |
NeXT_sound_file AIFC_sound_file RIFF_sound_file |
160 |
BICSF_sound_file NIST_sound_file INRS_sound_file |
161 |
ESPS_sound_file SVX_sound_file VOC_sound_file |
162 |
SNDT_sound_file raw_sound_file SMP_sound_file |
163 |
SD2_sound_file AVR_sound_file IRCAM_sound_file |
164 |
SD1_sound_file SPPACK_sound_file MUS10_sound_file |
165 |
HCOM_sound_file PSION_sound_file MAUD_sound_file |
166 |
IEEE_sound_file DeskMate_sound_file DeskMate_2500_sound_file |
167 |
Matlab_sound_file ADC_sound_file SoundEdit_sound_file |
168 |
SoundEdit_16_sound_file DVSM_sound_file MIDI_file |
169 |
Esignal_file soundfont_sound_file gravis_sound_file |
170 |
comdisco_sound_file goldwave_sound_file srfs_sound_file |
171 |
MIDI_sample_dump DiamondWare_sound_file RealAudio_sound_file |
172 |
ADF_sound_file SBStudioII_sound_file Delusion_sound_file |
173 |
Farandole_sound_file Sample_dump_sound_file Ultratracker_sound_file |
174 |
Yamaha_SY85_sound_file Yamaha_TX16_sound_file digiplayer_sound_file |
175 |
Covox_sound_file SPL_sound_file AVI_sound_file |
176 |
OMF_sound_file Quicktime_sound_file asf_sound_file |
177 |
Yamaha_SY99_sound_file Kurzweil_2000_sound_file AIFF_sound_file |
178 |
|
179 |
BANDPASS DIFFERENTIATOR HILBERT |
180 |
); |
181 |
|
182 |
for (@cconsts) { |
183 |
local $_ = $_; |
184 |
s/^SNDLIB_(.*)_DEVICE/DEV_$1/; |
185 |
s/^SNDLIB_/FORMAT_/; |
186 |
s/MIDI_file/FILE_MIDI/; |
187 |
s/MIDI_sample_dump/FILE_MIDI_sample_dump/; |
188 |
s/Esignal_file/FILE_Esignal/; |
189 |
s/(.*)_sound_file/FILE_$1/; |
190 |
push @pconsts, uc $_; |
191 |
} |
192 |
|
193 |
@METHODS = qw( |
194 |
waudio cut_leading_silence cut_trailing_silence cut_silence filter_zpolar |
195 |
partials2polynomial scale2short filter_fir filter_iir rfft irfft spectrum |
196 |
filter_comb filter_notch filter_allpass filter_src filter_granulate |
197 |
filter_contrast_enhance describe_audio filter_center ring_modulate |
198 |
amplitude_modulate filter_ppolar design_remez_fir concat |
199 |
gen_oscil gen_sawtooth gen_square gen_triangle gen_asymmetric_fm gen_rand_1f |
200 |
gen_env audiomix gen_sum_of_cosines gen_sine_summation gen_fft_window |
201 |
gen_pulse_train gen_rand gen_from_table gen_from_partials partials2waveshape |
202 |
|
203 |
); |
204 |
@EXPORT = (qw( |
205 |
sound_format_name sound_type_name raudio |
206 |
FORMAT_ULAW_BYTE FORMAT_ALAW_BYTE FORMAT_LINEAR_BYTE |
207 |
FORMAT_LINEAR_SHORT FORMAT_LINEAR_USHORT FORMAT_LINEAR_LONG |
208 |
FORMAT_LINEAR_FLOAT FORMAT_LINEAR_DOUBLE |
209 |
M_PI M_2PI |
210 |
), @METHODS, @pconsts); |
211 |
|
212 |
for (@EXPORT) { |
213 |
pp_add_exported "", $_; |
214 |
} |
215 |
|
216 |
pp_addpm "\@METHODS = qw(@METHODS);\n"; |
217 |
|
218 |
$addboot = "{\nHV *stash = gv_stashpvn(\"PDL::Audio\", 10, TRUE);"; |
219 |
|
220 |
for (0..$#cconsts) { |
221 |
$addboot .= " newCONSTSUB(stash, \"$pconsts[$_]\", newSViv($cconsts[$_]));\n"; |
222 |
} |
223 |
$addboot .= " newCONSTSUB(stash, \"M_PI\" , newSVnv(M_PI ));\n"; |
224 |
$addboot .= " newCONSTSUB(stash, \"M_2PI\", newSVnv(M_2PI));\n"; |
225 |
$addboot .= '} |
226 |
mus_set_error_handler (mus_barfer); |
227 |
'; |
228 |
|
229 |
pp_add_boot $addboot; |
230 |
|
231 |
pp_addpm <<'EOD'; |
232 |
|
233 |
use PDL::LiteF; |
234 |
use PDL::Complex; |
235 |
use Carp; |
236 |
|
237 |
$bigendian = 0x55 == unpack "S", "\0U"; |
238 |
|
239 |
sub FORMAT_ULAW_BYTE (){ FORMAT_8_MULAW } |
240 |
sub FORMAT_ALAW_BYTE (){ FORMAT_8_ALAW } |
241 |
sub FORMAT_LINEAR_BYTE (){ FORMAT_8_UNSIGNED } |
242 |
sub FORMAT_LINEAR_SHORT (){ $bigendian ? FORMAT_16_LINEAR : FORMAT_16_LINEAR_LITTLE_ENDIAN } |
243 |
sub FORMAT_LINEAR_USHORT(){ $bigendian ? FORMAT_16_UNSIGNED : FORMAT_16_UNSIGNED_LITTLE_ENDIAN } |
244 |
sub FORMAT_LINEAR_LONG (){ $bigendian ? FORMAT_32_LINEAR : FORMAT_32_LINEAR_LITTLE_ENDIAN } |
245 |
sub FORMAT_LINEAR_FLOAT (){ $bigendian ? FORMAT_32_FLOAT : FORMAT_32_FLOAT_LITTLE_ENDIAN } |
246 |
sub FORMAT_LINEAR_DOUBLE(){ $bigendian ? FORMAT_32_DOUBLE : FORMAT_32_DOUBLE_LITTLE_ENDIAN } |
247 |
|
248 |
sub FILE_AU (){ FILE_NEXT } |
249 |
|
250 |
# provide some accessor methods |
251 |
for my $field (qw(rate comment filetype path format)) { |
252 |
*{"PDL::$field"} = sub { |
253 |
my $pdl = shift; |
254 |
my $hdr = $pdl->gethdr; |
255 |
if (@_) { |
256 |
$hdr->{$field} = $_[0]; |
257 |
$pdl->sethdr($hdr); |
258 |
}; |
259 |
$hdr->{$field}; |
260 |
}; |
261 |
} |
262 |
|
263 |
=head2 sound_format_name format_code |
264 |
|
265 |
Return the human-readable name of the file format with code C<format_code>. |
266 |
|
267 |
=head2 sound_type_name type_code |
268 |
|
269 |
Return the human-readable name of the sample type with code C<type_code>. |
270 |
|
271 |
=head2 describe_audio piddle |
272 |
|
273 |
Describe the audio stream contained in piddle and return it as a string. A |
274 |
fresh piddle might return: |
275 |
|
276 |
mono sound with 27411 samples |
277 |
|
278 |
Whereas a freshly loaded soundfile might yield: |
279 |
|
280 |
stereo sound with 27411 samples, original name "kongas.wav", type 2 (RIFF), |
281 |
rate 11025/s (duration 2.49s), format 7 (8-bit unsigned) |
282 |
|
283 |
=cut |
284 |
|
285 |
sub describe_audio($) { |
286 |
my $pdl = shift; |
287 |
my ($channels, $samples) = $pdl->dims; |
288 |
($channels, $samples) = (1, $channels) unless defined $samples; |
289 |
my $chan = $channels < 2 ? "mono" : |
290 |
$channels == 2 ? "stereo" : |
291 |
$channels == 4 ? "quad channel" : |
292 |
"$channels channel"; |
293 |
my $desc = "$chan sound with $samples samples"; |
294 |
$desc .= sprintf ", original name \"%s\"", $pdl->path if $pdl->path; |
295 |
$desc .= sprintf ", type %d (%s)", $pdl->filetype, sound_type_name($pdl->filetype) if $pdl->filetype; |
296 |
$desc .= sprintf ", rate %d/s (duration %.2fs)", $pdl->rate, $samples/$pdl->rate if $pdl->rate; |
297 |
$desc .= sprintf ", format %d (%s)", $pdl->format, sound_format_name($pdl->format) if $pdl->format; |
298 |
$desc |
299 |
} |
300 |
|
301 |
=head2 raudio path, [option-hash], option => value, ... |
302 |
|
303 |
Reads audio data into the piddle. Options can be anything, most useful |
304 |
values are C<filetype>, C<rate>, C<channels> and C<format>. The returned |
305 |
piddle is represents "time" in the outer dimension, and samples in the |
306 |
inner (i.e. scalars for mono files, 2-vectors for stereo files): |
307 |
|
308 |
[ [left0, right0], [left1, right1], [left2, right2], ...] |
309 |
|
310 |
# read any file |
311 |
$pdl = raudio "file.wav"; |
312 |
# read a file. if it is a raw file preset values |
313 |
$pdl = raudio "file.raw", filetype => FILE_RAW, rate => 44100, channels => 2; |
314 |
|
315 |
=head2 waudio pdl, [option-hash], option => value, ... |
316 |
|
317 |
Writes a pdl as a file. See L<raudio> for options and format. The path and |
318 |
other metadata is taken from the header, whcih cna be overwritten using |
319 |
options, e.g.: |
320 |
|
321 |
# write a file, using the header of another piddle |
322 |
$pdl->waudio ($orig_file->gethdr); |
323 |
# write pdl as .au file, take rate from the header |
324 |
$pdl->waudio (path => "piddle.au", filetype => FILE_AU, format => FORMAT_16_LINEAR; |
325 |
|
326 |
=cut |
327 |
|
328 |
# read a sound file |
329 |
sub raudio { |
330 |
my $path = shift; |
331 |
my %hdr = (); %hdr = (%hdr, %{+shift}) if ref $_[0]; %hdr = (%hdr, @_) if @_; |
332 |
|
333 |
# for raw files this is necessary |
334 |
mus_set_raw_header_defaults $hdr{rate} || 44100, |
335 |
$hdr{channels}|| 1, |
336 |
$hdr{format} || FORMAT_LINEAR_SHORT; |
337 |
|
338 |
$hdr{path} = $path; |
339 |
$hdr{filetype} = sound_header_type $path; |
340 |
$hdr{filetype} >= 0 or barf "$path: ".audio_error_name audio_error; |
341 |
$hdr{rate} = sound_srate $path; |
342 |
$hdr{comment} = sound_comment $path if defined sound_comment $path; |
343 |
$hdr{format} = sound_data_format $path; |
344 |
my $channels = sound_chans $path; |
345 |
my $frames = sound_frames $path; |
346 |
|
347 |
my $fd = open_sound_input $path; $fd >= 0 or barf "$path: ".audio_error_name audio_error; |
348 |
|
349 |
my $pdl = zeroes long, $frames, $channels; |
350 |
$pdl->clump(-1)->read_sound ($fd, $channels, $frames) |
351 |
>= 0 or barf "$path: ".audio_error_name audio_error; |
352 |
(close_sound_input $fd) >= 0 or barf "$path: ".audio_error_name audio_error; |
353 |
$pdl = $pdl->short->xchg(0,1); |
354 |
$pdl = $pdl->clump(2) if $channels == 1; |
355 |
$pdl->sever; |
356 |
$pdl->sethdr(\%hdr); |
357 |
$pdl |
358 |
} |
359 |
|
360 |
sub _audio_make_plain { |
361 |
my $pdl = shift; |
362 |
if ($pdl->getndims == 1) { |
363 |
($pdl, 1, $pdl->getdim(0)) |
364 |
} else { |
365 |
($pdl->xchg(0,1)->clump(-1), $pdl->dims) |
366 |
} |
367 |
} |
368 |
|
369 |
sub waudio { |
370 |
my $pdl = shift; |
371 |
my %hdr = %{$pdl->gethdr || {}}; %hdr = (%hdr, %{+shift}) if ref $_[0]; %hdr = (%hdr, @_) if @_; |
372 |
my ($channels, $frames); |
373 |
|
374 |
$hdr{filetype} = FILE_NEXT unless exists $hdr{filetype}; |
375 |
$hdr{format} ||= FORMAT_16_LINEAR; |
376 |
$hdr{rate} ||= 44100; |
377 |
|
378 |
($pdl, $channels, $frames) = _audio_make_plain $pdl->convert(long); |
379 |
|
380 |
1 <= $channels && $channels <= 2 |
381 |
or croak "can only write mono or stereo (one or two channel) files, not $channels channel files"; |
382 |
|
383 |
my $fd = open_sound_output $hdr{path}, $hdr{rate}, $channels, $hdr{format}, $hdr{filetype}, $hdr{comment}; |
384 |
$fd >= 0 or barf "$hdr{$path}: ".audio_error_name audio_error; |
385 |
$pdl->clump(-1)->write_sound($fd, $channels, $frames) |
386 |
>= 0 or barf "$path: ".audio_error_name audio_error; |
387 |
(close_sound_output $fd, mus_samples2bytes $hdr{format}, $frames * $channels) |
388 |
>= 0 or barf "$hdr{$path}: ".audio_error_name audio_error; |
389 |
} |
390 |
|
391 |
=head2 cut_leading_silence pdl, level |
392 |
|
393 |
Cuts the leading silence (i.e. all samples with absolute value < level) |
394 |
and returns the resulting part. |
395 |
|
396 |
=head2 cut_trailing_silence pdl, level |
397 |
|
398 |
Cuts the trailing silence. |
399 |
|
400 |
=head2 cut_silence pdl, level |
401 |
|
402 |
Calls C<cut_leading_silence> and C<cut_trailing_silence> and returns the |
403 |
result. |
404 |
|
405 |
=cut |
406 |
|
407 |
sub cut_leading_silence { |
408 |
my $pdl = shift; |
409 |
my $level = 1*shift; |
410 |
my $skip = which (abs($pdl) > $level); |
411 |
$skip = $skip->nelem ? $skip->at(0) : 0; |
412 |
$pdl->slice("$skip:-1") |
413 |
} |
414 |
|
415 |
sub cut_trailing_silence { |
416 |
my $pdl = shift; |
417 |
my $level = 1*shift; |
418 |
$level = 400000; |
419 |
my $skip = which (abs($pdl) > $level); |
420 |
$skip = $skip->nelem ? $skip->at(-1) : -1; |
421 |
$skip-- if $skip > 0; |
422 |
$pdl->slice("0:$skip") |
423 |
} |
424 |
|
425 |
sub cut_silence { |
426 |
$_[0]->cut_leading_silence($_[1]) |
427 |
->cut_trailing_silence($_[1]) |
428 |
} |
429 |
|
430 |
# have we been a bad boy? |
431 |
|
432 |
for (@METHODS) { |
433 |
*{"PDL::$_"} = \&$_; |
434 |
} |
435 |
EOD |
436 |
|
437 |
for my $d (qw(read write)) { |
438 |
pp_def($d.'_sound', |
439 |
Pars => ($d eq "read" ? '[o]' : '') . 'sample(n)', |
440 |
OtherPars => "int fd; int chans; int frames", |
441 |
GenericTypes => [L], |
442 |
PMCode => '', |
443 |
Doc => undef, |
444 |
Code => q@ |
445 |
int chans = $COMP(chans); |
446 |
int frames = $COMP(frames); |
447 |
int **bufs = malloc (chans * sizeof (int *)); |
448 |
int i; |
449 |
dSP; |
450 |
|
451 |
/* hmm... */ |
452 |
assert (sizeof (PDL_long) == sizeof (int)); |
453 |
|
454 |
for (i = 0; i < chans; i++) |
455 |
bufs[i] = (int *)$P(sample) + i * frames; |
456 |
|
457 |
XPUSHs (sv_2mortal (newSViv (@.$d.q@_sound ($COMP(fd), 0, frames - 1, chans, bufs)))); |
458 |
|
459 |
free (bufs); |
460 |
@, |
461 |
); |
462 |
} |
463 |
|
464 |
pp_addhdr <<'EOH'; |
465 |
#include "sndlib/sndlib.h" |
466 |
#include "xlib.h" |
467 |
|
468 |
static void |
469 |
mus_barfer (int err_type, char *err_msg) |
470 |
{ |
471 |
barf ("%s [MUSERR]", err_msg); |
472 |
} |
473 |
|
474 |
EOH |
475 |
|
476 |
pp_addpm <<'EOP'; |
477 |
|
478 |
=head2 playaudio pdl, [option-hash], option => value ... |
479 |
|
480 |
Play the piddle as an audio file. Options can be supplied either through |
481 |
the option hash (a hash-reference), through the pdl header or the options: |
482 |
|
483 |
# play a piddle that has a valid header (e.g. from raudio) |
484 |
$pdl->playaudio; |
485 |
# play it with a different samplerate |
486 |
$pdl->playaudio(rate => 22050); |
487 |
|
488 |
=cut |
489 |
|
490 |
EOP |
491 |
|
492 |
pp_def('playaudio', |
493 |
Pars => 'sample(n)', |
494 |
OtherPars => "int srate; int chans; int format; int dev; int bufsize", |
495 |
GenericTypes => [B,U,S,L], |
496 |
Doc => undef, |
497 |
PMCode => q! |
498 |
sub PDL::playaudio { |
499 |
my $pdl = shift; |
500 |
my %hdr = %{$pdl->gethdr || {}}; %hdr = (%hdr, %{+shift}) if ref $_[0]; %hdr = (%hdr, @_) if @_; |
501 |
my $chans; |
502 |
$hdr{format} ||= FORMAT_LINEAR_SHORT; |
503 |
($pdl, $chans) = _audio_make_plain $pdl; |
504 |
&PDL::_playaudio_int($pdl, |
505 |
$hdr{rate} || 44100, |
506 |
$chans, |
507 |
$hdr{format}, |
508 |
$hdr{device} || DEV_DEFAULT, |
509 |
$hdr{bufsize} || 32768); |
510 |
} |
511 |
!, |
512 |
Code => q@ |
513 |
int ns; |
514 |
int line; |
515 |
|
516 |
if (sizeof ($GENERIC ()) != mus_format2bytes ($COMP(format))) |
517 |
barf ("pdl datatype and selected audio format do not match"); |
518 |
|
519 |
if (initialize_audio () < 0) |
520 |
barf ("playaudio: %s", audio_error_name (audio_error ())); |
521 |
|
522 |
if ((line = open_audio_output ($COMP(dev), $COMP(srate), $COMP(chans), $COMP(format), $COMP(bufsize))) < 0) |
523 |
barf ("playaudio: %s", audio_error_name (audio_error ())); |
524 |
|
525 |
ns = $SIZE(n); |
526 |
threadloop %{ |
527 |
if (write_audio (line, (void *)$P(sample), ns * sizeof ($GENERIC()))) |
528 |
barf ("playaudio: %s", audio_error_name (audio_error ())); |
529 |
%} |
530 |
if (close_audio (line) < 0) |
531 |
barf ("playaudio: %s", audio_error_name (audio_error ())); |
532 |
@, |
533 |
); |
534 |
|
535 |
pp_def 'ulaw2linear', |
536 |
Pars => 'byte u(n); short [o] s(n)', |
537 |
GenericTypes => [B], |
538 |
Doc => 'conversion from (m)u-law into signed, linear, 16 bit samples (rather slow)', |
539 |
Code => q! loop(n) %{ $s() = st_ulaw_to_linear ($u()); %} !, |
540 |
; |
541 |
|
542 |
pp_def 'linear2ulaw', |
543 |
Pars => 'short s(n); byte [o] u(n)', |
544 |
GenericTypes => [S], |
545 |
Doc => 'conversion from signed, linear, 16 bit samples into (m)u-law (rather slow)', |
546 |
Code => q! loop(n) %{ $u() = st_linear_to_ulaw ($s()); %} !, |
547 |
; |
548 |
|
549 |
pp_def 'alaw2linear', |
550 |
Pars => 'byte u(n); short [o] s(n)', |
551 |
GenericTypes => [B], |
552 |
Doc => 'conversion from A-law into signed, linear, 16 bit samples (rather slow)', |
553 |
Code => q! loop(n) %{ $s() = st_Alaw_to_linear ($u()); %} !, |
554 |
; |
555 |
|
556 |
pp_def 'linear2alaw', |
557 |
Pars => 'short s(n); byte [o] u(n)', |
558 |
GenericTypes => [S], |
559 |
Doc => 'conversion from signed, linear, 16 bit samples into A-law (rather slow)', |
560 |
Code => q! loop(n) %{ $u() = st_linear_to_Alaw ($s()); %} !, |
561 |
; |
562 |
|
563 |
pp_addpm <<'EOP'; |
564 |
|
565 |
=head2 gen_oscil duration*, freq/, phase-mod, [fm-mod/] |
566 |
|
567 |
=head2 gen_sawtooth duration*, freq/, phase-mod, [fm-mod/] |
568 |
|
569 |
=head2 gen_square duration*, freq/, phase-mod, duty, [fm-mod/] |
570 |
|
571 |
=head2 gen_triangle duration*, freq/, phase-mod, [fm-mod/] |
572 |
|
573 |
=head2 gen_pulse_train duration*, freq/, phase-mod, [fm-mod/] |
574 |
|
575 |
=head2 gen_rand duration*, freq/ |
576 |
|
577 |
=head2 gen_rand_1f duration* |
578 |
|
579 |
All of these functions generate appropriate waveforms with frequency |
580 |
C<freq> (cycles/sample) and phase C<phase> (0..1). |
581 |
|
582 |
The C<duration> might be either a piddle (which gives the form of the |
583 |
output) or the number of samples to generate. |
584 |
|
585 |
The output samples are between -1 and +1 (i.e. "-1 <= s <= +1"). |
586 |
|
587 |
The C<duty> parameter of the square generator influences the duty cycle of |
588 |
the signal. Zero means 50%-50%, 0.5 means 75% on, 25% off, -0.8 means 10% |
589 |
on, 90% off etc... Of course, the C<duty> parameter might also be a vector |
590 |
of size C<duration>. |
591 |
|
592 |
=cut |
593 |
|
594 |
# return the time offset for duration, freq, phase, fm_mod |
595 |
sub _dur2time($$;$$) { |
596 |
my ($dur, $freq, $phase, $fm_mod) = @_; |
597 |
zeroes($dur)->xvals*$freq+cumusumover($fm_mod)+$phase |
598 |
} |
599 |
|
600 |
sub gen_oscil($$;$$) { |
601 |
my ($dur, $freq, $phase, $fm_mod) = @_; |
602 |
sin M_2PI*_dur2time($dur,$freq,$phase,$fm_mod); |
603 |
} |
604 |
|
605 |
sub gen_sawtooth($$;$$) { |
606 |
my ($dur, $freq, $phase, $fm_mod) = @_; |
607 |
# maybe better use abs? |
608 |
$freq = $fm_mod * $freq if defined $fm_mod; |
609 |
(_dur2time($dur,$freq,$phase,$fm_mod) % 1 - 0.5) * 2; |
610 |
} |
611 |
|
612 |
sub gen_square($$;$$$) { |
613 |
my ($dur, $freq, $phase, $duty, $fm_mod) = @_; |
614 |
(gen_sawtooth($dur,$freq,$phase,$fm_mod) < $duty) * 2 - 1; |
615 |
} |
616 |
|
617 |
sub gen_triangle($$;$$) { |
618 |
my ($dur, $freq, $phase, $fm_mod) = @_; |
619 |
my $st = gen_sawtooth($dur,$freq,$phase,$fm_mod); |
620 |
$st * (($st < 0) * 2 - 1); |
621 |
} |
622 |
|
623 |
sub gen_pulse_train($$;$$) { |
624 |
my ($dur, $freq, $phase, $fm_mod) = @_; |
625 |
my $st = gen_sawtooth($dur,$freq,$phase,$fm_mod); |
626 |
$st < $st->rotate(1); |
627 |
} |
628 |
|
629 |
sub gen_rand($$) { |
630 |
my ($dur, $freq) = @_; |
631 |
my $z = zeroes $dur; |
632 |
my $r = (random zeroes $freq * $z->getdim(0) + 1) * 2 - 1; |
633 |
$r->index($freq * $z->xvals)->sever; |
634 |
} |
635 |
|
636 |
sub gen_rand_1f($) { |
637 |
my ($dur) = @_; |
638 |
$dur = zeroes $dur; |
639 |
_gen_noise_1f($dur); |
640 |
$dur; |
641 |
} |
642 |
|
643 |
=head2 gen_env duration*, xvals, yvals, [base] |
644 |
|
645 |
Generates an interpolated envelope between the points given by xvals and |
646 |
yvals. When base == 1 (the default) then the values will be linearly |
647 |
interpolated, otherwise they follow an exponential curve that is bend |
648 |
inwards (base < 1) or outwards (base > 1). |
649 |
|
650 |
# generate a linear envelope with attack in the first 10% |
651 |
gen_env 5000, [0 1 2 9 10], [0 1 0.6 0.6 0]; |
652 |
|
653 |
=cut |
654 |
|
655 |
sub gen_env($;@) { |
656 |
my ($dur, $x, $y, $base) = @_; |
657 |
my $pdl = zeroes $dur; |
658 |
$base ||= 1; |
659 |
$base == 1 or barf "base values != 1 are not yet implemented\n"; |
660 |
my @x = ref $x eq "ARRAY" ? @$x : $x->list; |
661 |
my @y = ref $y eq "ARRAY" ? @$y : $y->list; |
662 |
my $f = $pdl->getdim(0) / $x[-1]; |
663 |
@x == @y or barf "gen_env: x and y lists have different sizes"; |
664 |
my $x0 = 0; |
665 |
my $y0 = 0; |
666 |
for (0..$#x) { |
667 |
my ($x,$y) = (int($x[$_]*$f),$y[$_]); |
668 |
if ($x > $x0) { |
669 |
my $a = $pdl->slice("$x0:".($x-1)); |
670 |
if ($y0 != $y) { |
671 |
$a .= $a->xlinvals($y0,$y); |
672 |
} else { |
673 |
$a .= $y; |
674 |
} |
675 |
} |
676 |
($x0,$y0)=($x,$y); |
677 |
} |
678 |
$pdl; |
679 |
} |
680 |
|
681 |
=head2 gen_adsr duration*, sustain-level, attack-time, decay-time, sustain-time, release-time |
682 |
|
683 |
Simple ADSR envelope generator. The C<sustain-level> is the amplitude (0 |
684 |
to 1) of the sustain level. The other for parameters give the relative |
685 |
interval times, in any unit you like, only their relative ratios |
686 |
are important. Any of these times might be zero, in which case the |
687 |
corresponding part is omitted from the envelope. |
688 |
|
689 |
=cut |
690 |
|
691 |
sub gen_adsr($$$$$$) { |
692 |
my ($dur, $amp, $a, $d, $s, $r) = @_; |
693 |
my (@x, @y); |
694 |
my $t = 0; |
695 |
push @x, 0; push @y, 0; |
696 |
if ($a > 0) { push @x, $t += $a; push @y, 1 } else { $y[-1] = 1 } |
697 |
if ($d > 0) { push @x, $t += $d; push @y, $amp } else { $y[-1] = $amp } |
698 |
if ($s > 0) { push @x, $t += $s; push @y, $amp } else { $y[-1] = $amp } |
699 |
if ($r > 0) { push @x, $t += $r; push @y, 0 } else { $y[-1] = 0 } |
700 |
zeroes($dur)->xlinvals(0,$t)->linear_interpolate(pdl(@x), pdl(@y)); |
701 |
} |
702 |
|
703 |
=head2 gen_asymmetric_fm duration*, freq/, phase, [r , [ratio]] |
704 |
|
705 |
C<gen_asymmetric_fm> provides a way around the symmetric spectra normally |
706 |
produced by FM. See Palamin and Palamin, "A Method of Generating and |
707 |
Controlling Asymmetrical Spectra" JAES vol 36, no 9, Sept 88, p671-685. |
708 |
|
709 |
=cut |
710 |
|
711 |
sub gen_asymmetric_fm($$;$$$) { |
712 |
my ($dur, $freq, $phase, $r, $ratio) = @_; |
713 |
$r ||= 1; |
714 |
$ratio ||= 1; |
715 |
my $cosr = 0.5 * ($r - 1/$r); |
716 |
my $sinr = 0.5 * ($r + 1/$r); |
717 |
$phase = zeroes($dur)->xvals*$freq + $phase; |
718 |
my $mth = $phase * $ratio; |
719 |
exp ($cosr * cos $mth) * sin ($phase + $sinr * sin $mth); |
720 |
} |
721 |
|
722 |
=head2 gen_sum_of_cosines duration*, freq/, phase, ncosines, [fm_mod/] |
723 |
|
724 |
Generates a sum of C<n> cosines C<(1 + 2(cos(x) + cos(2x) + ... cos(nx)) = |
725 |
sin((n+.5)x) / sin(x/2))>. Other arguments are similar to to C<gen_oscil>. |
726 |
|
727 |
=head2 gen_sine_summation duration*, freq/, phase, [nsines, [a, [b_ratio, [fm_mod/]]]] |
728 |
|
729 |
C<gen_sine_summation> provides a kind of additive synthesis. See |
730 |
J.A.Moorer, "Signal Processing Aspects of Computer Music" and "The |
731 |
Synthesis of Complex Audio Spectra by means of Discrete Summation |
732 |
Formulae" (Stan-M-5). The basic idea is very similar to that used in |
733 |
gen_sum_of_cosines generator. |
734 |
|
735 |
The default value for C<nsines> is 1 (but zero is a valid value), for C<a> |
736 |
is 0.5 and for C<b_ratio> is 1. |
737 |
|
738 |
(btw, either my formula is broken or the output indeed does not lie |
739 |
between -1 and +1, but rather -5 .. +5). |
740 |
|
741 |
=cut |
742 |
|
743 |
sub gen_sum_of_cosines { |
744 |
my ($dur, $freq, $phase, $cosines, $fm_mod) = @_; |
745 |
$cosines ||= 1; |
746 |
$dur = M_2PI*_dur2time($dur,$freq,$phase+1e-6,$fm_mod); |
747 |
sin ($dur * ($cosines+0.5)) / (sin ($dur * 0.5) * (1+2*$cosines)); |
748 |
} |
749 |
|
750 |
sub gen_sine_summation { |
751 |
my ($dur, $freq, $phase, $n, $a, $ratio, $fm_mod) = @_; |
752 |
$n = 1 unless defined $n; |
753 |
$a ||= 0.5; |
754 |
$ratio ||= 1; |
755 |
|
756 |
$dur = M_2PI*_dur2time($dur,$freq,$phase+1e-6,$fm_mod); |
757 |
my $an = $n ? $a**($n+1) : 0; |
758 |
my $B = $dur * $ratio; |
759 |
(sin ($dur) |
760 |
- $a * sin ($dur - $B) |
761 |
- $an * (sin ($dur + $B*($n+1)) |
762 |
- $a * sin ($dur + $B*$n))) |
763 |
/ (1 + $a*$a - (2*$a*cos($B))); |
764 |
} |
765 |
|
766 |
=head2 gen_from_table duration*, frequency/, table, [phase], [fm_mod/] |
767 |
|
768 |
C<gen_from_table> generates a waveform by repeating a waveform given |
769 |
in C<table>, linearly interpolating between successive points of the |
770 |
C<waveform>. |
771 |
|
772 |
=head2 partials2waveshape size*, partials, amplitudes, [phase], [fm_mod/] |
773 |
|
774 |
Take a (perl or pdl) list of (integer) C<partials> and a list of |
775 |
C<amplitudes> and generate a single wave shape that results by adding |
776 |
these partial sines. |
777 |
|
778 |
This could (and should) be used by the C<gen_from_table> generator. |
779 |
|
780 |
=head2 gen_from_partials duration*, frequency/, partials, amplitudes, [phase], [fm_mod/] |
781 |
|
782 |
Take a (perl list or pdl) list of (possibly noninteger) C<partials> and a |
783 |
list of C<amplitudes> and generate the waveform resulting by summing up |
784 |
all these partial sines. |
785 |
|
786 |
=cut |
787 |
|
788 |
sub gen_from_table { |
789 |
my ($dur, $freq, $table, $phase, $fm_mod) = @_; |
790 |
my $r = _dur2time($dur,$freq,$phase,$fm_mod); |
791 |
my $tx = ($table->dims)[0]; |
792 |
$table = cat($table, $table->at(0))->clump(-1); |
793 |
$r %= 1; $r += 1; $r %= 1; $r *= $tx; |
794 |
$r->linear_interpolate($table->xvals, $table); |
795 |
} |
796 |
|
797 |
sub partials2waveshape { |
798 |
my $dur = zeroes $_[0]; |
799 |
my @idx = ref $_[1] eq "ARRAY" ? @{$_[1]} : $_[1]->list; |
800 |
my @amp = ref $_[2] eq "ARRAY" ? @{$_[2]} : $_[2]->list; |
801 |
|
802 |
my $angle = $dur->xvals * (M_2PI/$dur->getdim(0)); |
803 |
|
804 |
for (0..$#idx) { |
805 |
$dur += $amp[$_] * sin $idx[$_]*$angle; |
806 |
} |
807 |
$dur / max abs $dur; |
808 |
} |
809 |
|
810 |
sub gen_from_partials { |
811 |
my ($dur, $freq, $ti, $ta, $phase, $fm_mod) = @_; |
812 |
my @idx = ref $ti eq "ARRAY" ? @{$ti} : $ti->list; |
813 |
my @amp = ref $ta eq "ARRAY" ? @{$ta} : $ta->list; |
814 |
|
815 |
$dur = zeroes $dur; |
816 |
|
817 |
my $angle = _dur2time($dur,$freq,$phase,$fm_mod); |
818 |
|
819 |
for (0..$#idx) { |
820 |
$dur += $amp[$_] * fast_sin($idx[$_]*$angle); |
821 |
} |
822 |
$dur / max abs $dur; |
823 |
} |
824 |
|
825 |
EOP |
826 |
|
827 |
pp_def '_gen_noise_1f', |
828 |
Pars => '[o]f(n)', |
829 |
GenericTypes => [F,D], |
830 |
Doc => undef, |
831 |
PMCode => undef, |
832 |
Code => q^ |
833 |
static unsigned long ctz[64] = { |
834 |
6, 0, 1, 0, 2, 0, 1, 0, |
835 |
3, 0, 1, 0, 2, 0, 1, 0, |
836 |
4, 0, 1, 0, 2, 0, 1, 0, |
837 |
3, 0, 1, 0, 2, 0, 1, 0, |
838 |
5, 0, 1, 0, 2, 0, 1, 0, |
839 |
3, 0, 1, 0, 2, 0, 1, 0, |
840 |
4, 0, 1, 0, 2, 0, 1, 0, |
841 |
3, 0, 1, 0, 2, 0, 1, 0, |
842 |
}; |
843 |
static unsigned int dice[7]; |
844 |
static unsigned int total = 0; |
845 |
unsigned int n, prevrand, newrand, seed, k; |
846 |
|
847 |
threadloop %{ |
848 |
for (n = 0; n < $SIZE(n); n++) |
849 |
{ |
850 |
k = ctz [n & 63]; |
851 |
|
852 |
prevrand = dice[k]; |
853 |
seed = 1664525 * seed + 1013904223; |
854 |
newrand = (seed >> 3); |
855 |
dice[k] = newrand; |
856 |
|
857 |
total += newrand - prevrand; |
858 |
|
859 |
seed = 1103515245 * seed + 12345; |
860 |
newrand = (seed >> 3); |
861 |
|
862 |
$f(n=>n) = ($GENERIC())(total + newrand) * (1./(3<<29)) - 1; |
863 |
} |
864 |
%} |
865 |
^ |
866 |
; |
867 |
|
868 |
pp_def 'filter_one_zero', |
869 |
Pars => 'in(n); [o] out(n)', |
870 |
OtherPars => 'double a0; double a1', |
871 |
GenericTypes => [F,D], |
872 |
Doc => 'apply a one zero filter, y(n) = a0 x(n) + a1 x(n-1)', |
873 |
Code => q! |
874 |
double a0 = $COMP(a0); |
875 |
double a1 = $COMP(a1); |
876 |
double x1 = 0.; |
877 |
|
878 |
threadloop %{ loop(n) %{ |
879 |
$out() = a0 * $in() + a1 * x1; |
880 |
x1 = $in(); |
881 |
%} %} |
882 |
! |
883 |
; |
884 |
|
885 |
pp_def 'filter_one_pole', |
886 |
Pars => 'in(n); [o] out(n)', |
887 |
OtherPars => 'double a0; double b1', |
888 |
GenericTypes => [F,D], |
889 |
Doc => 'apply a one pole filter, y(n) = a0 x(n) - b1 y(n-1)', |
890 |
Code => q! |
891 |
double a0 = $COMP(a0); |
892 |
double b1 = $COMP(b1); |
893 |
double y2 = 0.; |
894 |
|
895 |
threadloop %{ loop(n) %{ |
896 |
$out() = y2 = a0 * $in() - b1 * y2; |
897 |
%} %} |
898 |
! |
899 |
; |
900 |
|
901 |
pp_def 'filter_two_zero', |
902 |
Pars => 'in(n); [o] out(n)', |
903 |
OtherPars => 'double a0; double a1; double a2', |
904 |
GenericTypes => [F,D], |
905 |
Doc => 'apply a two zero filter, y(n) = a0 x(n) + a1 x(n-1) + a2 x(n-2)', |
906 |
Code => q! |
907 |
double a0 = $COMP(a0); |
908 |
double a1 = $COMP(a1); |
909 |
double a2 = $COMP(a2); |
910 |
double x1 = 0.; |
911 |
double x2 = 0.; |
912 |
|
913 |
threadloop %{ loop(n) %{ |
914 |
$out() = a0 * $in() + a1 * x1 + a2 * x2; |
915 |
x2 = x1; x1 = $in(); |
916 |
%} %} |
917 |
! |
918 |
; |
919 |
|
920 |
pp_def 'filter_two_pole', |
921 |
Pars => 'in(n); [o] out(n)', |
922 |
OtherPars => 'double a0; double b1; double b2', |
923 |
GenericTypes => [F,D], |
924 |
Doc => 'apply a two pole filter, y(n) = a0 x(n) - b1 y(n-1) - b2 y(n-2)', |
925 |
Code => q! |
926 |
double a0 = $COMP(a0); |
927 |
double b1 = $COMP(b1); |
928 |
double b2 = $COMP(b2); |
929 |
double y1 = 0.; |
930 |
double y2 = 0.; |
931 |
|
932 |
threadloop %{ loop(n) %{ |
933 |
$out() = a0 * $in() - b1 * y1 - b2 * y2; |
934 |
y2 = y1; y1 = $out(); |
935 |
%} %} |
936 |
! |
937 |
; |
938 |
|
939 |
pp_def 'filter_formant', |
940 |
Pars => 'in(n); [o] out(n)', |
941 |
OtherPars => 'double radius; double frequency; double gain', |
942 |
GenericTypes => [F,D], |
943 |
Doc => 'apply a formant filter, y(n) = x(n) - r*x(n-2) + 2*r*cos(2*pi*frequency)*y(n-1) - r*r*y(n-2). A good value for C<gain> is 1.', |
944 |
Code => q! |
945 |
threadloop %{ |
946 |
double radius = 1. - $COMP(radius); |
947 |
double a0 = $COMP(gain) * sin ($COMP(frequency) * M_2PI) * (1. - (radius * radius)); |
948 |
double a2 = -radius; |
949 |
double b1 = -2. * radius * cos ($COMP(frequency) * M_2PI); |
950 |
double b2 = radius * radius; |
951 |
double x1 = 0.; |
952 |
double x2 = 0.; |
953 |
double y1 = 0.; |
954 |
double y2 = 0.; |
955 |
|
956 |
loop(n) %{ |
957 |
double inval = a0 * $in(); |
958 |
$out() = inval + a2 * x2 - b1 * y1 - b2 * y2; |
959 |
y2 = y1; y1 = $out(); |
960 |
x2 = x1; x1 = inval; |
961 |
%} |
962 |
%} |
963 |
! |
964 |
; |
965 |
|
966 |
pp_addpm <<'EOP'; |
967 |
|
968 |
=head2 filter_ppolar pdl, radius/, frequency/ |
969 |
|
970 |
apply a two pole filter (given in polar form). The filter has two poles, |
971 |
one at (radius,frequency), the other at (radius,-frequency). Radius |
972 |
is between 0 and 1 (but less than 1), and frequency is between 0 and |
973 |
0.5. This is the standard resonator form with poles specified by the |
974 |
polar coordinates of one pole. |
975 |
|
976 |
=head2 filter_zpolar pdl, radius/, frequency/ |
977 |
|
978 |
apply a two zero filter (given in polar form). See C<filter_ppolar>. |
979 |
|
980 |
=cut |
981 |
|
982 |
sub filter_ppolar { |
983 |
my ($pdl, $radius, $freq) = @_; |
984 |
$pdl->filter_two_pole(1, -2*$radius*cos($freq*M_2PI), $radius**2); |
985 |
} |
986 |
|
987 |
sub filter_zpolar { |
988 |
my ($pdl, $radius, $freq) = @_; |
989 |
$pdl->filter_two_zero(1, -2*$radius*cos($freq*M_2PI), $radius**2); |
990 |
} |
991 |
|
992 |
=head2 partials2polynomial partials, [kind] |
993 |
|
994 |
C<partials2polynomial> takes a list of harmonic amplitudes and returns a |
995 |
list of Chebychev polynomial coefficients. The argument C<kind> determines |
996 |
which kind of Chebychev polynomial we are interested in, 1st kind or 2nd |
997 |
kind. (default is 1). |
998 |
|
999 |
=cut |
1000 |
|
1001 |
sub partials2polynomial { |
1002 |
my ($partials, $kind) = @_; |
1003 |
$kind = 1 unless defined $kind; |
1004 |
my $t0 = zeroes($partials->getdim(0)+1); |
1005 |
my $t1 = $t0->copy; |
1006 |
my $cc1 = $t0->copy; |
1007 |
|
1008 |
$t0->set(0,1); |
1009 |
$t1->set(1,$kind); |
1010 |
|
1011 |
for my $amp ($partials->list) { |
1012 |
$cc1 += $amp * $t1; |
1013 |
$tn = 2 * $t1->rshift(1) - $t0; |
1014 |
$t0 = $t1; |
1015 |
$t1 = $tn; |
1016 |
} |
1017 |
$cc1; |
1018 |
} |
1019 |
|
1020 |
=head2 ring_modulate in1, in2 |
1021 |
|
1022 |
ring modulates in1 with in2 (this is just a multiply). |
1023 |
|
1024 |
=head2 amplitude_modulate am_carrier, in1, in2 |
1025 |
|
1026 |
amplitude modulates am_carrier and in2 with in1 (this calculates in1 * (am_carrier + in2)). |
1027 |
|
1028 |
=cut |
1029 |
|
1030 |
sub ring_modulate { |
1031 |
$_[0] * $_[1]; |
1032 |
} |
1033 |
|
1034 |
sub amplitude_modulate { |
1035 |
$_[1] * ($_[0] + $_[2]); |
1036 |
} |
1037 |
|
1038 |
EOP |
1039 |
|
1040 |
pp_def 'filter_sir', |
1041 |
Pars => 'x(n); a(an); b(bn); [o]y(n)', |
1042 |
GenericTypes => [L,F,D], |
1043 |
Doc => <<'EOD', |
1044 |
Generic (short delay) impulse response filter. C<x> is the input signal |
1045 |
(which is supposed to be zero for negative indices). C<a> contains the |
1046 |
input (x) coefficients (a0, a1, .. an), whereas C<b> contains the output |
1047 |
(y) coefficients (b0, b1, ... bn), i.e.: |
1048 |
|
1049 |
y(n) = a0 x(n) - b1 y(n-1) + a1 x(n-1) - b2 y(n-2) + a2 x(n-2) - b3 ... |
1050 |
|
1051 |
This can be used to generate fir and iir filters of any length, or even |
1052 |
more complicated constructs. |
1053 |
|
1054 |
C<b0> (then first element of C<b>) is being ignored currently AND SHOULD |
1055 |
BE SPECIFIED AS ONE FOR FUTURE COMPATIBILITY |
1056 |
EOD |
1057 |
Code => q^ |
1058 |
int an = $SIZE(an); |
1059 |
int bn = $SIZE(bn); |
1060 |
int i, n, I; |
1061 |
$GENERIC() v; |
1062 |
|
1063 |
threadloop %{ |
1064 |
for (n = 0; n < $SIZE(n); n++) |
1065 |
{ |
1066 |
v = 0; |
1067 |
|
1068 |
/* apply loop splitting manually later! */ |
1069 |
|
1070 |
for (i = 0; i < an; i++) |
1071 |
{ |
1072 |
I = n-i; |
1073 |
if (I >= 0) |
1074 |
v += $a(an=>i) * $x(n=>I); |
1075 |
} |
1076 |
|
1077 |
for (i = 0; i < bn; i++) |
1078 |
{ |
1079 |
I = n-i; |
1080 |
if (I >= 0) |
1081 |
v -= $b(bn=>i) * $y(n=>I); |
1082 |
} |
1083 |
|
1084 |
$y(n=>n) = v; |
1085 |
} |
1086 |
%} |
1087 |
^ |
1088 |
; |
1089 |
|
1090 |
pp_def 'filter_lir', |
1091 |
Pars => 'x(n); int a_x(an); a_y(an); int b_x(bn); b_y(bn); [o]y(n)', |
1092 |
GenericTypes => [L,F,D], |
1093 |
Doc => <<'EOD', |
1094 |
|
1095 |
Generic (long delay) impulse response filter. The difference to |
1096 |
C<filter_sir> is that the filter coefficients need not be consecutive, |
1097 |
but instead their indices are given by the C<a_x> and C<b_x> (integer) |
1098 |
vectors, while the corresponding coefficients are in C<a_y> and |
1099 |
C<b_y>. (All C<a_x> must be >= 0, while all the C<b_x> must be >= 1, as |
1100 |
you should expect). |
1101 |
|
1102 |
See C<filter_sir> for more info. |
1103 |
EOD |
1104 |
Code => q^ |
1105 |
int i, n, I; |
1106 |
$GENERIC() v; |
1107 |
|
1108 |
threadloop %{ |
1109 |
for (n = 0; n < $SIZE(n); n++) |
1110 |
{ |
1111 |
v = 0; |
1112 |
|
1113 |
loop(an) %{ |
1114 |
I = n-$a_x(); |
1115 |
if (I >= 0) |
1116 |
v += $a_y() * $x(n=>I); |
1117 |
%} |
1118 |
|
1119 |
loop(bn) %{ |
1120 |
I = n-$b_x(); |
1121 |
if (I >= 0) |
1122 |
v -= $b_y() * $y(n=>I); |
1123 |
%} |
1124 |
|
1125 |
$y(n=>n) = v; |
1126 |
} |
1127 |
%} |
1128 |
^ |
1129 |
; |
1130 |
|
1131 |
pp_addpm <<'EOP'; |
1132 |
|
1133 |
=head2 filter_fir input, xcoeffs |
1134 |
|
1135 |
Apply a fir (finite impulse response) filter to C<input>. This is the same as |
1136 |
calling: |
1137 |
|
1138 |
filter_sir input, xcoeffs, pdl() |
1139 |
|
1140 |
=head2 filter_iir input, ycoeffs |
1141 |
|
1142 |
Apply a iir (infinite impulse response) filter to C<input>. This is just |
1143 |
another way of saying: |
1144 |
|
1145 |
filter_sir input, pdl(1), ycoeffs |
1146 |
|
1147 |
That is, the first member of C<ycoeffs> is being ignored AND SHOULD BE |
1148 |
SPECIFIED AS ONE FOR FUTURE COMPATIBILITY! |
1149 |
|
1150 |
=cut |
1151 |
|
1152 |
sub filter_fir($$) { |
1153 |
filter_sir ($_[0], $_[1], PDL->pdl()); |
1154 |
} |
1155 |
|
1156 |
sub filter_iir($$) { |
1157 |
filter_sir ($_[0], PDL->pdl(1), $_[1]); |
1158 |
} |
1159 |
|
1160 |
=head2 filter_comb input, delay*, scaler |
1161 |
|
1162 |
Apply a comb filter to the piddle C<input>. This is implemented using a |
1163 |
delay line of length C<delay> (which must be 1 or larger and can be |
1164 |
non-integer) and a feedback scaler. |
1165 |
|
1166 |
y(n) = x(n-size-1) + scaler * y(n-size) |
1167 |
|
1168 |
cf. C<filter_notch> and http://www.harmony-central.com/Effects/Articles/Reverb/comb.html |
1169 |
|
1170 |
=head2 filter_notch input, delay*, scaler |
1171 |
|
1172 |
Apply a comb filter to the piddle C<input>. This is implemented using a |
1173 |
delay line of length C<delay> (which must be 1 or larger and can be |
1174 |
non-integer) and a feedforward scaler. |
1175 |
|
1176 |
y(n) = x(n-size-1) * scaler + y(n-size) |
1177 |
|
1178 |
As a rule of thumb, the decay time of the feedback part is |
1179 |
C<7*delay/(1-scaler)> samples, so to get a decay of Dur seconds, |
1180 |
C<scaler <= 1-7*delay/(Dur*Srate)>. The peak gain is C<1/(1-(abs |
1181 |
scaler))>. The peaks (or valleys in notch's case) are evenly spaced at |
1182 |
C<srate/delay>. The height (or depth) thereof is determined by scaler |
1183 |
-- the closer to 1.0, the more pronounced. See Julius Smith's "An |
1184 |
Introduction to Digital Filter Theory" in Strawn "Digital Audio Signal |
1185 |
Processing", or Smith's "Music Applications of Digital Waveguides" |
1186 |
|
1187 |
=head2 filter_allpass input, delay*, scaler-feedback, scaler-feedforward |
1188 |
|
1189 |
C<filter_allpass> or "moving average comb" is just like C<filter_comb> |
1190 |
but with an added feedforward term. If C<scaler-feedback == 0>, we get a |
1191 |
moving average comb filter. If both scaler terms == 0, we get a pure delay |
1192 |
line. |
1193 |
|
1194 |
y(n) = feedforward*x(n-1) + x(n-size-1) + feedback*y(n-size) |
1195 |
|
1196 |
cf. http://www.harmony-central.com/Effects/Articles/Reverb/allpass.html |
1197 |
|
1198 |
=cut |
1199 |
|
1200 |
sub _filter_combnotchall { |
1201 |
my ($pdl, $size, $scalercomb, $scalernotch, $scalerall) = @_; |
1202 |
my $D = int $size; |
1203 |
my $F = $size - $D; |
1204 |
my $a = zeroes $D+2; $a->set($D,1-$F); $a->set($D+1, $F); |
1205 |
my $b = zeroes $D+2; $b->set($D,$F-1); $b->set($D+1,-$F); |
1206 |
$a->set(1, $a->at(1) + $scalerall); |
1207 |
$b->set(0, 1); |
1208 |
$pdl->filter_sir($a*$scalernotch,$b*$scalercomb); |
1209 |
} |
1210 |
|
1211 |
sub filter_comb { |
1212 |
_filter_combnotchall $_[0], $_[1], $_[2], 1, 0; |
1213 |
} |
1214 |
|
1215 |
sub filter_notch { |
1216 |
_filter_combnotchall $_[0], $_[1], 1, $_[2], 0; |
1217 |
} |
1218 |
|
1219 |
sub filter_allpass { |
1220 |
_filter_combnotchall $_[0], $_[1], $_[2], 1, $_[3]; |
1221 |
} |
1222 |
|
1223 |
=head2 design_remez_fir filter_size, bands(2,b), desired_gain(b), type, [weight(b)] |
1224 |
|
1225 |
Calculates the optimal (in the Chebyshev/minimax sense) FIR filter |
1226 |
impulse response given a set of band edges, the desired reponse on those |
1227 |
bands, and the weight given to the error in those bands, using the |
1228 |
Parks-McClellan exchange algorithm. |
1229 |
|
1230 |
The first argument sets the filter size: C<design_remez_fir> returns as |
1231 |
many coefficients as specified by this parameter. |
1232 |
|
1233 |
C<bands> is a vector of band edge pairs (start - end), which specify the |
1234 |
start and end of the bands in the filter specification. These must be |
1235 |
non-overlapping and sorted in increasing order. Only values between C<0> |
1236 |
(0 Hz) and C<0.5> (the Nyquist frequency) are allowed. |
1237 |
|
1238 |
C<des> specifies the desired gain in these bands. |
1239 |
|
1240 |
C<weight> can be used to give each band a different weight. If absent, a |
1241 |
vector of ones is used. |
1242 |
|
1243 |
C<type> is any of the exported constants C<BANDPASS>, C<DIFFERENTIATOR> or |
1244 |
C<HILBERT> and can be used to select various design types (use C<BANDPASS> |
1245 |
until this is documented ;) |
1246 |
|
1247 |
=cut |
1248 |
|
1249 |
sub design_remez_fir { |
1250 |
my ($size, $bands, $des, $type, $weight) = @_; |
1251 |
$weight = ones $des unless defined $weight; |
1252 |
$size = zeroes $size; |
1253 |
_design_remez_fir($bands, $des, $weight, $size, $type); |
1254 |
$size; |
1255 |
} |
1256 |
|
1257 |
=head2 filter_src input, srate, [width], [sr-mod] |
1258 |
|
1259 |
Generic sampling rate conversion, implemented by convoluting C<input> with |
1260 |
a sinc function of size C<width> (default when unspecified or zero: 5). |
1261 |
|
1262 |
C<srate> determines the input rate / output rate ratio, i.e. values > 1 |
1263 |
speed up, values < 1 slow down. Values < 0 are allowed and reverse the |
1264 |
signal. |
1265 |
|
1266 |
If C<sr_mod> is omitted, the size of the output piddle is calculcated |
1267 |
as C<length(input)/abs(srate)>, e.g. it provides the full stretched or |
1268 |
shrinked input signal. |
1269 |
|
1270 |
If C<sr_mod> is specified it must be as large as the desired output, i.e. |
1271 |
it's size determines the output size. Each value in C<sr_mod> is added to |
1272 |
C<srate> at the given point in "time", so it can be used to "modulate" the |
1273 |
sampling rate change. |
1274 |
|
1275 |
# create a sound effect in the style of "Forbidden Planet" |
1276 |
$osc = 0.3 * gen_oscil $osc, 30 / $pdl->rate; |
1277 |
$output = filter_src($input, 1, 0, $osc); |
1278 |
|
1279 |
=cut |
1280 |
|
1281 |
sub filter_src($$;$$) { |
1282 |
my ($input, $srate, $width, $sr_mod) = @_; |
1283 |
|
1284 |
$width ||= 5; |
1285 |
$width*2 < $input->getdim(0) or barf "src width too large (> 0.5 * input size)"; |
1286 |
|
1287 |
my $output; |
1288 |
|
1289 |
if (defined $sr_mod) { |
1290 |
$output = zeroes $sr_mod; |
1291 |
} else { |
1292 |
$output = zeroes ($input->getdim(0) / abs($srate)); |
1293 |
$sr_mod = PDL->null; |
1294 |
} |
1295 |
|
1296 |
_filter_src($input, $output, $sr_mod, $srate, $width); |
1297 |
|
1298 |
$output->rate($input->rate / abs($srate)) if $input->rate; |
1299 |
$output; |
1300 |
} |
1301 |
|
1302 |
=head2 filter_contrast_enhance input, enhancement |
1303 |
|
1304 |
Contrast-enhancement phase-modulates a sound file. It's like audio |
1305 |
MSG. The actual algorithm is (applied to the normalised sound) |
1306 |
C<sin(input*pi/2 + (enhancement*sin(input*2*pi)))>. The result is to |
1307 |
brighten the sound, helping it cut through a huge mix. |
1308 |
|
1309 |
=cut |
1310 |
|
1311 |
sub filter_contrast_enhance { |
1312 |
my ($pdl, $idx) = @_; |
1313 |
$idx = 0.1 unless defined $idx; |
1314 |
$pdl = $pdl - min $pdl; |
1315 |
$pdl = ($pdl * (M_PI / max $pdl)) - M_PI * 0.5; |
1316 |
sin $pdl + $idx * sin $pdl*4; |
1317 |
} |
1318 |
|
1319 |
=head2 filter_granulate input, expansion, [option-hash], option => value... |
1320 |
|
1321 |
C<filter_granulate> "granulates" the sound file file. It is the poor |
1322 |
man's way to change the speed at which things happen in a recorded sound |
1323 |
without changing the pitches. It works by slicing the input file into |
1324 |
short pieces, then overlapping these slices to lengthen (or shorten) the |
1325 |
result; this process is sometimes known as granular synthesis, and is |
1326 |
similar to the "freeze" function. The duration of each slice is C<length> -- |
1327 |
the longer, the more like reverb the effect. The portion of the length (on |
1328 |
a scale from 0 to 1.0) spent on each ramp (up or down) is C<ramp>. This can |
1329 |
control the smoothness of the result of the overlaps. The more-or-less |
1330 |
average time between successive segments is C<hop>. The accuracy at which we |
1331 |
handle this hopping is set by the float C<jitter> -- if C<jitter> is very small, |
1332 |
you may get an annoying tremolo. The overall amplitude scaler on each |
1333 |
segment is C<scaler> -- this is used to try to to avoid overflows as we add |
1334 |
all these zillions of segments together. C<expansion> determines the input |
1335 |
hop in relation to the output hop; an expansion-amount of C<2.0> should more |
1336 |
or less double the length of the original, whereas an expansion-amount |
1337 |
of C<1.0> should return something close to the original speed. |
1338 |
|
1339 |
The defaults for the arguments/options are: |
1340 |
|
1341 |
expansion 1.0 |
1342 |
length(*) 0.15 |
1343 |
scaler 0.6 |
1344 |
hop(*) 0.05 |
1345 |
ramp 0.4 |
1346 |
jitter(*) 0.5 |
1347 |
maxsize infinity |
1348 |
|
1349 |
The parameters/options marked with (*) actually depend on the sampling |
1350 |
rate, and are always multiplied by the C<rate> attribute of the |
1351 |
piddle internally. If the piddle lacks that attribute, 44100 is |
1352 |
assumed. NOTE: This is different to most other filters, but should be ok |
1353 |
since C<filter_granulate> only makes sense for audiofiles. |
1354 |
|
1355 |
=cut |
1356 |
|
1357 |
sub filter_granulate { |
1358 |
my $input = shift; |
1359 |
my $expansion = shift || 1; |
1360 |
my %hdr = (); %hdr = (%hdr, %{+shift}) if ref $_[0]; %hdr = (%hdr, @_) if @_; |
1361 |
|
1362 |
my $rate = $hdr{rate} || 44100; |
1363 |
my $length = ($hdr{length} || 0.15) * $rate; |
1364 |
my $scaler = $hdr{scaler} || 0.6; |
1365 |
my $hop = ($hdr{hop} || 0.05) * $rate; |
1366 |
my $ramp = $hdr{ramp} || 0.4; |
1367 |
my $jitter = ($hdr{jitter} || 0.5) * $rate; |
1368 |
my $maxsize = $hdr{maxsize}|| 0; |
1369 |
|
1370 |
$output = zeroes $expansion * $input->getdim(0); |
1371 |
|
1372 |
_filter_granulate($input, $output, |
1373 |
$expansion, $length, $scaler, |
1374 |
$hop, $ramp, $jitter, $maxsize); |
1375 |
$output; |
1376 |
} |
1377 |
|
1378 |
EOP |
1379 |
|
1380 |
pp_def '_design_remez_fir', |
1381 |
Pars => 'bands(2,b); des(b); weight(b); [o]h(n)', |
1382 |
OtherPars => 'int type', |
1383 |
GenericTypes => [D], |
1384 |
Doc => undef, |
1385 |
PMCode => undef, |
1386 |
Code => q^ |
1387 |
int type = $COMP(type); |
1388 |
|
1389 |
if (type != BANDPASS |
1390 |
&& type != DIFFERENTIATOR |
1391 |
&& type != HILBERT) |
1392 |
barf ("design_remez_fir: illegal type specified (none of BANDPASS, DIFFERENTIATOR, HILBERT)"); |
1393 |
|
1394 |
remez ($P(h), $SIZE(n), $SIZE(b), $P(bands), $P(des), $P(weight), type); |
1395 |
^ |
1396 |
; |
1397 |
|
1398 |
pp_def '_filter_src', |
1399 |
Pars => 'input(n); [o]output(m); sr_mod(m2)', |
1400 |
OtherPars => 'double srate; int width', |
1401 |
GenericTypes => [D], |
1402 |
Doc => undef, |
1403 |
PMCode => undef, |
1404 |
Code => q! |
1405 |
mus_src ($P(input), $SIZE(n), $P(output), $SIZE(m), |
1406 |
$COMP(srate), |
1407 |
$SIZE(m) == $SIZE(m2) ? $P(sr_mod) : 0, |
1408 |
$COMP(width)); |
1409 |
! |
1410 |
; |
1411 |
|
1412 |
pp_def '_filter_granulate', |
1413 |
Pars => 'input(n); [o]output(m)', |
1414 |
OtherPars => 'double expansion; double length; double scaler; double hop; double ramp; double jitter; int max_size', |
1415 |
GenericTypes => [D], |
1416 |
Doc => undef, |
1417 |
PMCode => undef, |
1418 |
Code => q! |
1419 |
mus_granulate ($P(input), $SIZE(n), $P(output), $SIZE(m), |
1420 |
$COMP(expansion), $COMP(length), $COMP(scaler), |
1421 |
$COMP(hop), $COMP(ramp), $COMP(jitter), |
1422 |
$COMP(max_size)); |
1423 |
! |
1424 |
; |
1425 |
|
1426 |
pp_addpm <<'EOP'; |
1427 |
|
1428 |
=head2 audiomix pos1, data1, pos2, data2, ... |
1429 |
|
1430 |
Generate a mix of all given piddles. The resulting piddle will contain the |
1431 |
sum of all data-piddles at their respective positions, so some scaling |
1432 |
will be necessary before or after the mixing operation (e.g. scale2short). |
1433 |
|
1434 |
# mix the sound gong1 at position 0, the sound bass5 at position 22100 |
1435 |
# and gong2 at position 44100. The resulting piddle will be large enough |
1436 |
# to accomodate all the sounds: |
1437 |
$mix = audiomix 0, $gong1, 44100, $gong2, 22100, $gong2 |
1438 |
|
1439 |
=cut |
1440 |
|
1441 |
sub audiomix { |
1442 |
my(@x,@p); |
1443 |
my($dur,$i,$end,$s,$p); |
1444 |
for ($i = 0; $i < @_; $i += 2) { |
1445 |
$end = $_[$i] + $_[$i+1]->getdim(0); |
1446 |
$dur = $end if $end > $dur; |
1447 |
} |
1448 |
my $pdl = zeroes $dur; |
1449 |
for ($i = 0; $i < @_; $i += 2) { |
1450 |
($s,$p) = (int($_[$i]), $_[$i+1]); |
1451 |
($end = $pdl->slice("$s:".($s+$p->getdim(0)-1))) += $p; |
1452 |
} |
1453 |
$pdl; |
1454 |
} |
1455 |
|
1456 |
=head2 filter_center piddle |
1457 |
|
1458 |
Normalize the piddle so that it is centered around C<y = 0> and has |
1459 |
maximal amplitude of 1. |
1460 |
|
1461 |
=cut |
1462 |
|
1463 |
sub filter_center($) { |
1464 |
my $piddle = shift; |
1465 |
$piddle = $piddle - min $piddle; |
1466 |
$piddle * (2 / max $piddle) - 1; |
1467 |
} |
1468 |
|
1469 |
=head2 scale2short piddle |
1470 |
|
1471 |
This method takes a sound in any format (preferably float or double) and |
1472 |
scales it to fit into a signed short value, suitable for playback using |
1473 |
C<playudio> or similar functions. |
1474 |
|
1475 |
=cut |
1476 |
|
1477 |
sub scale2short { |
1478 |
my $pdl = shift->float; |
1479 |
($pdl * (1 / max abs $pdl) * 32767.5 - 0.5)->short; |
1480 |
} |
1481 |
|
1482 |
=head2 gen_fft_window size*, type, [$beta] |
1483 |
|
1484 |
Creates and returns a specific fft window. The C<type> is any of the following. |
1485 |
These are (case-insensitive) strings, so you might need to quote them. |
1486 |
|
1487 |
RECTANGULAR just ones (the identity window) |
1488 |
HANNING 0.50 - 0.50 * cos (0 .. 2pi) |
1489 |
HAMMING 0.54 - 0.46 * cos (0 .. 2pi) |
1490 |
WELCH 1 - (-1 .. 1) ** 2 |
1491 |
PARZEN the triangle window |
1492 |
BARTLETT the symmetric triangle window |
1493 |
BLACKMAN2 blackman-harris window of order 2 |
1494 |
BLACKMAN3 blackman-harris window of order 3 |
1495 |
BLACKMAN4 blackman-harris window of order 4 |
1496 |
EXPONENTIAL the exponential window |
1497 |
KAISER the kaiser/bessel window (using the parameter C<beta>) |
1498 |
CAUCHY the cauchy window (using the parameter <beta>) |
1499 |
POISSON the poisson window (exponential using parameter C<beta>) |
1500 |
RIEMANN the riemann window (sinc) |
1501 |
GAUSSIAN the gaussian window of order C<beta>) |
1502 |
TUKEY the tukey window (C<beta> specifies how much of the window |
1503 |
consists of ones). |
1504 |
COSCOS the cosine-squared window (a partition of unity) |
1505 |
SINC same as RIEMANN |
1506 |
HANN same as HANNING (his name was Hann, not Hanning) |
1507 |
|
1508 |
LIST this "type" is special in that it returns a list of all types |
1509 |
|
1510 |
=cut |
1511 |
|
1512 |
sub gen_fft_window { |
1513 |
my ($size, $type, $beta) = @_; |
1514 |
|
1515 |
$beta = 2.5 unless defined $beta; |
1516 |
|
1517 |
$size = $size->getdim(0) if ref $size; |
1518 |
$size > 2 or barf "fft window size too small"; |
1519 |
|
1520 |
my $midn = $size >> 1; |
1521 |
my $midm1 = ($size-1) >> 1; |
1522 |
my $midp1 = ($size+1) >> 1; |
1523 |
my $dur = zeroes $size; |
1524 |
my $sf = ($size-1)/$size; |
1525 |
%fft_window = ( |
1526 |
RECTANGULAR => sub { |
1527 |
$dur->ones |
1528 |
}, |
1529 |
HANNING => sub { |
1530 |
0.5 - 0.5 * cos $dur->xlinvals(0,M_2PI*$sf) |
1531 |
}, |
1532 |
HAMMING => sub { |
1533 |
0.53836 - 0.46164 * cos $dur->xlinvals(0,M_2PI*$sf) |
1534 |
}, |
1535 |
WELCH => sub { |
1536 |
1 - $dur->xlinvals(-1,$sf)**2; |
1537 |
}, |
1538 |
PARZEN => sub { |
1539 |
1 - abs $dur->xlinvals(-$sf,$sf); |
1540 |
}, |
1541 |
BARTLETT => sub { |
1542 |
1 - abs $dur->xlinvals(-1,1); |
1543 |
}, |
1544 |
BLACKMAN2 => sub { |
1545 |
my $cx = cos $dur->xlinvals(0,M_2PI*$sf); |
1546 |
0.34401 + ($cx * (-0.49755 + ($cx * 0.15844))); |
1547 |
}, |
1548 |
BLACKMAN3 => sub { |
1549 |
my $cx = cos $dur->xlinvals(0,M_2PI*$sf); |
1550 |
0.21747 + ($cx * (-0.45325 + ($cx * (0.28256 - ($cx * 0.04672))))); |
1551 |
}, |
1552 |
BLACKMAN4 => sub { |
1553 |
my $cx = cos $dur->xlinvals(0,M_2PI*$sf); |
1554 |
0.084037 + ($cx * (-0.29145 + ($cx * (0.375696 + ($cx * (-0.20762 + ($cx * 0.041194))))))); |
1555 |
}, |
1556 |
EXPONENTIAL => sub { |
1557 |
2 ** (1 - abs $dur->xlinvals(-1,$sf)) - 1; |
1558 |
}, |
1559 |
KAISER => sub { |
1560 |
bessi0 ($beta * sqrt(1 - $dur->xlinvals(-1,$sf)**2)) / bessi0 ($beta); |
1561 |
}, |
1562 |
CAUCHY => sub { |
1563 |
1 / (1 + ($dur->xlinvals(-1,$sf)*$beta)**2); |
1564 |
}, |
1565 |
POISSON => sub { |
1566 |
exp (-$beta * abs $dur->xlinvals(-1,$sf)); |
1567 |
}, |
1568 |
RIEMANN => sub { |
1569 |
my $dur1 = $dur->slice("0:$midm1"); |
1570 |
my $dur2 = $dur->slice("-1:$midp1:-1"); |
1571 |
my $cx; |
1572 |
$cx = $dur1->xlinvals(M_PI,M_2PI/$size); $dur1 .= sin ($cx) / $cx; |
1573 |
$cx = $dur2->xlinvals(M_PI,M_2PI/$size); $dur2 .= sin ($cx) / $cx; |
1574 |
($cx = $dur->slice("$midm1:$midp1")) .= 1; |
1575 |
$dur; |
1576 |
}, |
1577 |
GAUSSIAN => sub { |
1578 |
exp (-0.5 * ($beta * abs $dur->xlinvals(-1,$sf))**2); |
1579 |
}, |
1580 |
TUKEY => sub { |
1581 |
$beta >= 0 && $beta <= 1 or barf "beta must be between 0 and 1 for the tukey window"; |
1582 |
my $cx = int ($midn * (1 - $beta)); |
1583 |
my $cX = $size-$cx-1; |
1584 |
my $dur1 = $dur->slice("0:".($cx-1)); |
1585 |
my $dur2 = $dur->slice("-1:".($cX+1).":-1"); |
1586 |
my $dur3 = $dur->slice("$cx:$cX"); |
1587 |
|
1588 |
$dur1 .= 0.5 * (1 - cos $dur1->xlinvals(0, M_PI)); |
1589 |
$dur2 .= 0.5 * (1 - cos $dur2->xlinvals(0, M_PI)); |
1590 |
$dur3 .= 1; |
1591 |
|
1592 |
$dur; |
1593 |
}, |
1594 |
COSCOS => sub { |
1595 |
(cos $dur->xlinvals(M_PI/-2,M_PI/2))**2; |
1596 |
}, |
1597 |
LIST => sub { |
1598 |
grep $_ ne "LIST", keys %fft_window; |
1599 |
}, |
1600 |
); |
1601 |
|
1602 |
$fft_window{SINC} = $fft_window{RIEMANN}; |
1603 |
$fft_window{HANN} = $fft_window{HANNING}; |
1604 |
|
1605 |
$type = uc $type; |
1606 |
$fft_window{$type} or barf "$type: no such window"; |
1607 |
$fft_window{$type}->(); |
1608 |
} |
1609 |
|
1610 |
=head2 cplx(2,n) = rfft real(n) |
1611 |
|
1612 |
Do a (complex fft) of C<real> (extended to complex so that the imaginary |
1613 |
part is zero), and return the complex fft result. This function tries |
1614 |
to use L<PDL::FFTW|PDL::FFTW> (which is faster for large vectors) when |
1615 |
available, and falls back to L<PDL::FFT|PDL::FFT>, which is likely |
1616 |
to return different phase signs (due to different kernel functions), |
1617 |
so beware! In fact, since C<rfft> has to shuffle the data when using |
1618 |
PDL::FFT, the fallback is always slower. |
1619 |
|
1620 |
When using PDL::FFTW, a wisdom file ~/.pdl_wisdom is used and updated, if |
1621 |
possible. |
1622 |
|
1623 |
=head2 real(n) = irfft cplx(2,n) |
1624 |
|
1625 |
The inverse transformation (see C<rfft>). C<irfft rfft $pdl == $pdl> |
1626 |
always holds. |
1627 |
|
1628 |
=cut |
1629 |
|
1630 |
my $fftw_loaded; |
1631 |
sub _fftw { |
1632 |
defined $fftw_loaded or eval { |
1633 |
$fftw_loaded = 0; |
1634 |
require PDL::FFTW; |
1635 |
PDL::FFTW::load_wisdom("$ENV{HOME}/.pdl_wisdom") |
1636 |
if -r "$ENV{HOME}/.pdl_wisdom"; |
1637 |
$fftw_loaded = 1; |
1638 |
}; |
1639 |
$fftw_loaded; |
1640 |
} |
1641 |
|
1642 |
sub rfft { |
1643 |
my $data = $_[0]; |
1644 |
if (_fftw) { |
1645 |
my $x = $data->r2C; |
1646 |
$x = PDL::FFTW::fftw $x; |
1647 |
$x; |
1648 |
} else { |
1649 |
require PDL::FFT; |
1650 |
my @fft = ($data->copy, $data->zeroes); |
1651 |
PDL::FFT::fft(@fft); |
1652 |
cat(@fft)->xchg(0,1); |
1653 |
} |
1654 |
} |
1655 |
|
1656 |
sub irfft { |
1657 |
if (_fftw) { |
1658 |
$x = $_[0]->copy; |
1659 |
$x = PDL::FFTW::ifftw $x; |
1660 |
re $x / $x->getdim(1); |
1661 |
} else { |
1662 |
require PDL::FFT; |
1663 |
my @fft = $_[0]->xchg(0,1)->dog({Break => 1}); |
1664 |
PDL::FFT::ifft(@fft); |
1665 |
$fft[0]; |
1666 |
} |
1667 |
} |
1668 |
|
1669 |
=head2 spectrum data, [norm], [window], [beta] |
1670 |
|
1671 |
Returns the spectrum of a given pdl. If C<norm> is absent (or C<undef>), |
1672 |
it returns the magnitude of the fft of C<data>. When C<norm> == 1 (or |
1673 |
C<eq 'NORM'>, case-insensitive), it returns the magnitude, normalized to be |
1674 |
between zero and one. If C<norm> == 0 (or C<eq 'dB'>, case-insensitive), then |
1675 |
it returns the magnitude in dB. |
1676 |
|
1677 |
C<data> is multiplied with C<window> (if not C<undef>) before calculating |
1678 |
the fft, and usually contains a window created with C<gen_fft_window> |
1679 |
(using C<beta>). If C<window> is a string, it is handed over to |
1680 |
C<gen_fft_window> (together with the beta parameter) to create a window of |
1681 |
suitable size. |
1682 |
|
1683 |
This function could be slightly faster. |
1684 |
|
1685 |
=cut |
1686 |
|
1687 |
sub spectrum { |
1688 |
my ($data, $norm, $window, $beta) = @_; |
1689 |
my $len; |
1690 |
if (defined $window) { |
1691 |
$window = gen_fft_window ($data->getdim (0), $window, $beta) unless ref $window; |
1692 |
$data = $data * $window; |
1693 |
$len = $window->getdim (0); |
1694 |
} else { |
1695 |
$len = $data->getdim (0); |
1696 |
} |
1697 |
$data = rfft ( |
1698 |
$data->slice ("0:" . ($len - 1)) |
1699 |
->sever |
1700 |
) |
1701 |
->slice (",0:" . int ($len / 2)) |
1702 |
->PDL::Complex::Cr2p |
1703 |
->slice ("(0)"); |
1704 |
if ($norm == 1 || lc $norm eq "norm") { |
1705 |
$data / max $data; |
1706 |
} elsif (($norm =~ /^[.0]+$/) || (lc $norm eq "db")) { |
1707 |
log (1e-37 + $data / max $data) * (20 / log 10); |
1708 |
} else { |
1709 |
$data; |
1710 |
} |
1711 |
} |
1712 |
|
1713 |
=head2 concat pdl, pdl... |
1714 |
|
1715 |
This is not really an audio-related function. It simply takes all piddles |
1716 |
and concats them into a larger one. At the moment it only supports |
1717 |
single-dimensional piddles and is implemented quite slowly using perl and |
1718 |
data-copying, but that might change... |
1719 |
|
1720 |
=cut |
1721 |
|
1722 |
sub concat { |
1723 |
my $len = sum pdl map $_->getdim(0), @_; |
1724 |
my $r = zeroes $len; |
1725 |
my $o = 0; |
1726 |
for (@_) { |
1727 |
my $e = $o + $_->getdim(0) - 1; |
1728 |
(my $t = $r->slice("$o:$e")) .= $_; |
1729 |
$o = $e + 1; |
1730 |
} |
1731 |
$r; |
1732 |
} |
1733 |
|
1734 |
EOP |
1735 |
|
1736 |
# TODO: document! |
1737 |
pp_def 'filter_convolve', |
1738 |
Pars => 'input(n); kernel(m); int fftsize(); [o]output(n)', |
1739 |
GenericTypes => [D], |
1740 |
Code => q! |
1741 |
mus_convolve ($P(input), $P(output), $SIZE(n), $P(kernel), $fftsize(), $SIZE(m)); |
1742 |
! |
1743 |
; |
1744 |
|
1745 |
pp_def 'rshift', |
1746 |
Doc => <<'EOD', |
1747 |
=for ref |
1748 |
|
1749 |
Shift vector elements without wrap and fill the free space with a |
1750 |
constant. Flows data back & forth, for values that overlap. |
1751 |
|
1752 |
Positive values shift right, negative values shift left. |
1753 |
|
1754 |
EOD |
1755 |
Pars => 'x(n); int shift(); c(); [oca]y(n)', |
1756 |
DefaultFlow => 1, |
1757 |
Reversible => 1, |
1758 |
PMCode => ' |
1759 |
sub PDL::rshift { |
1760 |
my @a = @_; |
1761 |
if($#a == 3) { |
1762 |
&PDL::_rshift_int(@a);@a=(); |
1763 |
} elsif($#a == 1 || $#a == 2) { |
1764 |
$a[3] = PDL->nullcreate($a[0]); |
1765 |
&PDL::_rshift_int(@a); |
1766 |
$a[3]; |
1767 |
} else { |
1768 |
barf "Invalid number of arguments for shiftin"; |
1769 |
} |
1770 |
} |
1771 |
', |
1772 |
Code=>' |
1773 |
int i,j; |
1774 |
int n_size = $SIZE(n); |
1775 |
for(i = -$shift(), j=0; j < n_size; i++, j++) |
1776 |
$y(n=>j) = i >= 0 && i < n_size ? $x(n=>i) : $c(); |
1777 |
', |
1778 |
BackCode=>' |
1779 |
int i,j; |
1780 |
int n_size = $SIZE(n); |
1781 |
for(i = -$shift(), j=0; j < n_size; i++, j++) |
1782 |
if (i >= 0 && i < n_size) |
1783 |
$x(n=>i) = $y(n=>j); |
1784 |
' |
1785 |
; |
1786 |
|
1787 |
pp_def 'polynomial', |
1788 |
Pars => 'coeffs(n); x(m); [o]out(m)', |
1789 |
Doc => 'evaluate the polynomial with coefficients C<coeffs> at the position(s) C<x>. C<coeffs[0]> is the constant term.', |
1790 |
Code => q! |
1791 |
loop(m) %{ |
1792 |
$GENERIC() x = 1; |
1793 |
$GENERIC() o = 0; |
1794 |
|
1795 |
loop(n) %{ |
1796 |
o += $coeffs() * x; |
1797 |
x *= $x(); |
1798 |
%} |
1799 |
|
1800 |
$out() = o; |
1801 |
%} |
1802 |
! |
1803 |
; |
1804 |
|
1805 |
pp_def 'linear_interpolate', |
1806 |
Pars => 'x(); fx(n); fy(n); [o]y()', |
1807 |
GenericTypes => [L,F,D], |
1808 |
Doc => ' |
1809 |
Look up the ordinate C<x> in the function given by C<fx> and C<fy> |
1810 |
and return a linearly interpolated value (somewhat optimized for many |
1811 |
lookups). |
1812 |
|
1813 |
C<fx> specifies the ordinates (x-coordinates) of the function and most be |
1814 |
sorted in increasing order. C<fy> are the y-coordinates of the function in |
1815 |
these points. |
1816 |
', |
1817 |
Code => q! |
1818 |
int d = 0; |
1819 |
int D = $SIZE(n) - 1; /* a tribute to PP's stupidity */ |
1820 |
$GENERIC() xmin =$fx(n=>0); |
1821 |
$GENERIC() xmax =$fx(n=>D); |
1822 |
|
1823 |
threadloop %{ |
1824 |
$GENERIC() x = $x(); |
1825 |
$GENERIC() x1, x2, y1, y2; |
1826 |
|
1827 |
if (x <= xmin) |
1828 |
$y() = $fy(n=>0); |
1829 |
else if (x >= xmax) |
1830 |
$y() = $fy(n=>D); |
1831 |
else |
1832 |
{ |
1833 |
while ($fx(n=>d) > x) d--; |
1834 |
while ($fx(n=>d) <= x) d++; |
1835 |
/* 0 < d <= D */ |
1836 |
/* fx(d-1) < x <= fx(d) */ |
1837 |
x2 = $fx(n=>d); |
1838 |
y2 = $fy(n=>d); |
1839 |
d--; |
1840 |
x1 = $fx(n=>d); |
1841 |
y1 = $fy(n=>d); |
1842 |
$y() = y1 + (x-x1)*(y2-y1)/(x2-x1); |
1843 |
} |
1844 |
%} |
1845 |
! |
1846 |
; |
1847 |
|
1848 |
pp_def 'bessi0', |
1849 |
Pars => 'a(); [o]b()', |
1850 |
Doc => 'calculate the (approximate) modified bessel function of the first kind', |
1851 |
GenericTypes => [F,D], |
1852 |
Code => q! |
1853 |
$GENERIC() x = $a(); |
1854 |
if (x > -3.75 && x < 3.75) |
1855 |
{ |
1856 |
$GENERIC() y = (x / 3.75); y *= y; |
1857 |
$b() = 1 + (y * (3.5156229 + |
1858 |
(y * (3.0899424 + |
1859 |
(y * (1.2067492 + |
1860 |
(y * (0.2659732 + |
1861 |
(y * (0.360768e-1 + |
1862 |
(y * 0.45813e-2))))))))))); |
1863 |
} |
1864 |
else |
1865 |
{ |
1866 |
$GENERIC() ax = x < 0 ? -x : x; |
1867 |
$GENERIC() y = ax / 3.75; |
1868 |
$b() = ((exp(ax) / sqrt(ax)) * |
1869 |
(0.39894228 + |
1870 |
(y * (0.1328592e-1 + |
1871 |
(y * (0.225319e-2 + |
1872 |
(y * (-0.157565e-2 + |
1873 |
(y * (0.916281e-2 + |
1874 |
(y * (-0.2057706e-1 + |
1875 |
(y * (0.2635537e-1 + |
1876 |
(y * (-0.1647633e-1 + |
1877 |
(y * 0.392377e-2))))))))))))))))); |
1878 |
} |
1879 |
!, |
1880 |
; |
1881 |
|
1882 |
pp_def 'fast_sin', |
1883 |
Pars => 'r(n); [o]s(n)', |
1884 |
GenericTypes => [F,D], |
1885 |
Doc => 'fast sine function (inaccurate table lookup with ~12 bits precision)', |
1886 |
Code => q^ |
1887 |
# define SINE_SIZE 16384 |
1888 |
static float *sine_table = 0; |
1889 |
|
1890 |
if (!sine_table) |
1891 |
{ |
1892 |
int i; |
1893 |
Float phase, incr; |
1894 |
sine_table = (float *) calloc (SINE_SIZE + 1, sizeof (float)); |
1895 |
incr = M_2PI / (Float) SINE_SIZE; |
1896 |
for (i = 0, phase = 0.0; i < SINE_SIZE + 1; i++, phase += incr) |
1897 |
sine_table[i] = (float) sin (phase); |
1898 |
} |
1899 |
|
1900 |
threadloop %{ |
1901 |
loop(n) %{ |
1902 |
$s() = sine_table[((int)($r() * (SINE_SIZE / M_2PI)) % SINE_SIZE + SINE_SIZE) % SINE_SIZE]; |
1903 |
%} |
1904 |
%}^ |
1905 |
; |
1906 |
|
1907 |
pp_addpm {At => Bot}, <<'EOD'; |
1908 |
|
1909 |
=head1 AUTHOR |
1910 |
|
1911 |
Marc Lehmann <schmorp@schmorp.de>. The ideas were mostly taken from common |
1912 |
lisp music (CLM), by Bill Schottstaedt C<bil@ccrma.stanford.edu>. I also |
1913 |
borrowed many explanations (and references) from the clm docs and some |
1914 |
code from clm.c. Highly inspiring! |
1915 |
|
1916 |
=head1 SEE ALSO |
1917 |
|
1918 |
perl(1), L<PDL>. |
1919 |
|
1920 |
=cut |
1921 |
EOD |
1922 |
|
1923 |
pp_addxs "\nINCLUDE: xlib.xs\n"; |
1924 |
|
1925 |
pp_done; |
1926 |
|