Xenomai API  2.5.6.1
Synchronisation Services
Collaboration diagram for Synchronisation Services:

Functions

int rtdm_select_bind (int fd, rtdm_selector_t *selector, enum rtdm_selecttype type, unsigned fd_index)
 Bind a selector to specified event types of a given file descriptor.

RTDM_SELECTTYPE_xxx

Event types select can bind to

enum  rtdm_selecttype { RTDM_SELECTTYPE_READ = XNSELECT_READ, RTDM_SELECTTYPE_WRITE = XNSELECT_WRITE, RTDM_SELECTTYPE_EXCEPT = XNSELECT_EXCEPT }

Global Lock across Scheduler Invocation

#define RTDM_EXECUTE_ATOMICALLY(code_block)
 Execute code block atomically.

Spinlock with Preemption Deactivation

typedef rthal_spinlock_t rtdm_lock_t
 Lock variable.
typedef unsigned long rtdm_lockctx_t
 Variable to save the context while holding a lock.
#define RTDM_LOCK_UNLOCKED   RTHAL_SPIN_LOCK_UNLOCKED
 Static lock initialisation.
#define rtdm_lock_init(lock)   rthal_spin_lock_init(lock)
 Dynamic lock initialisation.
#define rtdm_lock_get(lock)   rthal_spin_lock(lock)
 Acquire lock from non-preemptible contexts.
#define rtdm_lock_put(lock)   rthal_spin_unlock(lock)
 Release lock without preemption restoration.
#define rtdm_lock_get_irqsave(lock, context)   rthal_spin_lock_irqsave(lock, context)
 Acquire lock and disable preemption.
#define rtdm_lock_put_irqrestore(lock, context)   rthal_spin_unlock_irqrestore(lock, context)
 Release lock and restore preemption state.
#define rtdm_lock_irqsave(context)   rthal_local_irq_save(context)
 Disable preemption locally.
#define rtdm_lock_irqrestore(context)   rthal_local_irq_restore(context)
 Restore preemption state.

Timeout Sequence Management

void rtdm_toseq_init (rtdm_toseq_t *timeout_seq, nanosecs_rel_t timeout)
 Initialise a timeout sequence.

Event Services

void rtdm_event_init (rtdm_event_t *event, unsigned long pending)
 Initialise an event.
void rtdm_event_destroy (rtdm_event_t *event)
 Destroy an event.
void rtdm_event_pulse (rtdm_event_t *event)
 Signal an event occurrence to currently listening waiters.
void rtdm_event_signal (rtdm_event_t *event)
 Signal an event occurrence.
int rtdm_event_wait (rtdm_event_t *event)
 Wait on event occurrence.
int rtdm_event_timedwait (rtdm_event_t *event, nanosecs_rel_t timeout, rtdm_toseq_t *timeout_seq)
 Wait on event occurrence with timeout.
void rtdm_event_clear (rtdm_event_t *event)
 Clear event state.
int rtdm_event_select_bind (rtdm_event_t *event, rtdm_selector_t *selector, enum rtdm_selecttype type, unsigned fd_index)
 Bind a selector to an event.

Semaphore Services

void rtdm_sem_init (rtdm_sem_t *sem, unsigned long value)
 Initialise a semaphore.
void rtdm_sem_destroy (rtdm_sem_t *sem)
 Destroy a semaphore.
int rtdm_sem_down (rtdm_sem_t *sem)
 Decrement a semaphore.
int rtdm_sem_timeddown (rtdm_sem_t *sem, nanosecs_rel_t timeout, rtdm_toseq_t *timeout_seq)
 Decrement a semaphore with timeout.
void rtdm_sem_up (rtdm_sem_t *sem)
 Increment a semaphore.
int rtdm_sem_select_bind (rtdm_sem_t *sem, rtdm_selector_t *selector, enum rtdm_selecttype type, unsigned fd_index)
 Bind a selector to a semaphore.

Mutex Services

void rtdm_mutex_init (rtdm_mutex_t *mutex)
 Initialise a mutex.
void rtdm_mutex_destroy (rtdm_mutex_t *mutex)
 Destroy a mutex.
void rtdm_mutex_unlock (rtdm_mutex_t *mutex)
 Release a mutex.
int rtdm_mutex_lock (rtdm_mutex_t *mutex)
 Request a mutex.
int rtdm_mutex_timedlock (rtdm_mutex_t *mutex, nanosecs_rel_t timeout, rtdm_toseq_t *timeout_seq)
 Request a mutex with timeout.

Define Documentation

