$darkmode
Elektra 0.11.0
Files
I/O Bindings

Asynchronous I/O feature. More...

Files

file  kdbio.h
 Elektra-I/O structures for I/O bindings, plugins and applications.
 
file  kdbioplugin.h
 Elektra-I/O functions and declarations for the I/O binding test suite.
 
file  kdbiotest.h
 Elektra-I/O functions and declarations for the I/O binding test suite.
 

Detailed Description

Asynchronous I/O feature.

Asynchronous I/O with Elektra

Overview

I/O bindings allow Elektra and its plugins to integrate into different main loop APIs using a thin abstraction layer. For example, this is used for notification transport plugins which receive notifications using ZeroMQ, D-Bus, etc.

I/O bindings are created using an initialization function for a specific main loop API. Please see bindings for available I/O bindings and their according READMEs for more details. After creating, an I/O binding is associated to a KDB instance using elektraIoSetBinding(). Having different I/O bindings (e.g. same or different main loop APIs) for different KDB instances is supported.

The remainder of this page contains useful details for creating I/O bindings and using the operations provided by these bindings. Application developers are normally not required to do those tasks. For more information about using I/O bindings from an application developer perspective please read the Notification Tutorial.

Introduction

An I/O binding needs to handle different types of operations. These operations are used by plugins that require asynchronous I/O. In this document we will call developers of these plugins "users". The three types of operations are:

Each operation has a user callback that is called under the following conditions:

Each operation has different properties. The following properties are shared by all operations:

For brevity only file descriptor operation variants are listed here. Variants for timer and idle operations are called elektraIoTimer* and elektraIoIdle*. All elektraIo* utility functions are provided by the elektra-io library.

File descriptor watch operations have the following additional properties:

Timer operations have the following additional properties:

Idle operations have no additional properties.

Creating a new I/O Binding

Every I/O binding needs to provide ten functions:

In order to create a new I/O binding you have to create an entry point for your binding (e.g. elektraIoDocNew()). This entry point then calls elektraIoNewBinding() with pointers to the ten required functions.

// Initialize I/O interface
// file descriptors
// timers
// idle
// cleanup
if (binding == NULL)
{
ELEKTRA_LOG_WARNING ("elektraIoNewBinding failed");
return NULL;
}
ElektraIoInterface * elektraIoNewBinding(ElektraIoBindingAddFd *addFd, ElektraIoBindingUpdateFd *updateFd, ElektraIoBindingRemoveFd *removeFd, ElektraIoBindingAddTimer *addTimer, ElektraIoBindingUpdateTimer *updateTimer, ElektraIoBindingRemoveTimer *removeTimer, ElektraIoBindingAddIdle *addIdle, ElektraIoBindingUpdateIdle *updateIdle, ElektraIoBindingRemoveIdle *removeIdle, ElektraIoBindingCleanup *cleanup)
Create a new I/O binding.
Definition: io.c:43
int ioDocBindingUpdateFd(ElektraIoFdOperation *fdOp)
Update information about a file descriptor watched by I/O binding.
Definition: io_doc.c:368
int ioDocBindingUpdateTimer(ElektraIoTimerOperation *timerOp)
Update timer in I/O binding.
Definition: io_doc.c:432
int ioDocBindingAddTimer(ElektraIoInterface *binding, ElektraIoTimerOperation *timerOp)
Add timer for I/O binding.
Definition: io_doc.c:448
int ioDocBindingCleanup(ElektraIoInterface *binding)
Cleanup.
Definition: io_doc.c:551
int ioDocBindingUpdateIdle(ElektraIoIdleOperation *idleOp)
Update idle operation in I/O binding.
Definition: io_doc.c:492
int ioDocBindingRemoveFd(ElektraIoFdOperation *fdOp)
Remove file descriptor from I/O binding.
Definition: io_doc.c:414
int ioDocBindingRemoveIdle(ElektraIoIdleOperation *idleOp)
Remove idle operation from I/O binding.
Definition: io_doc.c:534
int ioDocBindingAddIdle(ElektraIoInterface *binding, ElektraIoIdleOperation *idleOp)
Add idle operation to I/O binding.
Definition: io_doc.c:507
int ioDocBindingAddFd(ElektraIoInterface *binding, ElektraIoFdOperation *fdOp)
Add file descriptor to I/O binding.
Definition: io_doc.c:384
int ioDocBindingRemoveTimer(ElektraIoTimerOperation *timerOp)
Remove timer from I/O binding.
Definition: io_doc.c:476
struct _ElektraIoInterface ElektraIoInterface
I/O binding handle.
Definition: kdbio.h:22

If your I/O management library requires you to store additional data you can do so using elektraIoBindingSetData(). Let's assume you have the following data structure:

typedef struct DocBindingData
{
char * foo;
// Add additional members as required
struct DocBindingData DocBindingData
[kdbio operation data]
[kdbio operation data]
Definition: io_doc.c:269
char * foo
Example member.
Definition: io_doc.c:271

Then you can store your data with the I/O binding.

// Store binding relevant data in the interface
DocBindingData * bindingData = elektraMalloc (sizeof (*bindingData));
if (bindingData == NULL)
{
ELEKTRA_LOG_WARNING ("elektraMalloc failed");
return NULL;
}
elektraIoBindingSetData (binding, bindingData);
bindingData->foo = foo;
void * elektraMalloc(size_t size)
Allocate memory for Elektra.
Definition: internal.c:274
int elektraIoBindingSetData(ElektraIoInterface *binding, void *data)
Set private data from I/O Binding.
Definition: io.c:296

Of course if you need to store only a single pointer (e.g. a handle) you can omit the struct and directly use elektraIoBindingSetData() with your pointer.

Implementing Operations

The next step is to implement operation functions. We'll walk through the implementation of the functions for managing file descriptor watch operations. Timer and idle variants are the same except for the operation properties.

For reconstructing the user callback it is advisable to store a context for each operation in your I/O management library. Most I/O management libraries let you pass this context when adding an operation to the library. This context is then passed by the library back to your callbacks. You can use the operation data itself as context and store additional data like handles from your I/O management library by using elektraIoFdSetBindingData().

Let's assume the data structure looks like this:

typedef struct DocOperationData
{
char * bar;
// Add additional members as required
struct DocOperationData DocOperationData
[kdbio operation data]
[kdbio operation data]
Definition: io_doc.c:257
char * bar
Example member.
Definition: io_doc.c:259

Using this struct's members you can store additional data like handles in operations. The member bar is just an example.

The following snippet from ioDocBindingAddFd() shows example code for ElektraIoBindingAddFd. Code for ElektraIoBindingAddTimer and ElektraIoBindingAddIdle is similar.

DocOperationData * operationData = newOperationData ();
if (operationData == NULL)
{
return 0;
}
// You can use private data stored in the I/O binding
// e.g. MyData data = (MyData *)elektraIoBindingGetData (binding);
elektraIoFdSetBindingData (fdOp, operationData);
// You can store additional data for each operation in your operationData structure
operationData->bar = "foo";
// Here you need to add the operation to the I/O management library
// ioDocBindingFdCallback() holds an example callback to pass to your I/O management library
// assume SomeIoLibHandle * someIoLibAddFd (int fd, int flags, int enabled, void * privateData, callback)
// operationData->handle = someIoLibAddFd (elektraIoFdGetFd (fdOp), elektraIoFdGetFlags (fdOp), elektraIoFdIsEnabled (fdOp), &fdOp,
// ioDocBindingFdCallback)
return 1;
int elektraIoFdSetBindingData(ElektraIoFdOperation *fdOp, void *data)
Set private binding data for operation.
Definition: io.c:487
DocOperationData * newOperationData(void)
[kdbio binding data]
Definition: io_doc.c:280

In ElektraIoBindingUpdateFd or ElektraIoBindingRemoveFd you can access your binding operation data by using elektraIoFdGetBindingData().

void * elektraIoFdGetBindingData(ElektraIoFdOperation *fdOp)
Get private binding data from operation.
Definition: io.c:499

When your I/O management library detects a change of the file descriptor status it will call a callback supplied by your I/O binding. We will assume for file descriptor watch operations this is ioDocBindingFdCallback(). Your I/O binding's task is to call the operation callback supplied by the user with the correct arguments.

/*static*/ void ioDocBindingFdCallback (SomeIoLibHandle * handle, int bitmask)
{
// For this example let's assume handle is passed as argument
ELEKTRA_NOT_NULL (handle->data);
// Convert bitmask to Elekta's flags
}
ElektraIoFdCallback elektraIoFdGetCallback(ElektraIoFdOperation *fdOp)
Get callback of file descriptor watch operation.
Definition: io.c:543
void ioDocBindingFdCallback(SomeIoLibHandle *handle, int bitmask)
Calls the associated operation callback.
Definition: io_doc.c:320
int someBitMaskToElektraIoFlags(int bitmask)
Convert your I/O library bit mask to Elektra's I/O flags.
Definition: io_doc.c:298
struct _ElektraIoFdOperation ElektraIoFdOperation
file descriptor watch operation handle
Definition: kdbio.h:25
Example I/O management library data structure.
Definition: io_doc.c:231
void * data
Let's you access the context you supplied to the I/O management library.
Definition: io_doc.c:233

We assumed SomeIoLibHandle->data let's you access your context. Since we have used the original operation data as context we directly obtain the operation data to retrieve the user callback using elektraIoFdGetCallback(). Additionally it is necessary to convert the I/O management library's bitmask to Elekta's I/O bitmask (ElektraIoFdFlags) and then call the user callback.

When implementing ElektraIoBindingRemoveFd (or the timer and idle equivalents) make sure to free data allocated in the add functions.

Cleanup

ElektraIoBindingCleanup is the place to free data allocated for your I/O binding.

At least you need to free the pointer returned from elektraIoNewBinding() in your I/O binding's entry point.

Linking

Make sure to link against the elektra-io library for the elektraIo* utility functions that create bindings or operations and allow access to their fields. This library is available via pkg-config.

Testing

Elektra provides a test suite for I/O bindings in order to make sure that transport plugins will work with all bindings. To run the test suite you need to execute elektraIoTestSuite() and provide the necessary callbacks for creating a new binding, starting and stopping asynchronous processing (ElektraIoTestSuiteCreateBinding, ElektraIoTestSuiteStart and ElektraIoTestSuiteStop).

int main (int argc, char ** argv)
{
init (argc, argv);
elektraIoTestSuite (createBinding, startLoop, stopLoop);
print_result ("iowrapper_doc");
return nbError;
}
void elektraIoTestSuite(ElektraIoTestSuiteCreateBinding createBinding, ElektraIoTestSuiteStart start, ElektraIoTestSuiteStop stop)
Test-Suite for I/O Bindings.
int main(int argc, char **argv)
[kdbio testsuite main]
Definition: testio_doc.c:48

The functions supplied to elektraIoTestSuite() are called for setup, starting and stopping of the tests.

For example ElektraIoTestSuiteCreateBinding of the "doc" binding:

static ElektraIoInterface * createBinding (void)
{
return elektraIoDocNew ("foo");
}
ElektraIoInterface * elektraIoDocNew(char *foo)
Create and initialize a new doc I/O binding.
Definition: io_doc.c:564

Of course starting and stopping is specific to your I/O management library.