Notifications and shared memory

This tutorial covers notification objects.

You will learn how to:

  1. Set up shared memory between tasks.
  2. Use notification objects for synchronisation between tasks.
  3. Use badges to differentiate notifications.

Prerequisites

  1. Set up your machine
  2. Capabilities tutorial
  3. Mapping tutorial
  4. Threads tutorial

Initialising

# For instructions about obtaining the tutorial sources see https://docs.sel4.systems/Tutorials/get-the-tutorials
#
# Follow these instructions to initialise the tutorial
# initialising the build directory with a tutorial exercise
./init --tut notifications
# building the tutorial exercise
cd notifications_build
ninja
Hint: tutorial solutions


All tutorials come with complete solutions. To get solutions run:

# For instructions about obtaining the tutorial sources see https://docs.sel4.systems/Tutorials/get-the-tutorials
#
# Follow these instructions to initialise the tutorial
# initialising the build directory with a tutorial exercise
./init --solution --tut notifications
# building the tutorial exercise
cd notifications_build
ninja

Answers are also available in drop down menus under each section.

CapDL Loader

This tutorial uses the capDL loader, a root task which allocates statically configured objects and capabilities.

Get CapDL

The capDL loader parses a static description of the system and the relevant ELF binaries. It is primarily used in Camkes projects but we also use it in the tutorials to reduce redundant code. The program that you construct will end up with its own CSpace and VSpace, which are separate from the root task, meaning CSlots like seL4_CapInitThreadVSpace have no meaning in applications loaded by the capDL loader.

More information about CapDL projects can be found here.

For this tutorial clone the CapDL repo. This can be added in a directory that is adjacent to the tutorials-manifest directory.

Background

Notifications allow processes to send asynchronous signals to each other, and are primarily used for interrupt handling and to synchronise access to shared data buffers.

Notification objects

Signals are sent and received with invocations on capabilities to notification objects. A notification object consists of a data word, which acts as an array of binary semaphores, and a queue of TCBs waiting for notifications.

Notification objects can be in three states:

  • Waiting - there are TCBs queued on this notification waiting for it to be signalled.
  • Active - TCBs have signalled data on this notification,
  • Idle - no TCBs are queued and no TCBs have signalled this object since it was last set to idle.

Signalling

When a task signals a notification object (using seL4_Signal), what occurs depends on the state of the object:

  • Idle - the data word is set to the badge of the capability used to send the signal, and the object is converted to active.
  • Active - the badge of the capability used to signal the notification object is bitwise-orred with the notifications data word.
  • Waiting - the head of the queue of TCBs is woken and the badge sent to that TCB. If the queue is empty, the notification object is transitioned to idle.

In this way notification objects can be seen as a binary array of semaphores - if the signallers all use a different bit in the badge, they can set different badge bits and waiters can observe which bits have been set.

Waiting

Tasks can wait on a notification object using seL4_Wait, which does the following:

  • Idle - the TCB is queued, and the notification transitioned to waiting.
  • Active - the TCB receives the data word, the data word is reset to 0 and the notification transitioned to idle,
  • Waiting - the TCB is appended to the queue.

Polling

Notification objects can also be polled with seL4_Poll, which is a non-blocking version of seL4_Wait that returns immediately regardless of the state.

Interrupts and IPC

Notification objects can be used to receive signals of interrupt delivery, and can also be bound to TCBs such that signals and IPC can be received by the same thread. This is explained in more detail in the timer tutorial.

Exercises

These exercises guide you through a basic producer consumer set up using notifications and shared memory. The tutorial uses the capDL loader, and already has 2 producer processes (producer_1.c and producer_2) and 1 consumer process running (consumer.c). Each has access to a number of capabilities.

Each producer shares a buffer with the consumer, and the consumer processes data from both producers when it is available.

When you start the tutorial, the output will look something like this:

Booting all finished, dropped to user space
Waiting for producer

Set up shared memory

Both producers start and block immediately, waiting for the consumer to send an IPC with the address of the shared mapping. We provide code below that sets up the shared page between producer 1 and the consumer:

    /* set up shared memory for consumer 1 */
    /* first duplicate the cap */
    error = seL4_CNode_Copy(cnode, mapping_1, seL4_WordBits,
                          cnode, buf1_frame_cap, seL4_WordBits, seL4_AllRights);
    ZF_LOGF_IFERR(error, "Failed to copy cap");
    /* now do the mapping */
    error = seL4_ARCH_Page_Map(mapping_1, producer_1_vspace, BUF_VADDR,
                               seL4_AllRights, seL4_ARCH_Default_VMAttributes);
    ZF_LOGF_IFERR(error, "Failed to map frame");

However, we do not map the second buffer in, so producer 2 crashes immediately.

Exercise Understand the above code, and create a second shared page between producer_2 and consumer.

    // TODO share buf2_frame_cap with producer_2

Quick solution
    /* set up shared memory for producer 2 */
    error = seL4_CNode_Copy(cnode, mapping_2, seL4_WordBits,
                            cnode, buf2_frame_cap, seL4_WordBits, seL4_AllRights);
    ZF_LOGF_IFERR(error, "Failed to copy cap");
   /* now do the mapping */
    error = seL4_ARCH_Page_Map(mapping_2, producer_2_vspace, BUF_VADDR,
                               seL4_AllRights, seL4_ARCH_Default_VMAttributes);
    ZF_LOGF_IFERR(error, "Failed to map frame");

Whether this is successful will be visible after the next exercise when the consumers access their buffers. If the shared page setup for producer 2 is not correct, it will fail with a vm fault.

Signal the producers to go

At this point, both producers are waiting on the empty notification for a signal that the buffer is ready to be written to.

Exercise signal both producers via the buf1_empty and buf2_empty notification objects.

    // TODO signal both producers
Quick solution
    seL4_Signal(buf1_empty);
    seL4_Signal(buf2_empty);

Differentiate signals

Now you should see something like the following:

Booting all finished, dropped to user space
Waiting for producer
2: produce
1: produce
Got badge: 2
Got badge: 1

At this point, the consumer should consume data from the appropriate buffer(s) and signal to the appropriate consumer(s) that the buffer is empty again. The capability to the full notification object has already been badged: producer_1s copy has a badge of 0b1 and producer_2 a badge of 0b10. By checking the bits in the badge, you can see which of the producers (it may be both) has produced data.

Exercise Check the badge and signal the empty notification for the producers according to the bits set in the badge value.

    // TODO, use the badge to check which producer has signalled you, and signal it back. Note that you
    // may recieve more than 1 signal at a time.

Quick solution
        if (badge & 0b01) {
            assert(*buf1 == 1);
            *buf1 = 0;
            seL4_Signal(buf1_empty);
        }
        if (badge & 0b10) {
            assert(*buf2 == 2);
            *buf2 = 0;
            seL4_Signal(buf2_empty);
        }

At this point, you should see signals from both producers being processed, and the final Success! message printed.

Further exercises

That’s all for the detailed content of this tutorial. Below we list other ideas for exercises you can try, to become more familiar with IPC.

  • Create a counting semaphore implementation using notification objects.
  • Create a bounded-buffer producer consumer with a buffer size greater than 1.

Next: Interrupts


Tutorial included from github repo edit