#define RTDM_EXECUTE_ATOMICALLY (   code_block)
Value:
{                                               \
        <ENTER_ATOMIC_SECTION>                  \
        code_block;                             \
        <LEAVE_ATOMIC_SECTION>                  \
}

Execute code block atomically.

Generally, it is illegal to suspend the current task by calling rtdm_task_sleep(), rtdm_event_wait(), etc. while holding a spinlock. In contrast, this macro allows to combine several operations including a potentially rescheduling call to an atomic code block with respect to other RTDM_EXECUTE_ATOMICALLY() blocks. The macro is a light-weight alternative for protecting code blocks via mutexes, and it can even be used to synchronise real-time and non-real-time contexts.

Parameters:
code_blockCommands to be executed atomically
Note:
It is not allowed to leave the code block explicitly by using break, return, goto, etc. This would leave the global lock held during the code block execution in an inconsistent state. Moreover, do not embed complex operations into the code bock. Consider that they will be executed under preemption lock with interrupts switched-off. Also note that invocation of rescheduling calls may break the atomicity until the task gains the CPU again.

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible, depends on functions called within code_block.

#define rtdm_lock_get (   lock)    rthal_spin_lock(lock)

Acquire lock from non-preemptible contexts.

Parameters:
lockAddress of lock variable

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

#define rtdm_lock_get_irqsave (   lock,
  context 
)    rthal_spin_lock_irqsave(lock, context)

Acquire lock and disable preemption.

Parameters:
lockAddress of lock variable
contextname of local variable to store the context in

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

#define rtdm_lock_init (   lock)    rthal_spin_lock_init(lock)

Dynamic lock initialisation.

Parameters:
lockAddress of lock variable

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

#define rtdm_lock_irqrestore (   context)    rthal_local_irq_restore(context)

Restore preemption state.

Parameters:
contextname of local variable which stored the context

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

#define rtdm_lock_irqsave (   context)    rthal_local_irq_save(context)

Disable preemption locally.

Parameters:
contextname of local variable to store the context in

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

#define rtdm_lock_put (   lock)    rthal_spin_unlock(lock)

Release lock without preemption restoration.

Parameters:
lockAddress of lock variable

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

#define rtdm_lock_put_irqrestore (   lock,
  context 
)    rthal_spin_unlock_irqrestore(lock, context)

Release lock and restore preemption state.

Parameters:
lockAddress of lock variable
contextname of local variable which stored the context

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.


Enumeration Type Documentation

Enumerator:
RTDM_SELECTTYPE_READ 

Select input data availability events.

RTDM_SELECTTYPE_WRITE 

Select ouput buffer availability events.

RTDM_SELECTTYPE_EXCEPT 

Select exceptional events.


Function Documentation

void rtdm_event_clear ( rtdm_event_t *  event)

Clear event state.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

void rtdm_event_destroy ( rtdm_event_t *  event)

Destroy an event.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

void rtdm_event_init ( rtdm_event_t *  event,
unsigned long  pending 
)

Initialise an event.

Parameters:
[in,out]eventEvent handle
[in]pendingNon-zero if event shall be initialised as set, 0 otherwise

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References xnselect_init(), and xnsynch_init().

void rtdm_event_pulse ( rtdm_event_t *  event)

Signal an event occurrence to currently listening waiters.

This function wakes up all current waiters of the given event, but it does not change the event state. Subsequently callers of rtdm_event_wait() or rtdm_event_timedwait() will therefore be blocked first.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

int rtdm_event_select_bind ( rtdm_event_t *  event,
rtdm_selector_t *  selector,
enum rtdm_selecttype  type,
unsigned  fd_index 
)

Bind a selector to an event.

This functions binds the given selector to an event so that the former is notified when the event state changes. Typically the select binding handler will invoke this service.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()
[in,out]selectorSelector as passed to the select binding handler
[in]typeType of the bound event as passed to the select binding handler
[in]fd_indexFile descriptor index as passed to the select binding handler
Returns:
0 on success, otherwise:
  • -ENOMEM is returned if there is insufficient memory to establish the dynamic binding.
  • -EINVAL is returned if type or fd_index are invalid.

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References xnselect_bind().

void rtdm_event_signal ( rtdm_event_t *  event)

Signal an event occurrence.

This function sets the given event and wakes up all current waiters. If no waiter is presently registered, the next call to rtdm_event_wait() or rtdm_event_timedwait() will return immediately.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

References xnpod_schedule(), and xnsynch_flush().

int rtdm_event_timedwait ( rtdm_event_t *  event,
nanosecs_rel_t  timeout,
rtdm_toseq_t *  timeout_seq 
)

Wait on event occurrence with timeout.

