## RTX Project Report

Xiang, Dian 20431601 dxiang@uwaterloo.ca

Adrian Cheung
20421743
a32cheun@uwaterloo.ca

 $\begin{array}{c} {\rm Justin~McGirr} \\ 20413625 \\ {\rm jmcgirr@uwaterloo.ca} \end{array}$ 

Aaron Morais
20413440
aemorais@uwaterloo.ca

April 5, 2014



## Contents

| Ι  | Introduction |        |                                       |    |  |  |  |
|----|--------------|--------|---------------------------------------|----|--|--|--|
| II | A            | weso   | me RTX                                | 7  |  |  |  |
| 1  | Glo          | bal Va | riables                               | 8  |  |  |  |
|    | 1.1          | Descri | ption                                 | 8  |  |  |  |
|    | 1.2          | Heap 1 | Data Structures                       | 8  |  |  |  |
|    | 1.3          | Proces | ss Scheduler Data Structures          | 9  |  |  |  |
|    |              | 1.3.1  | Ready and Blocked Priority Queues     | 10 |  |  |  |
|    |              | 1.3.2  | Priorities                            | 11 |  |  |  |
|    |              | 1.3.3  | Message Passing                       | 12 |  |  |  |
| 2  | Ker          | nel AI | PI 1                                  | .3 |  |  |  |
|    | 2.1          | Descri | ption                                 | 13 |  |  |  |
|    | 2.2          |        | _                                     | 13 |  |  |  |
|    |              | 2.2.1  |                                       | 13 |  |  |  |
|    |              | 2.2.2  |                                       | 14 |  |  |  |
|    |              | 2.2.3  |                                       | 15 |  |  |  |
|    | 2.3          | Proces |                                       | 15 |  |  |  |
|    | 2.4          |        |                                       | 16 |  |  |  |
|    |              | 2.4.1  |                                       | 16 |  |  |  |
|    |              | 2.4.2  |                                       | 17 |  |  |  |
|    |              | 2.4.3  | · · · · · · · · · · · · · · · · · · · | 18 |  |  |  |
|    | 2.5          | Proces |                                       | 18 |  |  |  |
|    |              | 2.5.1  | · · · · · · · · · · · · · · · · · · · | 19 |  |  |  |
|    |              | 2.5.2  | · · · · · · · · · · · · · · · · · · · | 19 |  |  |  |

| 3  | Inte                      | errupts and Their Handler and Processes | <b>20</b>    |  |  |  |  |  |  |
|----|---------------------------|-----------------------------------------|--------------|--|--|--|--|--|--|
|    | 3.1                       | Description                             | 20           |  |  |  |  |  |  |
|    | 3.2                       | UART                                    | 20           |  |  |  |  |  |  |
|    | 3.3                       | TIMER                                   | 20           |  |  |  |  |  |  |
| 4  | System and User Processes |                                         |              |  |  |  |  |  |  |
|    | 4.1                       | System Processes                        | 21           |  |  |  |  |  |  |
|    |                           | 4.1.1 Description                       | 21           |  |  |  |  |  |  |
|    |                           | 4.1.2 Null Process                      | 21           |  |  |  |  |  |  |
|    |                           | 4.1.3 CRT Display Process               | 22           |  |  |  |  |  |  |
|    |                           | 4.1.4 Keyboard Command Decoder Process  | 22           |  |  |  |  |  |  |
|    | 4.2                       | User Processes                          | 24           |  |  |  |  |  |  |
|    |                           | 4.2.1 Wall Clock Process                | 24           |  |  |  |  |  |  |
|    |                           | 4.2.2 'funProcess'                      | 24           |  |  |  |  |  |  |
|    |                           | 4.2.3 'schizophrenicProcess'            | 24           |  |  |  |  |  |  |
|    |                           | 4.2.4 'fibProcess'                      | 24           |  |  |  |  |  |  |
|    |                           | 4.2.5 'memoryMuncherProcess'            | 24           |  |  |  |  |  |  |
|    |                           | 4.2.6 'releaseProcess'                  | 24           |  |  |  |  |  |  |
| 5  | Initialization            |                                         |              |  |  |  |  |  |  |
|    | 5.1                       |                                         |              |  |  |  |  |  |  |
|    | 5.2                       | Memory Layout                           | 25<br>25     |  |  |  |  |  |  |
|    | 5.3                       | Process Mailbox                         | 25           |  |  |  |  |  |  |
| 6  | Major Design Changes      |                                         |              |  |  |  |  |  |  |
| Ŭ  | 6.1                       | Description                             | <b>26</b> 26 |  |  |  |  |  |  |
|    | 6.2                       | Heap                                    | 26           |  |  |  |  |  |  |
|    | 6.3                       | Scheduler                               | 26           |  |  |  |  |  |  |
|    | 6.4                       | What We Learned                         | 26           |  |  |  |  |  |  |
|    | T .                       |                                         | o <b>=</b>   |  |  |  |  |  |  |
| II | 1 '                       | Testing and Analysis                    | 27           |  |  |  |  |  |  |
| 7  | Testing                   |                                         |              |  |  |  |  |  |  |
|    | 7.1                       | Description                             | 28           |  |  |  |  |  |  |
|    | 7.2                       | Theoretical Analysis                    | 28           |  |  |  |  |  |  |
|    | 7.3                       | Measurements                            | 28           |  |  |  |  |  |  |

