Cornell University ECE4760
ProtoThreads
-- Three Simple examples
Pi Pico rp2040/rp2350

Protothreads 1.3 on RP2040
For a complete explanation of Protothreads capibilities and limitations look here.
You should read sections 1.3-1.6 of the reference manual to see all of the implementation details.
Below the description of the examples there is a table with all available protothreads functions, and below that a few debugging hints.

The three examples
In this section we are going to consider how to use protothreads by considering with three simple examples. The examples will show:

  1. The first example program just starts one thread that blinks the on-board LED.
    Main initializes the serial interface, identifies one function to the scheduler as a thread, and starts the scheduler in round-robin mode.
    Round-robin mode executes each thread in order.
    The one thread (protothread_toggle25):
    1. starts with a macro to identify an entry point: PT_BEGIN(pt);
    2. defines a static boolean variable for the LED state.
      All local variables in a thread must be static!
    3. Sets up gpio25 as an output controlled by the SIO and sets it to true.
    4. Since a scheduled thread function can never exit, the thread falls into a endless loop which
      toggles the LED state variable, copies it to gpio25, then yields the thread for
      100,000 uSec. A that point the scheduler gets control, notices that there are no other
      threads to run and stalls until the thread yield times out.
    5. the thread ends with a marker for the scheduler: PT_END(pt);

  2. The second example program starts two threads on one core. One to blink the LED at a duration set
    by a global variable and another to get user input for the duration variable.
    Main initializes the serial interface, identifies two functions to the scheduler as threads, and starts the scheduler in round-robin mode.
    The toggle thread is essentially identical to the first example, except that the blink time is a variable. User input
    from the serial UART channel needs careful handling because the serial is so slow compared to the cpu.  
    At 115200 baud, each sent character takes around 90 uSec to send, which is around 11000 cpu cycles we do not want to waste.
    Receiving text from a human typist is much slower, and therefore the receiver has to construct a string, while yielding between
    each typed character to any other thread which is ready to run.
    All of the yielding behavior in the serial thread is hidden from the programmer behind some macros and functions.

    The serial input thread (protothread_serial):
    1. starts with a macro to identify an entry point: PT_BEGIN(pt);
    2. Since a scheduled thread function can never exit, the thread falls into a endless loop which
      prompts for and receives the value of blink time.
    3. uses sprintf to construct a prompt string in a specific buffer, pt_serial_out_buffer.
    4. invokes serial_write to print the prompt, while blocking only this thread and allowing
      other threads to execute between characters.
    5. invokes serial_read to grab the user response, while blocking only this thread and allowing
      other threads to execute between characters.
    6. converts the input sring returned in pt_serial_in_buffer to an integer using sscanf and stores it in the global blink time variable.
    7. Note that there is no explicit yield in the serial thread because the the serial read/write functions handle the yields.
    8. the thread ends with a marker for the scheduler: PT_END(pt);

  3. The third example program starts two threads on core0 and one thread on core1.
    In this example, one of the threads on core0 turns the LED on for a certain time, while the thread on
    core1 turns off the same LED for a certain length of time. The second thread on core0 prompts the
    user for separate on-time and off-time to control the two other threads. There are two global
    variables to conrol the on/off times, and two semaphores to ensure precise alternation
    between the thread that turns on the LED and the thread that turns it off.

    There are now two Main functions, one for core0 and one for core1;
    1. Core1 main adds the protothread_toggle25_off function to the core1 scheduler,
      then starts the core 1 schduler.
    2. Core0 main:
      1. sleeps for a millseccond to let the programmer finish
      2. starts the serial i/o
      3. initializes two core-safe semaphores to control blink excution and
        sets one of them to one and the other to zero.
        Setting a semaphore to one allows immediate execution.
        In this case, to start the blinking.
      4. inits the gpio for the blink threads
      5. resets, then starts core1 and again waits a millisecond
      6. identifies the two core0 threads to the core0 scheduler
      7. starts the core0 scheduler.

    There are three threads. The first one turns the LED on from core0
    1. The protothread_toggle25_on thread runs on core0. It drops directly into
      an endless loop which immediately waits for the blink_on_go_s semaphore to become
      non-zero, which happens when the semaphore is signaled in the thread on core1.
    2. When the semaphore is non-zero two things happen: It is automatically set to zero
      and allows execution of rest of the program.
    3. The gpio pin is set to true
    4. the thread yields for the on-time
    5. the thread signals the other blink thread to execute using
      PT_SEM_SAFE_SIGNAL(pt, &blink_off_go_s) ;

    The second thead turns the LED off from core1.
    1. The protothread_toggle25_off thread runs on core1. It drops directly into
      an endless loop which immediately waits for the blink_off_go_s semaphore to become
      non-zero, which happens when the semaphore is signaled in the thread on core0.
    2. When the semaphore is non-zero two things happen: It is automatically set to zero
      and allows execution of rest of the program.
    3. The gpio pin is set to false
    4. the thread yields for the off-time
    5. the thread signals the other blink thread to execute using
      PT_SEM_SAFE_SIGNAL(pt, &blink_on_go_s) ;

    The third thread allows the user to set the blink on and blink off times.
    1. The protothread_serial thread drops directly into an endless loop
    2. Just as in example 2 above the thread prompts the user
    3. the user responds with two integers for two times in uSec.
    4. sscanf converts the input string into two integers. Since sscanf
      does not yield, you are assured that the two numbers will be stored
      before either of the other threads can see them.