This function waits or tests for the occurence of the given event, taking the provided timeout into account. On successful return, the event is reset.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()
[in]timeoutRelative timeout in nanoseconds, see RTDM_TIMEOUT_xxx for special values
[in,out]timeout_seqHandle of a timeout sequence as returned by rtdm_toseq_init() or NULL
Returns:
0 on success, otherwise:
  • -ETIMEDOUT is returned if the if the request has not been satisfied within the specified amount of time.
  • -EINTR is returned if calling task has been unblock by a signal or explicitly via rtdm_task_unblock().
  • -EIDRM is returned if event has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.
  • -EWOULDBLOCK is returned if a negative timeout (i.e., non-blocking operation) has been specified.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References XNBREAK, XNRMID, xnsynch_sleep_on(), and XNTIMEO.

Referenced by rtdm_event_wait().

int rtdm_event_wait ( rtdm_event_t *  event)

Wait on event occurrence.

This is the light-weight version of rtdm_event_timedwait(), implying an infinite timeout.

Parameters:
[in,out]eventEvent handle as returned by rtdm_event_init()
Returns:
0 on success, otherwise:
  • -EINTR is returned if calling task has been unblock by a signal or explicitly via rtdm_task_unblock().
  • -EIDRM is returned if event has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References rtdm_event_timedwait().

void rtdm_mutex_destroy ( rtdm_mutex_t *  mutex)

Destroy a mutex.

Parameters:
[in,out]mutexMutex handle as returned by rtdm_mutex_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

void rtdm_mutex_init ( rtdm_mutex_t *  mutex)

Initialise a mutex.

This function initalises a basic mutex with priority inversion protection. "Basic", as it does not allow a mutex owner to recursively lock the same mutex again.

Parameters:
[in,out]mutexMutex handle

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References xnsynch_init().

int rtdm_mutex_lock ( rtdm_mutex_t *  mutex)

Request a mutex.

This is the light-weight version of rtdm_mutex_timedlock(), implying an infinite timeout.

Parameters:
[in,out]mutexMutex handle as returned by rtdm_mutex_init()
Returns:
0 on success, otherwise:
  • -EIDRM is returned if mutex has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References rtdm_mutex_timedlock().

int rtdm_mutex_timedlock ( rtdm_mutex_t *  mutex,
nanosecs_rel_t  timeout,
rtdm_toseq_t *  timeout_seq 
)

Request a mutex with timeout.

This function tries to acquire the given mutex. If it is not available, the caller is blocked unless non-blocking operation was selected.

Parameters:
[in,out]mutexMutex handle as returned by rtdm_mutex_init()
[in]timeoutRelative timeout in nanoseconds, see RTDM_TIMEOUT_xxx for special values
[in,out]timeout_seqHandle of a timeout sequence as returned by rtdm_toseq_init() or NULL
Returns:
0 on success, otherwise:
  • -ETIMEDOUT is returned if the if the request has not been satisfied within the specified amount of time.
  • -EWOULDBLOCK is returned if timeout is negative and the semaphore value is currently not positive.
  • -EIDRM is returned if mutex has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References XNBREAK, XNRMID, xnsynch_acquire(), and XNTIMEO.

Referenced by rtdm_mutex_lock().

void rtdm_mutex_unlock ( rtdm_mutex_t *  mutex)

Release a mutex.

This function releases the given mutex, waking up a potential waiter which was blocked upon rtdm_mutex_lock() or rtdm_mutex_timedlock().

Parameters:
[in,out]mutexMutex handle as returned by rtdm_mutex_init()

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

int rtdm_select_bind ( int  fd,
rtdm_selector_t *  selector,
enum rtdm_selecttype  type,
unsigned  fd_index 
)

Bind a selector to specified event types of a given file descriptor.

This function is invoked by higher RTOS layers implementing select-like services. It shall not be called directly by RTDM drivers.

Parameters:
[in]fdFile descriptor to bind to
[in,out]selectorSelector object that shall be bound to the given event
[in]typeEvent type the caller is interested in
[in]fd_indexIndex in the file descriptor set of the caller
Returns:
0 on success, otherwise:
  • -EBADF is returned if the file descriptor fd cannot be resolved.
  • -EINVAL is returned if type or fd_index are invalid.

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References rtdm_dev_context::ops, rtdm_context_get(), rtdm_context_unlock(), and rtdm_operations::select_bind.

void rtdm_sem_destroy ( rtdm_sem_t *  sem)

Destroy a semaphore.

Parameters:
[in,out]semSemaphore handle as returned by rtdm_sem_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

int rtdm_sem_down ( rtdm_sem_t *  sem)

Decrement a semaphore.