| 8            | Timing Analysis                |    |  |  |  |  |  |  |
|--------------|--------------------------------|----|--|--|--|--|--|--|
|              | 8.1 Description                | 29 |  |  |  |  |  |  |
|              | 8.2 Acquiring Timings          | 29 |  |  |  |  |  |  |
| $\mathbf{A}$ | Raw Measurement Data           | 30 |  |  |  |  |  |  |
|              | A.1 Trial Information          | 30 |  |  |  |  |  |  |
|              | A.2 Function Runtime Profiling | 31 |  |  |  |  |  |  |

## List of Algorithms

## List of Figures

# Part I Introduction

# Part II Awesome RTX

## Global Variables

### 1.1 Description

There are various amounts of variables used throughout the RTX. This section describes the global variables that are used to accomplish tasks in the RTX. Other global variables such as the global process IDs can be seen in the appendix TODO. There are three major sections that required global variables and data structures: memory management in the heap, scheduler, and user processes.

## 1.2 Heap Data Structures

The RTX provides functionality to request and release memory from the heap, which is shared and stored in the RAM of the board. There is one main data structure that stores each memory block.

```
typedef struct HeapBlockHeader {
   int source_pid;
   int dest_pid;
   unsigned int send_time;
   struct HeapBlock* p_next;
   struct HeapBlock* p_next_usr;
} HeapBlockHeader;

#define HEAP_BLOCK_SIZE 128

typedef struct HeapBlock {
   HeapBlockHeader header;
```

```
byte data [HEAP_BLOCK_SIZE];
} HeapBlock;
```

The HeapBlock structure stores a header and the content of the block. When a memory is requested, users are given the data with a adjustable size of HEAP\_BLOCK\_SIZE and does not have knowledge of the header. Helper functions in the RTX turn the user block back into kernel block by adjusting the pointer of the block. Each header contains information about message passing (if the block is used for this purpose) and a pointer to the next block (for kernel memory management purposes). This global data structure is used in the following sections:

- 1. **Memory Management** this section uses this data structure to give and receive back heap memory blocks
- 2. **Process Message Passing** this section uses this data structure pass messages between processes. Message envelope are implemented as HeapBlocks.
- 3. **Timer I-Process** The process uses message passing to send delayed messages.
- 4. **UART I-Process** The process uses message passing to process input and output characters.
- 5. **CRT I-Process** The process receives messages and passes message for terminal output.
- 6. **KCD Process** The KCD uses message passing to CRT for display and user processes for processing.