The ZIP file contains all three example codes.

  1. Unzip the file
  2. Import the project using the Pico exentension in VScode
  3. Change the board type to either pico or pico2 (with no risc5)
  4. The cmakelists.txt file has two of the three example codes commented out.
    Uncomment the one you want to compile and comment the others.
  5. compile and run
  6. For examples 2 and 3, you will need to open the serial monitor.

Some more advanced examples
In this section we are going to look at some features of Protothreads which are very handy, but not as heavily used as in the previous examples.


Protothreads Function Summary
The following table give the syntax for the thread macros and functions written by Adam Dunkels.

Protothreads Functions (from Adam Dunkels) Description
PT_INIT(pt)

Initialize a protothread. This macro inits a protothread, but do not use it directly!. The pt_add() macro below will do this for you!
Note that you will proably never use this macro.

PT_THREAD(name_args)
Declaration of a protothread. name_args The name and arguments of the C function implementing the protothread.
PT_WAIT_THREAD(pt, thread)
Wait until a child protothread completes.
pt
A pointer to the protothread control structure.
thread
The child protothread with arguments
PT_WAIT_UNTIL(pt, condition)
PT_WAIT_WHILE(pt, condition)
Wait the thread and wait until/while condition is true. Conditions which may be set on either core are not safe. It is better to yield than wait.
PT_BEGIN(pt)
Declare the start of a protothread inside the C function implementing the protothread.
pt
A pointer to the protothread control structure.
PT_END(pt)
Declare the end of a protothread.
pt
A pointer to the protothread control structure.
PT_EXIT(pt)
This macro causes the protothread to exit. If the protothread was spawned by another protothread, the parent protothread will become unblocked and can continue to run.
pt A pointer to the protothread control structure.
PT_RESTART(pt)
This macro will block and cause the running protothread to restart its execution at the place of the PT_BEGIN()call. pt A pointer to the protothread control structure.
PT_SCHEDULE(f)
This macro shedules a protothread, but do not use it directly!. The scheduler will do this for you! The return value of the function is non-zero if the protothread is running or zero if the protothread has exited.
f The call to the C function implementing the protothread to be scheduled.
Note that you will proably never use this macro.
PT_SPAWN(pt, child, thread)
Spawn a child protothread and block the thread until it exits. Parameters:
pt A pointer to the current protothread control structure.
child A pointer to the child protothread’s control structure.
thread The child protothread with arguments
PT_YIELD(pt)
yield the protothread, thereby allowing other processing to take place in the system.
PT_YIELD_UNTIL(pt, cond)
Yield from the protothread until a condition occurs. Conditions which may be set on either core are not safe.
PT_SEM_INIT(s, c)
This macro initializes a semaphore with a value for the counter. Internally, the semaphores use an "unsigned int" to represent the counter, and therefore the "count" argument should be within range of an unsigned int. s A pointer to the pt_sem struct representing the semaphore. Only use these semaphores for signals between threads running on one core!
PT_SEM_SIGNAL(pt, s)
This macro carries out the "signal" operation on the semaphore. The signal operation increments the counter inside the semaphore, which eventually will cause waiting protothreads to continue executing. pt A pointer to the protothread (struct pt) in which the operation is executed. s A pointer to the pt_sem struct representing the semaphore. Only use these semaphores for signals between threads running on one core!
PT_SEM_WAIT(pt, s)
This macro carries out the "wait" operation on the semaphore. The wait operation causes the protothread to yield while the counter is zero. When the counter reaches a value larger than zero, the protothread will continue. Only use these semaphores for signals between threads running on one core!

The following table has the protothread macro extensions and functions
we wrote for the RP2040 and which are included in the Cornell header file.

Added Protothreads functions

Description

 

