71 lines
1.6 KiB
C++
71 lines
1.6 KiB
C++
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
#include <AP_HAL.h>
|
|
#include <AP_HAL_AVR.h>
|
|
#include "Semaphore.h"
|
|
using namespace AP_HAL_AVR;
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// Constructor
|
|
AVRSemaphore::AVRSemaphore() {}
|
|
|
|
// get - to claim ownership of the semaphore
|
|
bool AVRSemaphore::get(void* caller)
|
|
{
|
|
bool result = false;
|
|
hal.scheduler->begin_atomic();
|
|
if( !_taken ) {
|
|
_taken = true;
|
|
_owner = caller;
|
|
result = true;
|
|
}
|
|
hal.scheduler->end_atomic();
|
|
return result;
|
|
}
|
|
|
|
// release - to give up ownership of the semaphore
|
|
// returns true if successfully released
|
|
bool AVRSemaphore::release(void* caller)
|
|
{
|
|
|
|
// check legitimacy of release call
|
|
if( caller != _owner ) {
|
|
return false;
|
|
}
|
|
|
|
// if another process is waiting immediately call the provided kontinuation
|
|
if( _waiting_k != NULL ) {
|
|
|
|
// give ownership to waiting process
|
|
_owner = _waiting_owner;
|
|
AP_HAL::Proc k = _waiting_k;
|
|
|
|
// clear waiting process
|
|
_waiting_k = NULL;
|
|
_waiting_owner = NULL;
|
|
|
|
// callback
|
|
k();
|
|
}
|
|
|
|
// give up the semaphore
|
|
_taken = false;
|
|
return true;
|
|
}
|
|
|
|
// call_on_release - returns true if caller successfully added to the queue to
|
|
// be called back
|
|
bool AVRSemaphore::call_on_release(void* caller, AP_HAL::Proc k)
|
|
{
|
|
bool result = false;
|
|
hal.scheduler->begin_atomic();
|
|
if( _waiting_owner == NULL ) {
|
|
_waiting_owner = caller;
|
|
_waiting_k = k;
|
|
result = true;
|
|
}
|
|
hal.scheduler->end_atomic();
|
|
return result;
|
|
}
|