## 1.3 Process Scheduler Data Structures

The RTX has a fixed-priority based scheduler that acts as a uniprocessor system. Context switching is required and the following data structures are used for this purpose.

#### 1.3.1 Ready and Blocked Priority Queues

On initialization, each in memory process is given a process control block (PCB). The PCB contains data on the process such as its process ID, stack pointer, process state, and priority that the kernel will use for scheduling. The blocked and ready priority queues of PCBs keep track of which processes are blocked and ready for execution. Each process is in one of the 5 states at all time listed in ProcessState. More details about the states can be found in section TODO. Processes that are in state PROCES\_STATE\_READY are on the ready queue. Processes in state PROCESS\_STATE\_BLOCKED are in the blocked queue. There is also an implicit queue for processes blocked on message but we won't go into that in this section.

```
typedef enum {
    PROCESS_STATE_NEW
    PROCESS_STATE_READY
                                       = 1,
    PROCESS_STATE_RUNNING
                                       = 2.
    PROCESS_STATE_BLOCKED
    PROCESS\_STATE\_BLOCKED\_ON\_MESSAGE = 4,
7 } ProcessState;
 typedef struct PCB {
    // Stack pointer
    U32*
                     sp;
    ProcessID
                     pid;
    ProcessState
                     state;
    ProcessPriority priority;
    struct PCB*
                     p_next;
17
    // Incoming messages, waiting to be processed.
    HeapBlock*
                    message_queue;
  } PCB;
21
  PCB* g_ready_process_priority_queue [PROCESS_PRIORITY_NUM] = {
PCB* g_blocked_process_priority_queue[PROCESS_PRIORITY_NUM];
```

PCBs and the priority queues are used in the following sections:

1. **Memory Management** - When a process requests memory without available memory blocks, the memory management unit must add the process to a blocked queue. When a memory block is released, the pro-

cess with the highest priority in the blocked queue (if any) is moved into the ready queue. Thus, the memory management unit must have access to the PCBs, and blocked and ready queues in order to accomplish these task.

- 2. **Process Management** The process management unit is the one that schedules processes and keep track of the process priorities. Ready processes are taken from the ready queue to be ran if the current process is blocked or finished its execution. The process management unit also takes care of updating the two priority queues when the priority of a process has been changed.
- 3. **UART Process** The UART Process needs access to the PCB. The PCBs also contain a mailbox for messages. The UART displays any incoming messages to the terminal display. Thus, it needs to know the PCB structure in order to gain access to the mailbox.

#### 1.3.2 Priorities

The RTX is based on a fixed-priority scheduler with 5 user process priorities and 2 system process priorities given below. Users are given priorities PROCESS\_PRIORITY\_HIGH to PROCESS\_PRIORITY\_LOWEST. PROCESS\_PRIORITY\_NULL\_PROCESS is given to the null process and PROCESS\_PRIORITY\_SYSTEM\_PROCESS are given to critical processes such as the KCD, CRT, timer I-process, and UART I-process.

```
1 typedef enum {
    PROCESS_PRIORITY_INVALID
    PROCESS_PRIORITY_SYSTEM_PROCESS
    PROCESS_PRIORITY_HIGH
    PROCESS_PRIORITY_MEDIUM
    PROCESS_PRIORITY_LOW
    PROCESS_PRIORITY_LOWEST
    PROCESS_PRIORITY_NULL_PROCESS
    PROCESS_PRIORITY_UNSCHEDULABLE
    PROCESS_PRIORITY_NUM
                                     = 8
  } ProcessPriority;
13
  typedef enum {
    USER_PROCESS_PRIORITY_HIGH
                                          = 0,
    USER_PROCESS_PRIORITY_MEDIUM
                                          = 1.
```

```
USER_PROCESS_PRIORITY_LOW = 2,
USER_PROCESS_PRIORITY_LOWEST = 3,

USER_PROCESS_PRIORITY_NUM = 4,
User_Process_Priority;
```

