Skip to content
/ floem Public

Programming system for NIC-accelerated network applications

License

Notifications You must be signed in to change notification settings

mangpo/floem

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Floem

Floem is a DSL, compiler, runtime for NIC-accelerated network applications. Currently, the compiler can generate code for

  • a system with only a CPU with DPDK
  • a system with a CPU and Cavium LiquidIO nic

The DSL is a Python Library. The library provides mechanisms to connect elements, mapping elements to hardware resources, and create function interfacess for external applications. An element itself is implemented in C. The compiler then generates C program that can be executed.

Table of Contents

A. Prerequisites

B. Language

  1. Element
  2. Element Connection
  3. Segment
  4. State
  5. Shared Data
  6. Composite Element

C. Running on Cavium

D. Library of Elements

E. Library of C Functions

F. Queue Synchronization Layer

A. Prerequisites

Dependencies

  • GCC
  • Python 2

For a CPU-only system, you require DPDK to be installed. Set variable dpdk_dir in floem/target.py to DPDK directory's path. If you do not intent to use DPDK, leave the variable as is.

For a system with a Cavium LiquidIO NIC, you require Cavium SDK, source and binary for LiquidIO driver and firmware. Copy all files in include_cavium directory of this repo to liquidio-linux-driver-fwsrc-1.6.1_rc2/octeon/se/apps/nic directory in the LiquidIO firmware's source.

Setup

At the top level of the repo, install Floem local package:

python configure.py
pip install . --user

Testing

Run floem/programs/hello.py. It should print out a bunch of "hello world" before stopping. You should find tmp executable binary in floem/programs/. Try running tmp via commandline. It should keep printing "hello world" forever.

Import

To use Floem, simply import floem in your python program.

from floem import *

B. Language

1. Element

An element is an executable piece of code. To create an instance of an element, first you need to create a element class and use the element class to instantiate an instance of the element.

Example

class Inc(Element):
  def configure(self):
    self.inp = Input(Int)   # Input port
    self.out = Output(Int)  # Output port
    
  def impl(self):
    self.run_c(r'''
      int x = inp();        // Retrieve a value from input port inp.
      output { out(x+1); }  // Sent a value to output port out.
    ''')
    
  # This method can be removed because the implemention for Cavium is the same as that for CPU.
  def impl_cavium(self):
    self.run_c(r'''
      int x = inp();        // Retrieve a value from input port inp.
      output { out(x+1); }  // Sent a value to output port out.
    ''')
    
inc = Inc()  # create an element

Users must provide both the CPU implementation (via method impl) and the Cavium implementation (avia method impl_cavium) of an element. Often, CPU and Cavium implemntations are identical. In such case, users do not need to implment impl_cavium.

1.1 Element Port

Input and output ports are defined in configure method.

self.<port_name> = [Input | Output](arg_type0, arg_type1, ...)
Data Type

Provided data types are

  • Void (void), Bool (bool), Int (int), SizeT (size_t), Float (float), and Double (double)
  • Uint(<bits>) represents uint_t
  • Sint(<bits>) represents int_t
  • Pointer(<type>) represents pointer of
  • Array(<type>, <size>) represents array of with size

Users can create a composite data type (struct) using State. To refer to a struct in a C header file defined outside Floem, users can simply use string. For example, self.inp = Input("struct tuple *") is an input port that accepts a pointer to struct tuple defined outside Floem.

1.2 Element's Implementation (C Code)

Apart from the syntax to retrieve values from input ports and send values to output ports, self.run_c accepts any C code that can be written inside a C function.

Inputs

An element retrieves its input(s) by calling its input ports, e.g. int id = inp();. If an input port contains multiple values, an element can retrieves multiple values using a tuple, e.g. (int id1, int id2) = in();

Outputs

An element sends outputs by calling its output ports e.g. output{ out(id); }. The output ports can only be called within the output block output { ... } or output switch { ... }. An element must fire: (i) all its output ports, (ii) one of its output ports, or (iii) zero or one of its output ports. With an exception of a looping element, which can fire its only one output port many times outside the output block.

(i) Fire all output ports

In this case, all ports must be called inside the output block output { ... }.

