Threads
So youâve made your killer bassline and a phat beat. How do you play them at the same time? One solution is to weave them together manually - play some bass, then a bit of drums, then more bass⊠However, the timing soon gets hard to think about, especially when you start weaving in more elements.
What if Sonic Pi could weave things for you automatically? Well, it can, and you do it with a special thing called a thread.
Infinite Loops
To keep this example simple, youâll have to imagine that this is a phat beat and a killer bassline:
loop do
sample :drum_heavy_kick
sleep 1
end
loop do
use_synth :fm
play 40, release: 0.2
sleep 0.5
end
As weâve discussed previously, loops are like black holes for the program. Once you enter a loop you can never exit from it until you hit stop. How do we play both loops at the same time? We have to tell Sonic Pi that we want to start something at the same time as the rest of the code. This is where threads come to the rescue.
Threads to the Rescue
in_thread do
loop do
sample :drum_heavy_kick
sleep 1
end
end
loop do
use_synth :fm
play 40, release: 0.2
sleep 0.5
end
By wrapping the first loop in an in_thread
do/end block we tell Sonic Pi to run the contents of the do/end block at exactly the same time as the next statement after the do/end block (which happens to be the second loop). Try it and youâll hear both the drums and the bassline weaved together!
Now, what if we wanted to add a synth on top. Something like:
in_thread do
loop do
sample :drum_heavy_kick
sleep 1
end
end
loop do
use_synth :fm
play 40, release: 0.2
sleep 0.5
end
loop do
use_synth :zawa
play 52, release: 2.5, phase: 2, amp: 0.5
sleep 2
end
Now we have the same problem as before. The first loop is played at the same time as the second loop due to the in_thread
. However, the third loop is never reached. We therefore need another thread:
in_thread do
loop do
sample :drum_heavy_kick
sleep 1
end
end
in_thread do
loop do
use_synth :fm
play 40, release: 0.2
sleep 0.5
end
end
loop do
use_synth :zawa
play 52, release: 2.5, phase: 2, amp: 0.5
sleep 2
end
Runs as threads
What may surprise you is that when you press the Run button, youâre actually creating a new thread for the code to run. This is why pressing it multiple times will layer sounds over each other. As the runs themselves are threads, they will automatically weave the sounds together for you.
Scope
As you learn how to master Sonic Pi, youâll learn that threads are the most important building blocks for your music. One of the important jobs they have is to isolate the notion of current settings from other threads. What does this mean? Well, when you switch synths using use_synth
youâre actually just switching the synth in the current thread - no other thread will have their synth switched. Letâs see this in action:
play 50
sleep 1
in_thread do
use_synth :tb303
play 50
end
sleep 1
play 50
Notice how the middle sound was different to the others? The use_synth
statement only affected the thread it was in and not the outer main run thread.
Inheritance
When you create a new thread with in_thread
, the new thread will automatically inherit all of the current settings from the current thread. Letâs see that:
use_synth :tb303
play 50
sleep 1
in_thread do
play 55
end
Notice how the second note is played with the :tb303
synth even though it was played from a separate thread? Any of the settings modified with the various use_*
functions will behave in the same way.
When threads are created, they inherit all the settings from their parent but they donât share any changes back.
Naming Threads
Finally, we can give our threads names:
in_thread(name: :bass) do
loop do
use_synth :prophet
play chord(:e2, :m7).choose, release: 0.6
sleep 0.5
end
end
in_thread(name: :drums) do
loop do
sample :elec_snare
sleep 1
end
end
Look at the log pane when you run this code. See how the log reports the name of the thread with the message?
[Run 36, Time 4.0, Thread :bass]
|- synth :prophet, {release: 0.6, note: 47}
Only One Thread per Name Allowed
One last thing to know about named threads is that only one thread of a given name may be running at the same time. Letâs explore this. Consider the following code:
in_thread do
loop do
sample :loop_amen
sleep sample_duration :loop_amen
end
end
Go ahead and paste that into a buffer and press the Run button. Press it again a couple of times. Listen to the cacophony of multiple amen breaks looping out of time with each other. Ok, you can press Stop now.
This is the behaviour weâve seen again and again - if you press the Run button, sound layers on top of any existing sound. Therefore if you have a loop and press the Run button three times, youâll have three layers of loops playing simultaneously.
However, with named threads it is different:
in_thread(name: :amen) do
loop do
sample :loop_amen
sleep sample_duration :loop_amen
end
end
Try pressing the Run button multiple times with this code. Youâll only ever hear one amen break loop. Youâll also see this in the log:
==> Skipping thread creation: thread with name :amen already exists.
Sonic Pi is telling you that a thread with the name :amen
is already playing, so itâs not creating another.
This behaviour may not seem immediately useful to you now - but it will be very handy when we start to live codeâŠ