PT_YIELD_usec(delay_time) Causes the current thread to yield (stop executing) for the delay_time in microseconds. The time is derived from the core timer. Note that this is a non-blocking delay, not an interval timer. Overflows in about 300000 years for 64 bit timer read. (as of version 1.1.2.)
PT_INTERVAL_INIT()
PT_YIELD_INTERVAL(interval_time)

The yield interval macro attempts to make periodic execution of the thread. The interval_time is in microseconds. The time is derived from the core timer. Note that this is a non-blocking interval timer. Overflows in about 300000 years for 64 bit timer read (as of version 1.1.2.). The INIT macro must be called in the scheduled thread before the while(1) loop.
PT_GET_TIME_usec() Returns the current microsecond count on the system tick clock. This macro returns a uint_64 timer count. (as of version 1.1.2.)
PT_SEM_SAFE_INIT(s,c)
This macro initializes a semaphore with a value for the counter. Internally, the semaphores use an "unsigned int" to represent the counter, and therefore the "count" argument should be within range of an unsigned int. s is a pointer to the pt_sem struct representing the semaphore. Use these semaphores for signals between threads running on either core! This macro also grabs lock 25 for use by all defined semaphores.
PT_SEM_SAFE_WAIT(pt,s)
This macro carries out the "wait" operation on the semaphore. The wait operation causes the protothread to yield while the counter is zero. When the counter reaches a value larger than zero, the protothread will continue. pt A pointer to the protothread (struct pt) in which the operation is executed. s A pointer to the pt_sem struct representing the semaphore. Use these semaphores for signals between threads running on either core! Never wait on a semaphore in an ISR.
PT_SEM_SAFE_SIGNAL(pt,s)
This macro carries out the "signal" operation on the semaphore. The signal operation increments the counter inside the semaphore, which eventually will cause waiting protothreads to continue executing. pt A pointer to the protothread (struct pt) in which the operation is executed. s A pointer to the pt_sem struct representing the semaphore. Use these semaphores for signals between threads running on either core! These semaphores are core-safe, but should not be signaled in an ISR.
PT_LOCK_INIT(s,lock_num,lock_state)
Initializes a hardware spin_lock state to either LOCKED or UNLOCKED. s is a variable of type spin_lock_t *. lock_num must be between 26 and 31 and be unique. This macro also grabs lock 24 for use by all defined locks. These locks are global between cores.
PT_LOCK_WAIT(pt,lock_num)
Yields the thread until loc_num is UNLOCKED by another thread on either core, then LOCKs it. Never wait on a lock in an ISR. These locks are global between cores.
PT_LOCK_RELEASE(lock_num)
UNLOCKs lock_num
PT_FIFO_WRITE(data)
Sends one 32-bit word to the other core, and blocks until there is space in the FIFO.
PT_FIFO_READ(fifo_out)
Receives one 32-bit word from the other core, and blocks until there is data in the FIFO.
PT_FIFO_FLUSH
Drain outgoing FIFO for the current core.
Higher level control macros  
pt_add(function_name) ; Add a thread to the current core scheduler queue.
A thread is added on the core on which it runs.
See example code!
pt_schedule_start ;
Starts the scheduler for the current core. A separate scheduler runs on each core. This macro never exits!
serial_read ;
A thread which is spawned to get nonblocking string input from UART0. This function assumes that a human is typing at a terminal and thus supports backspace, termination by enter key and echos characters! String is returned in
pt_serial_in_buffer. If more than one thread can spawn this thread, then there must be semaphore protection. Control returns to the scheduler between every character received. The thread exits to the parent when it receives an <enter>.
serial_write ;
A thread which is spawned to send a string output from UART0. String to be sent in pt_serial_out_buffer.
If more than one thread can spawn this thread, then there must be semaphore protection. Control returns to the scheduler between every character sent. The thread exits to the parent after it sends the entire string (zero terminated).

Debugging ProtoThreads

If your program just sits there and does nothing:
-- Did you write each thread to include at least one YIELD (or YIELD_UNTIL, or YIELD_TIME_msec) in the while(1) loop? Remember that the UART input spawned
thread yields betwen eacy typed character. Humans are sooo slow.
-- Did you schedule the threads?
-- Does the board have correct Vdd?

If your program reboots about a few times/sec:
-- Did you exit from any scheduled thread? This blows out the stack.
-- Did you turn on any interrupt source and fail to write an ISR for that source?
-- Did you divide by zero? A divide by zero is an untrapped exception.
-- Did you write to a non-existant memory location, perhaps by using a negative (or very large) array index.
A write to a non-existant memory location is an untrapped exception.

If your thread state variables seem to change on their own:
--- Did you define any automatic local variable?
Local variables in a thread should be static.
-- Are your global arrays big enough the clobber the stack in high memory?
You should be able to use over 200 kbytes.


Copyright Cornell University March 15, 2025