(ii) Fire one of its output ports

In this case, the output block

output switch { 
  case <cond1>: <out1>(<args1>); 
  case <cond2>: <out2>(args2>); 
  ... 
  else: <outN>(argsN); 
  }

must be used. For example, output switch { case (id < 10): out1(id); else: out2(id); }.

(iii) Fire zero or one of its output ports

Similar to (ii), output switch block is used, but without the else case.

(iv) Fire an output port many times (looping element)
for(int i=0; i<10; i++) out(i);  // An output port can be called anywhere in the code.
output multiple;  // Annotate that this element fires an output port multiple times.

2. Element Connection

An output port of an element can be connected to an input port of another element. Let a, b, and c be elements, each of which has one input port and one output port. We can connect an output port of a to an input port of b, and connect an output port of b to an input port of c by.

a >> b >> c

Multiple Input Ports

If an element has multiple input ports, users must specify which port to connect explicitly.

a >> c.in0
b >> c.in1

Multiple Output Ports

If an element has multiple output ports, users must specify which port to connect explicitly.

a.out0 >> b
a.out1 >> c
Example 1

An output port of an element can be connected to multiple elements.

a >> b
a >> c
Example 2

If a fires only one of its output port, either b or c will be executed (not both).

a.out0 >> b
a.out1 >> c
Example 3

An input port of an element can be connected from multiple elements.

a >> c >> d
b >> c >> e

In this case, the same element c is executed after either a or b is executed. One thing to be caution is that both d and e are executed after c is executed. This is because c has one output port, which is connected to both d and e.

If we want to achieve sperate flows of a -> c -> d and b -> c -> e, we will need multiple instances of c as follows. Let C be a constructor of c.

c1 = C()
c2 = C()
a >> c1 >> e
a >> c2 >> e

Type Check

An output port out of an element a can be connected to an input port inp of an element b if their data types match. Otherwise, the compiler will throw an exception.

For one special case, an output port out of an element a can also be connected to an input port inp of an element b even when their types do not match, with the condition that port inp must has zero argument. In this case, the element a simply drops arguments that supposed to be sent to the element b. We introduce this feature as we find it to be quite convenient.

3. Segment

To execute a program, you need to assign an element to a segment to run. A segment is a set of connected elements that starts from a source element and ends at leaf elements (elements with no output ports) or queues. Packet handoff along a segment is initiated by the source element pushing a packet to subsequent elements.

3.1 Normal Segment

