Scheduling Examples Using Serpent Threads
Computer Music Systems and Information Processing

Roger B. Dannenberg
Jan 2018

Introduction

This is a companion document to From Theory to Practice. The examples on that page are recreated here using Serpent threads.

Threads in Serpent are non-preemptive, which means you normally do not have to worry about synchronization or race conditions where two threads are concurrently reading and writing to shared memory with unpredictable results. In contrast, threads in Serpent run until you explicitly call yield() or yieldto() or perhaps call a scheduling function that indirectly yields control to another thread.

The scheduler library (sched) has been extended to support threads, allowing threads to pause within a logical time system. Thus, threads support the same kind of precise timing you can obtain by scheduling function and method calls directly with the scheduler.

Scheduling Threads

Simple Example

(Code available here.) Let's make a simple program to print a message every second.

require "debug"
require "sched"

def activity():
    if fork(): return // main thread returns
    sched_wait(1)     // initial delay is 1 s
    var i = 101
    while true: // loop forever
        display "activity", i, sched_rtime, time_get()
        i = i + 1
        sched_wait(1)
    // this thread never ends

sched_init()
activity()
sched_run()
Notes:
Here's what some output looks like:
activity: i = 101, sched_rtime = 1, time_get() = 1
activity: i = 102, sched_rtime = 2, time_get() = 2.002
activity: i = 103, sched_rtime = 3, time_get() = 3.001
activity: i = 104, sched_rtime = 4, time_get() = 4.001
activity: i = 105, sched_rtime = 5, time_get() = 5
activity: i = 106, sched_rtime = 6, time_get() = 6
activity: i = 107, sched_rtime = 7, time_get() = 7.001
activity: i = 108, sched_rtime = 8, time_get() = 8
activity: i = 109, sched_rtime = 9, time_get() = 9.001
activity: i = 110, sched_rtime = 10, time_get() = 10.001
^C
Notice that the ideal event times, accessed as sched_rtime are precisely 1 second apart, as if the computer is infinitely fast and time is not quantized. The actual time, accessed as time_get() is close to the ideal time, but sometimes a millisecond or two late.

Two Activities

(Code available here.) Scheduling is most useful when there are many threads whose events are interleaved in time. This example is an alternate implementation of the temporal recursion example >Two Activities. The code here is very much like the previous example (above) since it mainly just adds another thread.

As with >Two Activities, the code runs two activities. activity_1 runs every second; activity_2 runs every 3 seconds.

require "debug"
require "sched"


def activity_1():
    if fork(): return
    var i = 101
    while true:
        display "act1    ", i, sched_rtime, time_get()
        i = i + 1
        sched_wait(1)


def activity_2():
    if fork(): return
    var i = 201
    while true:
        display "    act2", i, sched_rtime, time_get()
        i = i + 1
        sched_wait(3)

sched_init()
activity_1()
activity_2()
sched_run()
Again, here is some output generated by the program. Notice that when 2 events are scheduled for the same time, they run sequentially, but it is not obvious which will run first. (In this case, ordering is deterministic, but order depends on the implementation of the priority queue in the scheduler.)

Notice also that activity 1 ("act1") runs every second, while activity 2 ("act2") runs every 3 seconds.

    act2: i = 201, the_sched.time = 1, time_get() = 1.001
act1    : i = 101, sched_rtime = 1, time_get() = 1.001
act1    : i = 102, sched_rtime = 2, time_get() = 2
act1    : i = 103, sched_rtime = 3, time_get() = 3.001
act1    : i = 104, sched_rtime = 4, time_get() = 4.001
    act2: i = 202, sched_rtime = 4, time_get() = 4.002
act1    : i = 105, sched_rtime = 5, time_get() = 5
act1    : i = 106, sched_rtime = 6, time_get() = 6
act1    : i = 107, sched_rtime = 7, time_get() = 7
    act2: i = 203, sched_rtime = 7, time_get() = 7
act1    : i = 108, sched_rtime = 8, time_get() = 8.001
act1    : i = 109, sched_rtime = 9, time_get() = 9
act1    : i = 110, sched_rtime = 10, time_get() = 10
    act2: i = 204, sched_rtime = 10, time_get() = 10
act1    : i = 111, sched_rtime = 11, time_get() = 11.001
act1    : i = 112, sched_rtime = 12, time_get() = 12.002
^C

Multiple "Threads" Using One Function

