/*
|
/*
|
* Mutex Handler
|
* Mutex Handler
|
*
|
*
|
* DESCRIPTION:
|
* DESCRIPTION:
|
*
|
*
|
* This package is the implementation of the Mutex Handler.
|
* This package is the implementation of the Mutex Handler.
|
* This handler provides synchronization and mutual exclusion capabilities.
|
* This handler provides synchronization and mutual exclusion capabilities.
|
*
|
*
|
* COPYRIGHT (c) 1989-1999.
|
* COPYRIGHT (c) 1989-1999.
|
* On-Line Applications Research Corporation (OAR).
|
* On-Line Applications Research Corporation (OAR).
|
*
|
*
|
* The license and distribution terms for this file may be
|
* The license and distribution terms for this file may be
|
* found in the file LICENSE in this distribution or at
|
* found in the file LICENSE in this distribution or at
|
* http://www.OARcorp.com/rtems/license.html.
|
* http://www.OARcorp.com/rtems/license.html.
|
*
|
*
|
* coremutexseize.c,v 1.7 2001/08/30 18:33:57 joel Exp
|
* coremutexseize.c,v 1.7 2001/08/30 18:33:57 joel Exp
|
*/
|
*/
|
|
|
#include <rtems/system.h>
|
#include <rtems/system.h>
|
#include <rtems/score/isr.h>
|
#include <rtems/score/isr.h>
|
#include <rtems/score/coremutex.h>
|
#include <rtems/score/coremutex.h>
|
#include <rtems/score/states.h>
|
#include <rtems/score/states.h>
|
#include <rtems/score/thread.h>
|
#include <rtems/score/thread.h>
|
#include <rtems/score/threadq.h>
|
#include <rtems/score/threadq.h>
|
|
|
/*PAGE
|
/*PAGE
|
*
|
*
|
* _CORE_mutex_Seize (interrupt blocking support)
|
* _CORE_mutex_Seize (interrupt blocking support)
|
*
|
*
|
* This routine blocks the caller thread after an attempt attempts to obtain
|
* This routine blocks the caller thread after an attempt attempts to obtain
|
* the specified mutex has failed.
|
* the specified mutex has failed.
|
*
|
*
|
* Input parameters:
|
* Input parameters:
|
* the_mutex - pointer to mutex control block
|
* the_mutex - pointer to mutex control block
|
* timeout - number of ticks to wait (0 means forever)
|
* timeout - number of ticks to wait (0 means forever)
|
*/
|
*/
|
|
|
void _CORE_mutex_Seize_interrupt_blocking(
|
void _CORE_mutex_Seize_interrupt_blocking(
|
CORE_mutex_Control *the_mutex,
|
CORE_mutex_Control *the_mutex,
|
Watchdog_Interval timeout
|
Watchdog_Interval timeout
|
)
|
)
|
{
|
{
|
Thread_Control *executing;
|
Thread_Control *executing;
|
|
|
executing = _Thread_Executing;
|
executing = _Thread_Executing;
|
if ( _CORE_mutex_Is_inherit_priority( &the_mutex->Attributes ) ) {
|
if ( _CORE_mutex_Is_inherit_priority( &the_mutex->Attributes ) ) {
|
if ( the_mutex->holder->current_priority > executing->current_priority ) {
|
if ( the_mutex->holder->current_priority > executing->current_priority ) {
|
_Thread_Change_priority(
|
_Thread_Change_priority(
|
the_mutex->holder,
|
the_mutex->holder,
|
executing->current_priority,
|
executing->current_priority,
|
FALSE
|
FALSE
|
);
|
);
|
}
|
}
|
}
|
}
|
|
|
the_mutex->blocked_count++;
|
the_mutex->blocked_count++;
|
_Thread_queue_Enqueue( &the_mutex->Wait_queue, timeout );
|
_Thread_queue_Enqueue( &the_mutex->Wait_queue, timeout );
|
|
|
if ( _Thread_Executing->Wait.return_code == CORE_MUTEX_STATUS_SUCCESSFUL ) {
|
if ( _Thread_Executing->Wait.return_code == CORE_MUTEX_STATUS_SUCCESSFUL ) {
|
/*
|
/*
|
* if CORE_MUTEX_DISCIPLINES_PRIORITY_INHERIT then nothing to do
|
* if CORE_MUTEX_DISCIPLINES_PRIORITY_INHERIT then nothing to do
|
* because this task is already the highest priority.
|
* because this task is already the highest priority.
|
*/
|
*/
|
|
|
if ( _CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) ) {
|
if ( _CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) ) {
|
if (the_mutex->Attributes.priority_ceiling < executing->current_priority){
|
if (the_mutex->Attributes.priority_ceiling < executing->current_priority){
|
_Thread_Change_priority(
|
_Thread_Change_priority(
|
executing,
|
executing,
|
the_mutex->Attributes.priority_ceiling,
|
the_mutex->Attributes.priority_ceiling,
|
FALSE
|
FALSE
|
);
|
);
|
}
|
}
|
}
|
}
|
}
|
}
|
_Thread_Enable_dispatch();
|
_Thread_Enable_dispatch();
|
}
|
}
|
|
|
#if !defined(USE_INLINES)
|
#if !defined(USE_INLINES)
|
int _CORE_mutex_Seize_interrupt_trylock(
|
int _CORE_mutex_Seize_interrupt_trylock(
|
CORE_mutex_Control *the_mutex,
|
CORE_mutex_Control *the_mutex,
|
ISR_Level *level_p
|
ISR_Level *level_p
|
)
|
)
|
{
|
{
|
Thread_Control *executing;
|
Thread_Control *executing;
|
ISR_Level level = *level_p;
|
ISR_Level level = *level_p;
|
|
|
/* disabled when you get here */
|
/* disabled when you get here */
|
|
|
executing = _Thread_Executing;
|
executing = _Thread_Executing;
|
executing->Wait.return_code = CORE_MUTEX_STATUS_SUCCESSFUL;
|
executing->Wait.return_code = CORE_MUTEX_STATUS_SUCCESSFUL;
|
if ( !_CORE_mutex_Is_locked( the_mutex ) ) {
|
if ( !_CORE_mutex_Is_locked( the_mutex ) ) {
|
the_mutex->lock = CORE_MUTEX_LOCKED;
|
the_mutex->lock = CORE_MUTEX_LOCKED;
|
the_mutex->holder = executing;
|
the_mutex->holder = executing;
|
the_mutex->holder_id = executing->Object.id;
|
the_mutex->holder_id = executing->Object.id;
|
the_mutex->nest_count = 1;
|
the_mutex->nest_count = 1;
|
if ( _CORE_mutex_Is_inherit_priority( &the_mutex->Attributes ) ||
|
if ( _CORE_mutex_Is_inherit_priority( &the_mutex->Attributes ) ||
|
_CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) )
|
_CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) )
|
executing->resource_count++;
|
executing->resource_count++;
|
|
|
if ( !_CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) ) {
|
if ( !_CORE_mutex_Is_priority_ceiling( &the_mutex->Attributes ) ) {
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
return 0;
|
return 0;
|
}
|
}
|
/* else must be CORE_MUTEX_DISCIPLINES_PRIORITY_CEILING */
|
/* else must be CORE_MUTEX_DISCIPLINES_PRIORITY_CEILING */
|
{
|
{
|
Priority_Control ceiling;
|
Priority_Control ceiling;
|
Priority_Control current;
|
Priority_Control current;
|
|
|
ceiling = the_mutex->Attributes.priority_ceiling;
|
ceiling = the_mutex->Attributes.priority_ceiling;
|
current = executing->current_priority;
|
current = executing->current_priority;
|
if ( current == ceiling ) {
|
if ( current == ceiling ) {
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
return 0;
|
return 0;
|
}
|
}
|
if ( current > ceiling ) {
|
if ( current > ceiling ) {
|
_Thread_Disable_dispatch();
|
_Thread_Disable_dispatch();
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
_Thread_Change_priority(
|
_Thread_Change_priority(
|
the_mutex->holder,
|
the_mutex->holder,
|
the_mutex->Attributes.priority_ceiling,
|
the_mutex->Attributes.priority_ceiling,
|
FALSE
|
FALSE
|
);
|
);
|
_Thread_Enable_dispatch();
|
_Thread_Enable_dispatch();
|
return 0;
|
return 0;
|
}
|
}
|
/* if ( current < ceiling ) */ {
|
/* if ( current < ceiling ) */ {
|
executing->Wait.return_code = CORE_MUTEX_STATUS_CEILING_VIOLATED;
|
executing->Wait.return_code = CORE_MUTEX_STATUS_CEILING_VIOLATED;
|
the_mutex->nest_count = 0; /* undo locking above */
|
the_mutex->nest_count = 0; /* undo locking above */
|
executing->resource_count--; /* undo locking above */
|
executing->resource_count--; /* undo locking above */
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
return 0;
|
return 0;
|
}
|
}
|
}
|
}
|
return 0;
|
return 0;
|
}
|
}
|
|
|
if ( _Thread_Is_executing( the_mutex->holder ) ) {
|
if ( _Thread_Is_executing( the_mutex->holder ) ) {
|
switch ( the_mutex->Attributes.lock_nesting_behavior ) {
|
switch ( the_mutex->Attributes.lock_nesting_behavior ) {
|
case CORE_MUTEX_NESTING_ACQUIRES:
|
case CORE_MUTEX_NESTING_ACQUIRES:
|
the_mutex->nest_count++;
|
the_mutex->nest_count++;
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
return 0;
|
return 0;
|
case CORE_MUTEX_NESTING_IS_ERROR:
|
case CORE_MUTEX_NESTING_IS_ERROR:
|
executing->Wait.return_code = CORE_MUTEX_STATUS_NESTING_NOT_ALLOWED;
|
executing->Wait.return_code = CORE_MUTEX_STATUS_NESTING_NOT_ALLOWED;
|
_ISR_Enable( level );
|
_ISR_Enable( level );
|
return 0;
|
return 0;
|
case CORE_MUTEX_NESTING_BLOCKS:
|
case CORE_MUTEX_NESTING_BLOCKS:
|
break;
|
break;
|
}
|
}
|
}
|
}
|
|
|
return 1;
|
return 1;
|
}
|
}
|
#endif
|
#endif
|
|
|