By default, a segment is executed by one dedicated thread on a device, so elements within a segment run sequentially with respect to packet-flow dependencies. A thread starts invoke the source node (an element that doesn't has any data dependency). Then, the rest of the elements are executed in a topological order according to the dataflow. A thread repeats the execution of the segment forever. If an element assigned to a segment is unreachable (no data dependency) from the source element, the compiler will throw an error. In the current implementation, if there is no dependency between elements x and y according to the dataflow, x is executed before y if x appears before y in the program.

To create a segment of connected elements, simply create elements and connect them inside a segment. For example:

class P(Segment):
    def impl(self):
        Hello()  # one element in this segment
     
P('segment_name')
Compile and Run

To compile all the defined segments into an executable file, add the following code in your program:

c = Compiler()
c.testing = "while(1) pause();"
c.generate_code_and_run()

c.testing is the body of the main function. In this case, we want the main function to run forever.

Under the hood,

  1. Floem compiler compiles all segments to tmp.c.
  2. GCC compiles tmp.c to executable tmp.
  3. tmp is then executed for a few second by the script.

All the generated files remain after the compilation process is done. Therefore, tmp can be executed later using command line.

See floem/programs/hello.py for a working exmaple.

Dependencies

If the C implemenations of elements require external C header files, we can set the include field of the compiler object to include appropriate header files. For example:

c.include = r'''#include "hello.h"'''

If the C implemenations of elements require external object files, we can set the depend field of the compiler object to a list of all object files (without '.o'). For example, we want to compile the program with object file hello.o:

c.depend = ['hello']

The compiler will first compile each C program in c.depend to an object file, and then compile tmp.c with all the object files.

See floem/programs/hello_depend.py for a working exmaple.

Testing

We can provide a list of expected outputs from the program as an argument to generate_code_and_run(expect). For example:

c.generate_code_and_run([12])

The compiler captures the stdout of the program and checks against the provided list. The list can contains numbers and strings. The stdout of the program is splitted by whitespace and newline.

3.2 Callable Segment

To enable programmers to offload their existing applications to run on a NIC without having to port their entire applications into Floem, we introduce a callable segment, which is a segment that is exposed as a function that can be called from any C program. This way, programmers have to port only the parts they want to offload to a NIC into Floem.

Programmers can create a callable segment similar to a normal segment, but additionally, they must define the argument types and the return type of the function via an input and output port of the segment. Note that a callable segment can take multiple input arguments though one input port. A callable segment can return nothing (no output port) or return one return value via an output port.

For example, we can create a function inc2 (a function that returns its argument added by 2) from element Inc (an element that increments its input by 1) as follows.

class Inc(Element):
  def configure(self):
    self.inp = Input(Int)
    self.out = Output(Int)
    
  def impl(self):
    self.run_c("int x = inp() + 1; output { out(x); }")
    
class inc2(CallableSegment):
  def configure(self):
    self.inp = Input(Int)
    self.out = Output(Int)
    
  def impl(self):
    self.inp >> Inc() >> Inc() >> self.out
    
inc2('inc2', process='simple_math')

The function inc2 can be used in any C program like a normal C function. To use the function, users have to including #include "simple_math.h". Notice that you can choose the name of the header to include.

Compile and Run 1

Similar to normal segments, we can compile callable segments using the same commands:

c = Compiler()
c.testing = r'''
int x = inc2(42);
printf("%d\n", x);
'''
c.generate_code_and_run()

Notice, that the main body can call function inc2. The compiler generates executable tmp similar to how it is done for normal segments. This method for compiling and running is good for testing, but not suitable for an actual use because we want to use function inc2 in any C program not in tmp.c where it is defined.

See floem/programs/inc2_function.py for a working exmaple.

Compile and Run 2

To compile callable segments to be used in any external C program, we write:

c = Compiler()
c.generate_code_as_header('simple_math')
c.depend = ['simple_math']
c.compile_and_run('simple_math_ext')

The code shows how to use Floem to compile an external C program simple_math_ext.c that calls function inc2 written in Floem. Under the hood,

  1. Floem generates simple_math.c that contains actual implementation of inc2 and simple_math.h that contains the function signature.
  2. GCC compiles all C programs listed in c.depend into object files (compiles simple_math.c to simple_math.o).
  3. GCC compiles the C program given to c.compile_and_run with the object files to generate an executable binary. In this case, it generates binary simple_math_ext from simple_math_ext.c and simple_math.o.
  4. Finally, simple_math_ext is executed.

See floem/programs/inc2_function_ext.py for a working exmaple.

External C Program's Initialization

The external user's application must call the following functions to properly initialize and run the system:

  • init() initializes states used by elements.
  • run_threads() creates and runs normal segments.

Note that multiple segments can be compiled into one file; the file name is indicated by the parameter process when creating a segment. Therefore, a process/file can contains both normal and callable segments. In such case, it is crucial to call run_threads() in order to start running the normal segments.

See floem/programs/simple_math_ext.c and floem/programs/inc2_function_ext.py for a basic working exmaple. See the main function inapps/memcached_cpu_only/test_no_steer.c and the very end of apps/memcached_cpu_only/test_no_steer.py for a more complexing exmaple. Do not try to understand the program at this point.

Default Return

If a callable segment has a return value, but the segment may not produce a return value because one or more elements in the segment may not fire its output ports, users have to provide a default return value to be used as the return value when the returning element of the API does not produce the return value. The default return value can be provided by assigning the default value to

  • the field self.default_return = val of the segment class, or
  • the parameter default_return when instantiating the segment segment_class(segment_name, default_return=val)

See apps/storm/main_dccp_nic_bypass.py for an exmaple. Search for "inqueue_get(CallableSegment)". Do not try to understand the program at this point.

3.3 run_order

If there are multiple source elements in a segment, we have to choose which source element is the starting element of the segment, and explicitly create dependency for the other source elements. For example, if we have a >> b and c >> d, and both a and c are source elements. We want a to be the starting element, and make c run after b. We can specify this intent using run_order as follows:

class P(Segment):
  def impl(self):
     a >> b
     c >> d
     run_order(b, c)  # run b before c

In the scenario where there are multiple leaf nodes l1, l1, ..., ln reachable from the starting element, and only one of them will be executed per one packet, we write:

run_order([l1, l2, ..., ln], c)  # run c after either l1, l2, .., or ln

3.4 Mapping to Device

By default, a segment is executed by one dedicated thread on a CPU. We can map a segment to a Cavium NIC by assinging the parameter device of the segment to target.CAVIUM: segment_class(segment_name, device=target.CAVIUM)

3.5 Mapping to Multiple Threads

By default, a segment is executed by one dedicated thread on a CPU, or one dedicated core on a Cavium NIC. We can run a segment on mutiple threads/cores in parallel by assigning parameter cores to a list of thread/core ids. For example, segment_class(segment_name, device=target.CAVIUM, cores=[0,1,2,3]) runs this segment using cores 0--3 on Cavium.

On CPU, we can use an unlimited number of threads. Floem relies on an OS to schedule which threads to run. On Cavium, valid core ids are 0 to 11, as there are 12 cores on Cavium. Assigning the same core id to multiple segments leads to undefined behaviors.

core_id

Often, we need to determine which physical queue instance to enqueue/dequeue based on the core/thread id. To facilitate this, a segment can access its core id via self.core_id. For example,

class P(Segment):
    self impl(self):
        self.core_id >> Display()
        
P('P', device=target.CPU, cores=[0,1,2,3])

This program launches four CPU threads (with ids 0--4), where each thread X repeatedly prints X.

4. State

State is like C-language struct. To create an instance of a state, first you need to create a state class and use the state class to instantiate an instance of the state.

Example

class Tracker(State):
  total = Field(Int)
  last = Field(Int)
  
  def init(self, total=0, last=0):
    self.total = total
    self.last = last
    
tracker1 = Tracker()
# ^ use the constructor's init: total = 0, last = 0
tracker2 = Tracker(init=[10,42])  
# ^ use this init: total = 10, last = 42

4.1 Persistent State (Shared State)

State can be used as a global variable shared between multiple elements and persistent across all packets. Users must explicitly define which states an element can access to. Note that an element can access more than one states.

Example

class Observer(Element):
  tracker = Persistent(Tracker)  # Define a state variable and its type.
  def states(self, tracker): 
    self.tracker = tracker       # Initialze a state variable.
  
  def configure(self):
    self.inp = Input(Int)   # Input port
    self.out = Output(Int)  # Output port
    
  def impl(self):
    self.run_c(r'''
      int id = inp();     // Retrieve a value from inp input port.
      tracker->total++;   // State is referenced as a pointer to a struct.
      tracker->last = id; 
      output { out(id); }
    ''')
    
o1 = Observer(states=[tracker1])  # observer element 1
o2 = Observer(states=[tracker1])  # observer element 2

Notice that o1 and o2 share the same tracker state, so they both contribute to tracker1->total. Also notice that there can be a race condition here because of the shared state.

4.2 Flow and Per-Packet State

Floem provides a per-packet state abstraction that a packet and its metadata can be accessed anywhere for the same flow. With this abstraction, programmers can access the per-packet state in any element without explicitly passing the per-packet state around. Flow is a set of segments that connect to each other via queues.

To use this abstraction, programmers define a flow class, and set the field state of the flow class to a per-packet state associated with the flow. The elements and segments associated with the flow must be created inside the impl method of the flow.

class ProtocolBinaryReq(State):
    ...
    keylen = Field(Uint(16))
    extlen = Field(Uint(8))
    
class KVSmessage(State):
    ...
    mcr = Field(ProtocolBinaryReq)
    payload = Field(Array(Uint(8)))
    
class MyState(State):         
   pkt = Field(Pointer(KVSmessage))
   hash = Field(Uint(32))
   key = Field(Pointer(Uint(8)), size='state->pkt->mcr.keylen')  # variable-size field

class Main(Flow):             # define a flow
  state = PerPacket(MyState)  # associate per-packet state
  
  def impl(self):
    class Hash(Element):
      def configure(self):
        self.inp = Input(SizeT, Pointer(Void), Point(Void))  
        self.out = Ouput()    # no need to pass per-packet state
    
      def impl(self):
        self.run_c(r'''
        (size_t size, void* pkt, void* buff) = inp();
        state->pkt = pkt;      // save pkt in a per-packet state
        uint8_t* key = state->pkt->payload + state->pkt->mcr.extlen;
        state->key = key;
        state->hash = hash(key, pkt->mcr.keylen);
        output { out(); }
        ''')
           
    class HashtGet(Element):
      ...
      def impl(self):
        self.run_c(r'''       // state refers to per-packet state
        state->it = hasht_get(state->key, state->pkt->mcr.keylen, state->hash);
        output { out(); }
        ''')
    ...

See apps/memcached_cpu_only/test_no_steer.py as a working example.

4.3 Special fields: layout, defined, packed

  • layout By default, the defined fields in a state is laid out in an artitrary order in a struct. To control the order, we must explicitly assign the layout field of the state.
  • packed By default, Floem compiles a defined state to a C struct with __attribute__ ((packed)). In some cases, this is undesirable. For example, if the state contains a spin lock, we do not want the struct to be pakced. In such scenario, we can set the packed field to False.
  • defined By default, Floem compiles a defined state to a C struct. However, if that struct has been defined in a C header file somewhere else that we include, we do not want to redefine the struct, but we still want to inform Floem about the struct so that it can reason about the per-packet state. In such case, we can set the defined field to False so that Floem will not generate code for the struct.
Example
class MyState(State):         
   pkt = Field(Pointer(KVSmessage))
   hash = Field(Uint(32))
   key = Field(Pointer(Uint(8)), size='state->pkt->mcr.keylen')  # variable-size field
   layout = [pkt, hash, key]
   defined = False
   packed = False

5. Shared Data

5.1 Shared state between devices

If a state is used in elements that mapped to both a CPU and a NIC, a state will be allocated on host memory (CPU's memory), and the elements on the NIC must access the state via DMA operations.

For example,

class Observer(Element):
  tracker = Persistent(Tracker)  # Define a state variable and its type.
  def states(self, tracker): 
    self.tracker = tracker       # Initialze a state variable.
  
  def configure(self):
    self.inp = Input(Int)   # Input port
    self.out = Output(Int)  # Output port
    
  def impl(self):           # Implementation for CPU
    self.run_c(r'''
      int id = inp();      // Retrieve a value from inp input port.
      tracker->total++;    // State is referenced as a pointer to a struct.
      tracker->last = id; 
      output { out(id); }
    ''')
    
  def impl_cavium(self):    # Implementation for Cavium NIC
    self.run_c(r'''
      int id = inp();      // Retrieve a value from inp input port.
      
      // Update tracker->total
      int *total;
      dma_read(&tracker->total, sizeof(int), &total);
      *total = *total + 1;
      dma_write(&tracker->total, sizeof(int), total, 1);  // 1 indicates blocking write
      
      // Update tracker->last
      int *last;
      dma_buf_alloc(&last);  // use dma_buf_alloc if we don't need the old content
      *last = id;
      dma_write(&tracker->last, sizeof(int), last, 1);
      
      dma_free(total); dma_free(last);
      output { out(id); }
    ''')
    
class P_CPU(Segment):
    def impl(self):
        o1 = Observer(states=[tracker1])  # observer element on CPU
        ...
        
class P_NIC(Segment):
    def impl(self):
        o2 = Observer(states=[tracker1])  # observer element on NIC
        ...
        
P_CPU(device=target.CPU)
P_NIC(device=target.NIC)

Note that if impl_cavium method is unimplementated, Floem uses impl method to generate code for both the CPU and the NIC.

5.2 Shared memory region

Floem provides another method for sharing a memory region between a CPU and a NIC or between different processes on a CPU without defining state. MemoryRegion(<name>, <size_in_bytes>) allocates a memory on host memory. <name>can be accessed by any element. For elements on a CPU, <name> is a local pointer to the shared region. For elements on a Cavium NIC, <name> is a physical address of the shared region, so we must use DMA operations to access the region. <name> can also be used in an external C program that includes the header filed generated by Floem.

See floem/programs_perpacket_state/queue_shared_pointer.py for a working example. In this example, two external C programs (queue_shared_p1_main.c and queue_shared_p2_main.c) running in separate processes access the shared memory region called data_region created in Floem.

6. Composite Element

A composite element is a collection of smaller elements. Unlike a primitive element, the entire composite element may not be executed all at once. For example, a composite element is composed of two different independent elements a and b; the input to a does not come from b, and vice versa. In such case, when the input to a is ready, a is executed; when the input to b is ready, b is executed; they don't depend on each other. However, normally elements that compose a composite element usally related to each other in some way. For example, a composite element queue may be composed from enqueue and dequeue elements, which share a state storing queue content.

To create an instance of a composite element, first you need to create a composite element constructor and use the element constructor to instantiate an instance of the composite element.

Example

A queue composite element can be created from an enqueue and dequeue element that shares a queue storage state as follows:

class Queue(Composite):
  def configure(self):
    self.inp = Input(Entry)
    self.out = Oput(Entry)
    
  def impl(self):
    storage = Storage()  # Create a storage state.
    enq = Enqueue(states=[storage])
    deq = Dequeue(states=[storage])
    self.inp >> enq
    deq >> self.out
    
queue = Queue()

# Then you can use queue like a typical element.
a >> queue >> b

C. Running on Cavium

Prerequisite

Developers must have source for Cavium LiquidIO's firmware and Cavium SDK.

When compiling an application that runs on both a CPU and a Cavium NIC, Floem will generate:

  • CAVIUM.c and CAVIUM.h for the NIC
  • one or more executable binaries for the CPU. For example, when compiling apps/benchmarks_simple/inqueue.py, Floem will generate test_queue executable for the CPU.

To run the application,

  1. Run the CPU executable if there is one. It should print out physical address = <address>.
  2. Replace STATIC_ADDRESS_HERE in CAVIUM.c with <address>. This is the host physical address for NIC-CPU communications.
  3. Copy CAVIUM.c and CAVIUM.h to liquidio-linux-driver-fwsrc-1.6.1_rc2/octeon/se/apps/nic (the LiquidIO firmware source).
  4. Make and install LiquidIO's driver and firmware using the following commands:
cd liquidio-linux-driver-fwsrc-1.6.1_rc2
make
sudo make host_install
sudo rmmod liquidio.ko
sudo insmod bin/liquid.ko

Make sure that the CPU executable is running (if there is one) before running the last command. 5. Finally, we should set up an IP address for the ethernet port that the Cavium NIC is connected to. For example,

sudo ifconfig eth2 10.3.0.35
Additional object files

If the code running on the Cavium NIC depends on C functions implemented outside Floem in xxx.c, users must

  1. Include those files in liquidio-linux-driver-fwsrc-1.6.1_rc2/octeon/se/apps/nic directory.
  2. In the same directory, edit Makefile. Search for
#Action for making cvmcs-nic
OBJS =  $(OBJ_DIR)/cvmcs-nic-main.o   \
		$(OBJ_DIR)/cvmcs-nic-init.o   \
		$(OBJ_DIR)/cvmcs-nic-printf.o   \
		...

and add xxx.o to the list. Note that the name xxx must match your C file.

Data Synchronization Runtime for NIC-CPU Queues
  • Set the number of Cavium cores to run the runtime manager threads by changing #define RUNTIME_CORES in liquidio-linux-driver-fwsrc-1.6.1_rc2/octeon/se/apps/nic/floem-util.h to the desired number N. Floem will use the last N cores to manage the runtime (i.e. cores 11, 10, .., 12-N). 12-N cores will be used to run the NIC worker threads. N is usually set to 1-4.
  • Instead of using dedicated cores to manage the runtime, the queue managment can also be performed by the cores that use the queues. To use this setting, comment out #define RUNTIME in floem-util.h. RUNTIME_CORES is ignored when RUNTIME is not defined.