/* * This file is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This file is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #include "Device.h" #include #include #include #if HAL_USE_I2C == TRUE || HAL_USE_SPI == TRUE #include "Scheduler.h" #include "Semaphores.h" #include "Util.h" #include "hwdef/common/stm32_util.h" #ifndef HAL_DEVICE_THREAD_STACK #define HAL_DEVICE_THREAD_STACK 1024 #endif using namespace ChibiOS; extern const AP_HAL::HAL& hal; DeviceBus::DeviceBus(uint8_t _thread_priority) : thread_priority(_thread_priority) { bouncebuffer_init(&bounce_buffer_tx, 10, false); bouncebuffer_init(&bounce_buffer_rx, 10, false); } /* per-bus callback thread */ void DeviceBus::bus_thread(void *arg) { struct DeviceBus *binfo = (struct DeviceBus *)arg; while (true) { uint64_t now = AP_HAL::micros64(); DeviceBus::callback_info *callback; // find a callback to run for (callback = binfo->callbacks; callback; callback = callback->next) { if (now >= callback->next_usec) { while (now >= callback->next_usec) { callback->next_usec += callback->period_usec; } // call it with semaphore held if (binfo->semaphore.take(HAL_SEMAPHORE_BLOCK_FOREVER)) { callback->cb(); binfo->semaphore.give(); } } } // work out when next loop is needed uint64_t next_needed = 0; now = AP_HAL::micros64(); for (callback = binfo->callbacks; callback; callback = callback->next) { if (next_needed == 0 || callback->next_usec < next_needed) { next_needed = callback->next_usec; if (next_needed < now) { next_needed = now; } } } // delay for at most 50ms, to handle newly added callbacks uint32_t delay = 50000; if (next_needed >= now && next_needed - now < delay) { delay = next_needed - now; } // don't delay for less than 100usec, so one thread doesn't // completely dominate the CPU if (delay < 100) { delay = 100; } hal.scheduler->delay_microseconds(delay); } return; } #if CH_CFG_USE_HEAP == TRUE AP_HAL::Device::PeriodicHandle DeviceBus::register_periodic_callback(uint32_t period_usec, AP_HAL::Device::PeriodicCb cb, AP_HAL::Device *_hal_device) { if (!thread_started) { thread_started = true; hal_device = _hal_device; // setup a name for the thread const uint8_t name_len = 7; char *name = (char *)malloc(name_len); switch (hal_device->bus_type()) { case AP_HAL::Device::BUS_TYPE_I2C: snprintf(name, name_len, "I2C:%u", hal_device->bus_num()); break; case AP_HAL::Device::BUS_TYPE_SPI: snprintf(name, name_len, "SPI:%u", hal_device->bus_num()); break; default: break; } thread_ctx = thread_create_alloc(THD_WORKING_AREA_SIZE(HAL_DEVICE_THREAD_STACK), name, thread_priority, /* Initial priority. */ DeviceBus::bus_thread, /* Thread function. */ this); /* Thread parameter. */ if (thread_ctx == nullptr) { AP_HAL::panic("Failed to create bus thread %s", name); } } DeviceBus::callback_info *callback = new DeviceBus::callback_info; if (callback == nullptr) { return nullptr; } callback->cb = cb; callback->period_usec = period_usec; callback->next_usec = AP_HAL::micros64() + period_usec; // add to linked list of callbacks on thread callback->next = callbacks; callbacks = callback; return callback; } #endif // CH_CFG_USE_HEAP /* * Adjust the timer for the next call: it needs to be called from the bus * thread, otherwise it will race with it */ bool DeviceBus::adjust_timer(AP_HAL::Device::PeriodicHandle h, uint32_t period_usec) { if (chThdGetSelfX() != thread_ctx) { return false; } DeviceBus::callback_info *callback = static_cast(h); callback->period_usec = period_usec; callback->next_usec = AP_HAL::micros64() + period_usec; return true; } /* setup to use DMA-safe bouncebuffers for device transfers */ bool DeviceBus::bouncebuffer_setup(const uint8_t *&buf_tx, uint16_t tx_len, uint8_t *&buf_rx, uint16_t rx_len) { if (buf_rx) { if (!bouncebuffer_setup_read(bounce_buffer_rx, &buf_rx, rx_len)) { return false; } } if (buf_tx) { if (!bouncebuffer_setup_write(bounce_buffer_tx, &buf_tx, tx_len)) { if (buf_rx) { bouncebuffer_abort(bounce_buffer_rx); } return false; } } return true; } /* complete a transfer using DMA bounce buffer */ void DeviceBus::bouncebuffer_finish(const uint8_t *buf_tx, uint8_t *buf_rx, uint16_t rx_len) { if (buf_rx) { bouncebuffer_finish_read(bounce_buffer_rx, buf_rx, rx_len); } if (buf_tx) { bouncebuffer_finish_write(bounce_buffer_tx, buf_tx); } } #endif // HAL_USE_I2C || HAL_USE_SPI