/* * 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 #include "SPIDevice.h" #include "sdcard.h" #include "bouncebuffer.h" #include "hwdef/common/spi_hook.h" #include #include #include "bouncebuffer.h" #include "stm32_util.h" #include "hwdef/common/usbcfg.h" #include "stm32_util.h" extern const AP_HAL::HAL& hal; #if HAL_HAVE_USB_CDC_MSD bool write_protected = false; static void block_filesys_access() { AP::FS().block_access(); write_protected = true; } static void free_filesys_access() { write_protected = false; AP::FS().free_access(); } #endif #ifdef USE_POSIX static FATFS SDC_FS; // FATFS object #ifndef HAL_BOOTLOADER_BUILD static HAL_Semaphore sem; #endif static bool sdcard_running; #endif #if HAL_USE_SDC static SDCConfig sdcconfig = { SDC_MODE_4BIT, 0 }; #elif HAL_USE_MMC_SPI MMCDriver MMCD1; static AP_HAL::OwnPtr device; static MMCConfig mmcconfig; static SPIConfig lowspeed; static SPIConfig highspeed; #endif #if HAL_HAVE_USB_CDC_MSD static uint8_t blkbuf[512]; static uint8_t txbuf[512]; #endif /* initialise microSD card if avaialble. This is called during AP_BoardConfig initialisation. The parameter BRD_SD_SLOWDOWN controls a scaling factor on the microSD clock */ bool sdcard_init() { #ifdef USE_POSIX #ifndef HAL_BOOTLOADER_BUILD WITH_SEMAPHORE(sem); uint8_t sd_slowdown = AP_BoardConfig::get_sdcard_slowdown(); #else uint8_t sd_slowdown = 0; // maybe take from a define? #endif #if HAL_USE_SDC #if STM32_SDC_USE_SDMMC2 == TRUE auto &sdcd = SDCD2; #else auto &sdcd = SDCD1; #endif if (sdcd.bouncebuffer == nullptr) { // allocate 4k bouncebuffer for microSD to match size in // AP_Logger bouncebuffer_init(&sdcd.bouncebuffer, 4096, true); } if (sdcard_running) { sdcard_stop(); } const uint8_t tries = 3; for (uint8_t i=0; iget_device("sdcard"); if (!device) { printf("No sdcard SPI device found\n"); sdcard_running = false; return false; } device->set_slowdown(sd_slowdown); mmcObjectInit(&MMCD1, MMCD1.buffer); mmcconfig.spip = static_cast(device.get())->get_driver(); mmcconfig.hscfg = &highspeed; mmcconfig.lscfg = &lowspeed; /* try up to 3 times to init microSD interface */ const uint8_t tries = 3; for (uint8_t i=0; iset_speed(config == &lowspeed ? AP_HAL::Device::SPEED_LOW : AP_HAL::Device::SPEED_HIGH); } void spiStopHook(SPIDriver *spip) { } __RAMFUNC__ void spiAcquireBusHook(SPIDriver *spip) { if (sdcard_running) { ChibiOS::SPIDevice *devptr = static_cast(device.get()); devptr->acquire_bus(true, true); } } __RAMFUNC__ void spiReleaseBusHook(SPIDriver *spip) { if (sdcard_running) { ChibiOS::SPIDevice *devptr = static_cast(device.get()); devptr->acquire_bus(false, true); } } __RAMFUNC__ void spiSelectHook(SPIDriver *spip) { if (sdcard_running) { device->get_semaphore()->take_blocking(); device->set_chip_select(true); } } __RAMFUNC__ void spiUnselectHook(SPIDriver *spip) { if (sdcard_running) { device->set_chip_select(false); device->get_semaphore()->give(); } } void spiIgnoreHook(SPIDriver *spip, size_t n) { if (sdcard_running) { device->clock_pulse(n); } } __RAMFUNC__ void spiSendHook(SPIDriver *spip, size_t n, const void *txbuf) { if (sdcard_running) { device->transfer((const uint8_t *)txbuf, n, nullptr, 0); } } __RAMFUNC__ void spiReceiveHook(SPIDriver *spip, size_t n, void *rxbuf) { if (sdcard_running) { device->transfer(nullptr, 0, (uint8_t *)rxbuf, n); } } #endif