This priority structure is used by the process management unit to schedule and block processes based on their priorities. The RTX provides users with UserProcessPriority while keeping an internal structure of ProcessPriority.

#### 1.3.3 Message Passing

Message passing is a way for interprocess communication provided by the RTX. Messages are created in the form of a msgbuf structure, which includes the type and content. Some processes such as the CRT and KCD will require a certain type of message to be sent before the message can be processed correctly.

The message passing data structure is evident in the KCD, wall clock process, and all I-processes. It can also be used in all user processes.

## Kernel API

## 2.1 Description

This section describes the kernel API that is available to users in the RTX. It will only go into details of how to use each API function call and states of different scenarios. Details of the implementation can be found in section TODO or from the raw code.

#### 2.2 Memory Management

#### 2.2.1 Description

The RTX provides the utility of simple memory management of the heap. The main memory on the board (TODO: name the board) is divided into sections of the RTX Image, PCB data, the heap, and process stacks seen in figure blah. \*\*\* TODO insert figure blah here \*\*\*

Board LPC17xx (TODO insert the real name) does not have the hardware to support virtual memory. Thus, a fixed memory management scheme is used with variable block sizes. The OS kernel is alway loaded into main memory. When the OS boots, a user stated number of PCBs and process stack are allocated. The OS also allocates memory for system processes such as the Keyboard Command Decoder Process (KCD) and the CRT Display Process. After allocation of PCBs and process stacks, the remaining memory is used for the heap, which is shared between all processes. This section will be focused on the memory management of the heap.

The heap is further divided into a variable number of blocks. Each block contains a header (HeapBlockHeader) and 128 bytes of data. Depending on the number of user and system processes, the number of available heap memory will vary. The size of the data can vary from 128 bytes but must be set at compile time. The OS supports two kernel calls that gives access to these memory blocks.

#### 2.2.2 Requesting Memory

```
void *request_memory_block();
```

The first functionality supported by the OS is the ability to request memory. This call gives back a pointer to a memory block in the heap. The size of the memory block is defined by the constant HEAP\_BLOCK\_SIZE, which defaults to 128 bytes. These blocks are used for storing local variables or as envelopes for interprocess communication (described in section TODO). The user must cast the memory block to the proper type for his or her own use. An example of the usage is provided below, which stores the numbers 0-31 in an array of size 32.

```
void user_process() {
    int size = 32;

    int* array = (int*)request_memory_block();

    for( int i = 0; i < size; i++ ) {
        array[i] = i;
    }

    // ...

release_memory_block( (void*)array );
}</pre>
```

All processes will share the same heap memory pool. Thus, this primitive will block the process if the OS does not have anymore memory blocks to give out at the time, thus effecting the execution of the process. In that case, it will only be unblocked there is a new memory block available and if it has the highest priority on the list of processes waiting for a memory block. When using the memory block, the user must be aware of writing past the

heap block size. The OS does not check for segmentation faulting. Thus, undefined behavior may occur. Also, the user must remember to release the memory block or memory leakage will occur.

#### 2.2.3 Releasing Memory

```
int release_memory_block(void *memory_block);
```

The second functionality supported by the OS is the ability to release memory. This is a non-blocking call that returns the memory block back to the OS. This should be called when a message is received and not passed on or when the process is done using the memory block. If any process is blocked on memory, it will be unblocked and put to the ready queue. If the current process has lower priority than the unblocked process, then the current process will be preempted and the higher priority process will be executed instead. An example of this can be seen in figure CODE1 TODO on line 13.

### 2.3 Processor Management

```
int release_processor();
```

The OS manages processes as though it is on a uniprocessor. A priority based scheme with context switching is used for scheduling processes. A process can voluntarily release the processor to the OS at any time during its execution. If there are errors in the call, it will return an error without releasing the processor to the OS. If there are no errors and the invoking process is ready to execute, it is put to the end of the ready queue of its priority. If there are no other processes of equal or higher priority, the process will be chosen to execute again. However, if there is, another process may be chosen for execution. Below is an example which prints an increasing number and releases the processor at every turn.