This is the light-weight version of rtdm_sem_timeddown(), implying an infinite timeout.

Parameters:
[in,out]semSemaphore handle as returned by rtdm_sem_init()
Returns:
0 on success, otherwise:
  • -EINTR is returned if calling task has been unblock by a signal or explicitly via rtdm_task_unblock().
  • -EIDRM is returned if sem has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References rtdm_sem_timeddown().

void rtdm_sem_init ( rtdm_sem_t *  sem,
unsigned long  value 
)

Initialise a semaphore.

Parameters:
[in,out]semSemaphore handle
[in]valueInitial value of the semaphore

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References sem_init(), xnselect_init(), and xnsynch_init().

int rtdm_sem_select_bind ( rtdm_sem_t *  sem,
rtdm_selector_t *  selector,
enum rtdm_selecttype  type,
unsigned  fd_index 
)

Bind a selector to a semaphore.

This functions binds the given selector to the semaphore so that the former is notified when the semaphore state changes. Typically the select binding handler will invoke this service.

Parameters:
[in,out]semSemaphore handle as returned by rtdm_sem_init()
[in,out]selectorSelector as passed to the select binding handler
[in]typeType of the bound event as passed to the select binding handler
[in]fd_indexFile descriptor index as passed to the select binding handler
Returns:
0 on success, otherwise:
  • -ENOMEM is returned if there is insufficient memory to establish the dynamic binding.
  • -EINVAL is returned if type or fd_index are invalid.

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: never.

References xnselect_bind().

int rtdm_sem_timeddown ( rtdm_sem_t *  sem,
nanosecs_rel_t  timeout,
rtdm_toseq_t *  timeout_seq 
)

Decrement a semaphore with timeout.

This function tries to decrement the given semphore's value if it is positive on entry. If not, the caller is blocked unless non-blocking operation was selected.

Parameters:
[in,out]semSemaphore handle as returned by rtdm_sem_init()
[in]timeoutRelative timeout in nanoseconds, see RTDM_TIMEOUT_xxx for special values
[in,out]timeout_seqHandle of a timeout sequence as returned by rtdm_toseq_init() or NULL
Returns:
0 on success, otherwise:
  • -ETIMEDOUT is returned if the if the request has not been satisfied within the specified amount of time.
  • -EWOULDBLOCK is returned if timeout is negative and the semaphore value is currently not positive.
  • -EINTR is returned if calling task has been unblock by a signal or explicitly via rtdm_task_unblock().
  • -EIDRM is returned if sem has been destroyed.
  • -EPERM may be returned if an illegal invocation environment is detected.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: possible.

References sem_timedwait(), XNBREAK, XNRMID, xnsynch_sleep_on(), and XNTIMEO.

Referenced by rtdm_sem_down().

void rtdm_sem_up ( rtdm_sem_t *  sem)

Increment a semaphore.

This function increments the given semphore's value, waking up a potential waiter which was blocked upon rtdm_sem_down().

Parameters:
[in,out]semSemaphore handle as returned by rtdm_sem_init()

Environments:

This service can be called from:

  • Kernel module initialization/cleanup code
  • Interrupt service routine
  • Kernel-based task
  • User-space task (RT, non-RT)

Rescheduling: possible.

References xnpod_schedule(), and xnsynch_wakeup_one_sleeper().

void rtdm_toseq_init ( rtdm_toseq_t *  timeout_seq,
nanosecs_rel_t  timeout 
)

Initialise a timeout sequence.

This service initialises a timeout sequence handle according to the given timeout value. Timeout sequences allow to maintain a continuous timeout across multiple calls of blocking synchronisation services. A typical application scenario is given below.

Parameters:
[in,out]timeout_seqTimeout sequence handle
[in]timeoutRelative timeout in nanoseconds, see RTDM_TIMEOUT_xxx for special values

Application Scenario:

int device_service_routine(...)
{
        rtdm_toseq_t timeout_seq;
        ...

        rtdm_toseq_init(&timeout_seq, timeout);
        ...
        while (received < requested) {
                ret = rtdm_event_timedwait(&data_available, timeout, &timeout_seq);
                if (ret < 0) // including -ETIMEDOUT
                        break;

                // receive some data
                ...
        }
        ...
}

Using a timeout sequence in such a scenario avoids that the user-provided relative timeout is restarted on every call to rtdm_event_timedwait(), potentially causing an overall delay that is larger than specified by timeout. Moreover, all functions supporting timeout sequences also interpret special timeout values (infinite and non-blocking), disburdening the driver developer from handling them separately.

Environments:

This service can be called from:

  • Kernel-based task
  • User-space task (RT)

Rescheduling: never.

 All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Defines