(Code available here.) Let's go back to the first example and initialize the schedule with two calls to activity. We will pass in the initial delay and initial values of i so they can be different, thus duplicating the behavior of the Multiple Threads Using One Function example.
require "debug"
require "sched"


def activity(i):
    if fork(): return // main thread returns
    sched_wait(1)     // initial delay is 1 s
    while true: // loop forever
        display "activity", i, sched_rtime, time_get()
        i = i + 1
        sched_wait(1)
    // this thread never ends

sched_init()
activity(101)
activity(201)
sched_run()
Here is some output generated by the program. Notice that there are two independent threads, even though there in only one function. The events are separated by 1 second, with one set of events starting at 1 and the other starting at 1.1.
activity: i = 101, sched_rtime = 1, time_get() = 1.002
activity: i = 201, sched_rtime = 1.1, time_get() = 1.101
activity: i = 102, sched_rtime = 2, time_get() = 2.001
activity: i = 202, sched_rtime = 2.1, time_get() = 2.102
activity: i = 103, sched_rtime = 3, time_get() = 3
activity: i = 203, sched_rtime = 3.1, time_get() = 3.1
activity: i = 104, sched_rtime = 4, time_get() = 4
activity: i = 204, sched_rtime = 4.1, time_get() = 4.1
activity: i = 105, sched_rtime = 5, time_get() = 5.001
activity: i = 205, sched_rtime = 5.1, time_get() = 5.101
^C

The Stopping a “Thread” Pattern

(Code available here.) How can we stop a thread that runs a loop as in previous examples? The simplest way is to set a flag that causes the loop to exit immediately. As described in The Stopping a “Thread” Pattern, not a good solution.

A solution with threads that parallels the temporal recursion example follows.

require "debug"
require "sched"


activity_id = 0 // used to kill "threads"


def activity_start(delay, i):
    if fork(): return
    sched_wait(delay)
    activity_id = activity_id + 1 // kill any old running activity
    var id = activity_id
    while id == activity_id:
        display "activity", id, i, sched_rtime, time_get()
        i = i + 1
        sched_wait(1)
    display "activity terminates", id, sched_rtime, time_get()


sched_init()
activity_start(1, 101)
activity_start(4.5, 201)
sched_run()
Notice how this differs from the temporal recursion solution. Here is some output generated by the program. Notice that the first thread is effectivly killed at time 4.5 when activity_id is incremented, but the thread does not "know" this until it wakes up at time 5, "sees" activity_id has changed, and terminates (returns).
activity: id = 1, i = 101, sched_rtime = 1, time_get() = 1.001
activity: id = 1, i = 102, sched_rtime = 2, time_get() = 2.002 activity: id = 1, i = 103, sched_rtime = 3, time_get() = 3.001 activity: id = 1, i = 104, sched_rtime = 4, time_get() = 4.001 activity: id = 2, i = 201, sched_rtime = 4.5, time_get() = 4.5 activity terminates: id = 1, sched_rtime = 5, time_get() = 5.001 activity: id = 2, i = 202, sched_rtime = 5.5, time_get() = 5.501 activity: id = 2, i = 203, sched_rtime = 6.5, time_get() = 6.501 activity: id = 2, i = 204, sched_rtime = 7.5, time_get() = 7.501 ^C

Spawning New Activity

(Code available here.) A thread can be forked at any time. Here's a thread that runs every second. Each second, it forks another thread that runs every 0.1 seconds, but only 3 times. This has the same behavior as the Spawning New Activity example.

In this implementation, the “sub-thread” is syntactically nested within the main thread, so each iteration of the outer loop forks a copy of the inner loop. These are independent threads, so there is no constraint in this case that the inner thread completes before the next iteration of the outer loop.

It is important to understand when the inner threads complete. Find the line return // IMPORTANT!. This returns from the inner thread or “sub-thread”. Without this, the inner thread would continue executing the outer loop. Even though the fork() is inside the loop, it still duplicates the entire state of execution of the function.

require "debug"
require "sched"

def activity(i):
    if fork(): return
    sched_wait(1) // initial delay
    while true:
        display "activity", i, sched_rtime, time_get()
        i = i + 1
        if fork() == nil:
            for j = 0 to 3
                sched_wait(0.1)
                display "     sub", j, sched_rtime, time_get()
            return // IMPORTANT!
        sched_wait(1)