```
void user_process() {
    static num = 0;
    while(1) {
        print(num++);
        release_processor();
}
```

```
7 }
```

### 2.4 Interprocess Communication

Apart from heap memory, processes do not share information. Thus, communication between processes is done through message-based interprocess communication (IPC). Details of the internal layout of process mailboxes can be found in section TODO. The RTX gives three primitives to carry out this task, one for sending, one for delayed sending, and one for receiving.

#### 2.4.1 Send Messages

```
int send_message(int process_id, void *message_envelope);
```

A process can compose a message envelope to be sent to another process. Memory for envelope message must be requested from the RTX using the request\_memory() routine. The envelope consists of a type (mtype) and the message data (mtext) which must be filled in as seen below. The predefined message types are used for the KCD, CRT, and Wall Clock process, which are built into the RTX (see section TODO for more information). The message data has a predefined size which is a MessageType smaller than the HEAP\_BLOCK\_SIZE. Thus, any message that are longer will exhibit undefined behavior. The process ID of the receiving process must also be known ahead of time in order to use send\_message.

```
char mtext [HEAP_BLOCK_SIZE - sizeof (MessageType)];
};
```

The primitive returns a status which validates the, message, receiver process ID, and ready queue. User processes is allowed to send a message to any user processes or system processes such as the KCD. If the receiving process is currently blocked on receiving message, this call will add the message to the process' message box and put it back on the message queue. The current process continues to execute unless the receiving process has a higher priority. In that case, the current process will be preempted and put to the back of the ready queue. Thus, this primitive may effect the execution of the process.

#### 2.4.2 Receive Messages

```
void *receive_message(int *sender_id);
```

A process can use this primitive to receive messages from other processes. Unless the sender\_id is NULL, the sender of the message will be written to the sender\_id parameter. The current process will check its mailbox for any incoming messages. If there are no messages in its mailbox, the routine will block the process and select another process for execution. Execution of the process will occur again if a message is sent from another process and this process has the highest priority in the ready queue. The message\_envelope are heap memory underneath. Thus, it is required that unless the process passes the message onto another process, the final receiving process must release the message\_envelope block after its usage. An example of send\_message and receive\_message is shown below.

```
message_envelope->mtype = MESSAGE_TYPE_USER_DEFINED_2;
13
          strcpy(message_envelope->mtext, "print this");
          send_message(PROCESS_RECEIVE_ID, message_envelope);
      }
17
  void process_receive() {
19
      while (1) {
          struct msgbuf* message
21
              = (struct msgbuf*)receive_message(NULL);
          if ( message->mtype = MESSAGE_TYPE_USER_DEFINED_2 ) {
23
               // Send to CRT for printing; do not release memory
              message->mtype = MESSAGE_TYPE_CRT_DISPLAY_REQUEST;
25
               send_message(PROCESS_ID_CRT, message);
          } else {
               // Do something with the message then release it
               release_memory_block( (void*) message );
29
          }
      }
31
```

#### 2.4.3 Receive Messages

```
int delayed_send(int process_id, void *message_envelope, int
    delay);
```

This primitive is very similar to the primitive of send\_message in section 2.4.1 with the addition of a delay. Instead of sending the message immediately, the message will be sent delay number of milliseconds. The message\_envelope is constructed in the same way. The process\_id will be the receiving process. The execution of the process may be preempted after a the delay period if the receiving process was blocked on receive\_message and has a higher priority.

#### 2.5 Process Priority

The RTX schedules processes based on a fixed priority scheme. Thus, the RTX provides the utility to change priorities and to get the process priority of any user processes.

#### 2.5.1 Set Process Priority

```
int set_process_priority(int process_id, int priority);
```

