AP_InertialSensor: prevent a lockup in MPU6000 driver

thanks to the VRBrain port for noticing this bug.

Failing to get the semaphore is an expected error with the MPU6000, as
we read data both from timer context and mainline code. That means
semaphore conflicts are inevitable. We shouldn't consider them an
error, and shouldn't panic when some arbitrary number of them have
happened since boot.

Instead the wait_for_sample() code checks that we receive new data at
least every 50ms. That is a much safer test.
This commit is contained in:
Andrew Tridgell 2013-09-23 22:48:36 +10:00
parent c888f0256f
commit 770b7b5901

View File

@ -419,18 +419,14 @@ void AP_InertialSensor_MPU6000::_poll_data(uint32_t now)
*/ */
void AP_InertialSensor_MPU6000::_read_data_from_timerprocess() void AP_InertialSensor_MPU6000::_read_data_from_timerprocess()
{ {
static uint8_t semfail_ctr = 0; if (!_spi_sem->take_nonblocking()) {
bool got = _spi_sem->take_nonblocking(); /*
if (!got) { the semaphore being busy is an expected condition when the
semfail_ctr++; mainline code is calling num_samples_available() which will
if (semfail_ctr > 100) { grab the semaphore. We return now and rely on the mainline
hal.scheduler->panic(PSTR("PANIC: failed to take SPI semaphore " code grabbing the latest sample.
"100 times in AP_InertialSensor_MPU6000::" */
"_read_data_from_timerprocess"));
}
return; return;
} else {
semfail_ctr = 0;
} }
_last_sample_time_micros = hal.scheduler->micros(); _last_sample_time_micros = hal.scheduler->micros();