sched_init()
activity(101)
sched_run()
Here is some output generated by the program.
activity: i = 101, sched_rtime = 1, time_get() = 1.001
     sub: j = 0, sched_rtime = 1.1, time_get() = 1.102
     sub: j = 1, sched_rtime = 1.2, time_get() = 1.202
     sub: j = 2, sched_rtime = 1.3, time_get() = 1.3
activity: i = 102, sched_rtime = 2, time_get() = 2.002
     sub: j = 0, sched_rtime = 2.1, time_get() = 2.101
     sub: j = 1, sched_rtime = 2.2, time_get() = 2.2
     sub: j = 2, sched_rtime = 2.3, time_get() = 2.3
activity: i = 103, sched_rtime = 3, time_get() = 3.001
     sub: j = 0, sched_rtime = 3.1, time_get() = 3.101
     sub: j = 1, sched_rtime = 3.2, time_get() = 3.201
     sub: j = 2, sched_rtime = 3.3, time_get() = 3.302
^C

Virtual Time, Fixed Tempo

(Code available here.) This example uses threads to implement the temporal recursion example Virtual Time, Fixed Tempo. The thread-based solution is very similar, and the chief thing to note is that sched_wait() in activity() uses vtsched because it is selected when sched_wait() is called. If another scheduler is selected while sched_wait() is waiting, it does not matter, because the scheduler will be saved and restored by sched_wait(). However, sched_select just sets a global variable, so you should be careful. For example, if you call a function that calls sched_select in order to use it for a new thread, keep in mind that the current scheduler may change when you call that function.
require "debug"
require "sched"


def activity(i):
    if fork(): return
    sched_wait(1)
    while true
        display "activity", i, sched_rtime, time_get(), sched_vtime
        i = i + 1
        sched_wait(1)


sched_init()
sched_select(vtsched)
sched_set_bps(2) // 2 beats per second
sched_run()

Here's the output. Notice the_sched (which is vtsched) time is not even close to real time because this is virtual time (or beats, if you prefer).

activity: i = 101, sched_rtime = 0.5, time_get() = 0.5, sched_vtime = 1
activity: i = 102, sched_rtime = 1, time_get() = 1.002, sched_vtime = 2
activity: i = 103, sched_rtime = 1.5, time_get() = 1.502, sched_vtime = 3
activity: i = 104, sched_rtime = 2, time_get() = 2.001, sched_vtime = 4
activity: i = 105, sched_rtime = 2.5, time_get() = 2.502, sched_vtime = 5
^C

Virtual Time, Variable Tempo

(Code available here.) The speed or tempo can be changed. Here, we reimplement the Virtual Time, Variable Tempo example using threads. This example changes tempo to 0.5 beats per second at beat 4.

Notice how we wait until beat 4 to change tempo. Top-level commands in Serpent are compiled as the body of a parameter-less function, which is then called, so the semantics here are to fork the implicit parameter-less function; hence, sched_wait works. You might find it more direct

require "debug"
require "sched"

def activity(i):
    if fork(): return
    sched_wait(1)
    while true:
        display "activity", i, sched_rtime, time_get(), sched_vtime
        i = i + 1
        sched_wait(1)

sched_init()
sched_select(vtsched)
sched_set_bps(2) // 2 beats per second
// equivalent to: sched_set_period(0.5)
activity(101)
if fork() == nil:
    sched_wait(4)
    sched_set_bps(0.5)
sched_run()

Here's the output. Notice the real times (time_get()) are every 0.5s until beat 4, then every 2s.

activity: i = 101, sched_rtime = 1, time_get() = 0.5
activity: i = 101, sched_rtime = 0.5, time_get() = 0.5, sched_vtime = 1
activity: i = 102, sched_rtime = 1, time_get() = 1.002, sched_vtime = 2
activity: i = 103, sched_rtime = 1.5, time_get() = 1.501, sched_vtime = 3
activity: i = 104, sched_rtime = 2, time_get() = 2, sched_vtime = 4
activity: i = 105, sched_rtime = 4, time_get() = 4.002, sched_vtime = 5
activity: i = 106, sched_rtime = 6, time_get() = 6.002, sched_vtime = 6
^C

Conclusions

This should be enough to get you started with threads. The remaining examples in Scheduling in Practice can of course be implemented using threads.

Threads are most useful when combined with scheduling primitives such as sched_wait() which gives precisely timed thread behavior and enables many threads to run concurrently.

Keep in mind that threads do not run in parallel and if a thread goes into an long computation, it may starve other threads or cause timing problems. sched_wait() is especially useful here because it allows you to tell threads when to run and when to stop.