This primitive allows user processes to set the priorities of other user processes. User processes are not allowed to set the priorities of any system processes. System processes, however, are allowed to set the priorities of other system or i-processes. The primitive validates if the current process is allowed to set the priority of the process with process\_id and whether it's a valid process\_id. It also checks the validity of the priority. An error status of RTX\_ERR is given back if the validation does not pass. If properly set, a status of RTX\_OK is returned. The caller of this primitive is not blocked but can be preempted if the priority of the set process is higher. Otherwise, the process continues to execute. An example usage is shown in section 2.5.2

#### 2.5.2 Get Process Priority

```
int get_process_priority(int process_id);
```

Given a process\_id, this primitive gives back the priority of any processes including any system or i-processes to a user process. A invalid process\_id will result in a RTX\_ERR status. Otherwise, a RTX\_OK is returned to the caller. An example usage of both get and set process priority is shown below.

## Interrupts and Their Handler and Processes

- 3.1 Description
- **3.2** UART
- 3.3 TIMER

## System and User Processes

## 4.1 System Processes

#### 4.1.1 Description

The RTX consists of system processes that sit on top of the kernel. These system processes usually have higher priority than user processes because they carry out tasks for the OS that the kernel cannot do otherwise. System processes in this RTX includes the null process, and I/O processes.

#### 4.1.2 Null Process

The design of the RTX requires that a process should always be executing. Thus, if there are no user or other system processes, the null process executes. The only job of the null process is to keep the processor running. It has the lowest priority of any process using the RTX. If any process is added into the ready queue during the execution of this process, a preemption will occur and the new process will take control of the processor. The final implementation follows the following form.

function NULL\_PROCESS
while true do
end while
end function

#### 4.1.3 CRT Display Process

The RTX supports output to terminal display. CRT is a system process that the RTX uses to accomplish the task of display. The CRT process depends on the UART I-Process (details of the UART can be found in section TODO). The CRT is implemented using the following algorithm.

```
function CRT_PROCESS
  while true do
    message = RECEIVE_MESSAGE(NULL)
    if message is not a CRT display request then
        continue
    end if
        SEND_MESSAGE(PROCESS_ID_UART, message)
        Set interrupt bits
    end while
end function
```

The CRT display process accepts message requests from any process of type MESSAGE\_TYPE\_CRT\_DISPLAY\_REQUEST. This tells the CRT to send the message for display to the UART I-process, which interacts with the hardware registers to output to terminal. Since the UART process is an interrupt process, the interrupt bit must be set in order for the process to run. Thus, after sending the message to UART's mailbox, the CRT process enables the interrupt bits for the I-process to trigger. The UART I-process passes the message character by character to the hardware to display to the terminal.

#### 4.1.4 Keyboard Command Decoder Process

The RTX provides functionality to forward user input commands to a particular process. The keyboard command decoder process (KCD) handles two tasks: giving input to the CRT for terminal display and forwarding certain inputs to different processes for further processing. For the first task, all keyboard inputs are filtered by the KCD. The UART I-process is called when a key is first pressed. The UART I-process (explained further in section TODO) then passes a message to the KCD telling of type key press. In the first case, the KCD will take this message and pass it on to the CRT for terminal display.

In the latter case, processes can register a command with the KCD. Each

command registration consists of a single capital letter of the alphabet. If a command is taken, the KCD will not register the command and exit with an error. After a valid command has been registered, user input with the pattern "%" followed by a upper case character will be parsed by the KCD. If user types in "%" followed by a registered character, the KCD will buffer the input until the user presses the return key. In that scenario, the KCD will pass it onto the registered process. The following algorithm of the KCD explains this in more detail.

```
function PROCESS_INPUT(message)
   if There a "%" character in the buffer then
      if message. Content is a command character then
         add to buffer
      else
         clear buffer
      end if
   else if message.Content is "%" then
      add to buffer
   else if message. Content is enter and buffer is not empty then
      send buffer to registered command
   end if
   Send message to CRT to display
end function
function PROCESS_REGISTER(message)
   if message. Command has not been taken then
      register message.Command
   else
      error
   end if
end function
function KCD_PROCESS
   initialization
   while true do
      message = RECEIVE\_MESSAGE(NULL)
      if message type is key press input then
         PROCESS_INPUT (message)
      else if message type is command registration then
```

