Hello ChucK: // connect sine oscillator to D/A convertor (sound card) SinOsc s => dac;
The above does several things. (1) it creates a new unit generator of
type 'SinOsc' (sine oscillator), and store its reference in variable 's'.
(2) 'dac' (D/A convertor) is a special unit generator (created by the system)
which is our abstraction for the underlying audio interface. (3) we are
using the ChucK operator (=>) to ChucK 's' to 'dac'. In ChucK, when one
unit generator is ChucKed to another, we connect them. We can think of this
line as setting up a data flow from 's', a signal generator, to 'dac',
the sound card/speaker. Collectively, we will call this a 'patch'.
// connect sine oscillator to D/A convertor (sound card) SinOsc s => dac; // allow 2 seconds to pass 2::second => now;Let's now run this (assuming you saved the file as 'foo.ck'): %> chuck foo.ck This would cause the sound to play for 2 seconds, during which time audio data is processed (and heard), afterwhich the program exits (since it has reached the end). For now, we can just take the second line of code to mean "let time pass for 2 seconds (and let audio compute during that time)". If you want to play it indefinitely, we could write a loop: // connect sine oscillator to D/A convertor (sound card) SinOsc s => dac; // loop in time while( true ) { 2::second => now; }
In ChucK, this is called a 'time-loop' (in fact this is an 'infinite time
loop'). This program executes (and generate/process audio) indefinitely.
Try runnig this program.
// make our patch SinOsc s => dac; // time-loop, in which the osc's frequency is changed every 100 ms while( true ) { 100::ms => now; Std.rand2f(30.0, 1000.0) => s.freq; }
This should sound like computer mainframes in old sci-fi movies. Two
more things to note here. (1) We are advancing time inside the loop by
100::ms durations. (2) A random value between 30.0 and 1000.0 is generated and
'assigned' to the oscillator's frequency, every 100::ms.
%> chuck foo.ckPlay with the parameters in the program. change 100::ms to something else (like 50::ms or 500::ms, or 1::ms or perhaps 1::samp(every sample)), or change 1000.0 to 5000.0... Run and listen: %> chuck foo.ckOnce things work, hold on to this file - we will use it again soon. --- Concurrency in ChucK: Now let's write another (slightly longer) program: (these files can be found in the examples/ directory, so you don't have to type them in) // impulse to filter to dac Impulse i => BiQuad f => dac; // set the filter's pole radius .99 => f.prad; // set equal gain zero's 1 => f.eqzs; // initialize float variable 0.0 => float v; // infinite time-loop while( true ) { // set the current sample/impulse 1.0 => i.next; // sweep the filter resonant frequency Std.fabs(Math.sin(v)) * 4000.0 => f.pfreq; // increment v v + .1 => v; // advance time 100::ms => now; }Name this moe.ck, and run it: %> chuck moe.ck
Now, make two copies of moe.ck - larry.ck and curly.ck. Make the
following modifications. 1) change larry.ck to advance time by 99::ms
(instead of 100::ms). 2) change curly.ck to advance time by 101::ms
(instead of 100::ms). 3) optionally, change the 4000.0 to something else
(like 400.0 for curly).
%> chuck moe.ck larry.ck curly.ck
What you hear (if all goes well) should be 'phasing' between moe, larry,
and curly, with curly emitting the lower-frequency pulses.
For more detail on each particular aspect of the ChucK language and virtual machine environment, please see the language specification and additional resources for learning ChucK. A large collection of pre-made examples have been arranged and provided with this distribution in the /doc/examples directory, and are mirrored here |