## PROCESS\_REGISTER(message) end if end while end function

An example of this usage is the wall clock process (described in more detail in section 4.2.1). The wall clock process registers the character 'W' as a command. The registration will be done by passing a message to the KCD of command registration type. This message is received by the KCD. If the 'W' command has not been registered, it becomes registered by the wall clock process. Meanwhile, the user types in random characters. Any character that is not '%' is passed by the KCD to the CRT for display. Once a '%' is received, the KCD checks the next character to see if it's a registered command. If its not a registered command, the KCD clears the buffer and starts at the beginning. If it is a registered command, the KCD keeps a buffer of all the input characters. This buffer fills with data and is sent to the wall clock process for processing after the user presses enter. The buffer is only 20 characters long. Thus, any data for a process after 20 characters will not be sent to the registered process on enter.

## 4.2 User Processes

- 4.2.1 Wall Clock Process
- 4.2.2 'funProcess'
- 4.2.3 'schizophrenicProcess'
- 4.2.4 'fibProcess'
- 4.2.5 'memoryMuncherProcess'
- 4.2.6 'releaseProcess'

## Initialization

- 5.1 Description
- 5.2 Memory Layout
- 5.3 Process Mailbox

## Major Design Changes

- 6.1 Description
- 6.2 Heap
- 6.3 Scheduler
- 6.4 What We Learned

# Part III Testing and Analysis

## Testing

- 7.1 Description
- 7.2 Theoretical Analysis
- 7.3 Measurements

## Timing Analysis

- 8.1 Description
- 8.2 Acquiring Timings

## Appendix A

## Raw Measurement Data

## A.1 Trial Information

| Trial | Total Runtime | Notes                                |
|-------|---------------|--------------------------------------|
| 1     | 4.219         | Normal (no stress processes)         |
| 2     |               | Wall clock                           |
| 3     | 8.487         | Normal (no stress processes)         |
| 4     | 6.5           | No Memory Muncher or Release Process |
| 5     | 30.988        | Stress processes                     |

## A.2 Function Runtime Profiling

| Function                  | Trial | Time $(\mu s)$ | # of Calls | Average time / call $(\mu s)$ |
|---------------------------|-------|----------------|------------|-------------------------------|
| k_sendMessage             | 1     | 601.58         | 552        | 1.090                         |
| $k_{-}$ receive $Message$ | 1     | 408.22         | 565        | 0.723                         |
| $k_acquireMemoryBlock$    | 1     | 244.12         | 294        | 0.830                         |
| $k\_sendMessage$          | 2     | 647.44         | 594        | 1.090                         |
| $k_{ m receive Message}$  | 2     | 437.78         | 606        | 0.722                         |
| $k_{-}acquireMemoryBlock$ | 2     | 258.68         | 320        | 0.808                         |
| $k\_sendMessage$          | 3     | 630.99         | 579        | 1.090                         |
| $k_{-}$ receive $Message$ | 3     | 426.83         | 591        | 0.722                         |
| k_acquireMemoryBlock      | 3     | 259.24         | 321        | 0.808                         |
| $k\_sendMessage$          | 4     | 108.80         | 100        | 1.088                         |
| $k_{receiveMessage}$      | 4     | 74.44          | 110        | 0.677                         |
| k_acquireMemoryBlock      | 4     | 92.47          | 123        | 0.752                         |
| $k\_sendMessage$          | 5     | 750.63         | 687        | 1.093                         |
| $k$ _receiveMessage       | 5     | 497.09         | 693        | 0.717                         |
| $k_acquireMemoryBlock$    | 5     | 329.90         | 447        | 0.738                         |