/* spi.h Copyright 2021 Carl John Kugler III Licensed under the Apache License, Version 2.0 (the License); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an AS IS BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #pragma once #include #include #include // // Pico includes #include "pico/stdlib.h" #include "pico/mutex.h" #include "pico/types.h" // #include "hardware/dma.h" #include "hardware/gpio.h" #include "hardware/irq.h" #include "hardware/spi.h" // #include "my_debug.h" #include "sd_timeouts.h" #ifdef __cplusplus extern "C" { #endif #define SPI_FILL_CHAR (0xFF) // "Class" representing SPIs typedef struct spi_t { spi_inst_t *hw_inst; // SPI HW uint miso_gpio; // SPI MISO GPIO number (not pin number) uint mosi_gpio; uint sck_gpio; uint baud_rate; /* The different modes of the Motorola SPI protocol are: - Mode 0: When CPOL and CPHA are both 0, data sampled at the leading rising edge of the clock pulse and shifted out on the falling edge. This is the most common mode for SPI bus communication. - Mode 1: When CPOL is 0 and CPHA is 1, data sampled at the trailing falling edge and shifted out on the rising edge. - Mode 2: When CPOL is 1 and CPHA is 0, data sampled at the leading falling edge and shifted out on the rising edge. - Mode 3: When CPOL is 1 and CPHA is 1, data sampled at the trailing rising edge and shifted out on the falling edge. */ uint spi_mode; bool no_miso_gpio_pull_up; /* Drive strength levels for GPIO outputs: GPIO_DRIVE_STRENGTH_2MA, GPIO_DRIVE_STRENGTH_4MA, GPIO_DRIVE_STRENGTH_8MA, GPIO_DRIVE_STRENGTH_12MA */ bool set_drive_strength; enum gpio_drive_strength mosi_gpio_drive_strength; enum gpio_drive_strength sck_gpio_drive_strength; bool use_static_dma_channels; uint tx_dma; uint rx_dma; /* The following fields are not part of the configuration. They are dynamically assigned. */ dma_channel_config tx_dma_cfg; dma_channel_config rx_dma_cfg; mutex_t mutex; bool initialized; } spi_t; void spi_transfer_start(spi_t *spi_p, const uint8_t *tx, uint8_t *rx, size_t length); uint32_t calculate_transfer_time_ms(spi_t *spi_p, uint32_t bytes); bool spi_transfer_wait_complete(spi_t *spi_p, uint32_t timeout_ms); bool spi_transfer(spi_t *spi_p, const uint8_t *tx, uint8_t *rx, size_t length); bool my_spi_init(spi_t *spi_p); static inline void spi_lock(spi_t *spi_p) { myASSERT(mutex_is_initialized(&spi_p->mutex)); mutex_enter_blocking(&spi_p->mutex); } static inline void spi_unlock(spi_t *spi_p) { myASSERT(mutex_is_initialized(&spi_p->mutex)); mutex_exit(&spi_p->mutex); } /* This uses the Pico LED to show SD card activity. You can use it to get a rough idea of utilization. Warning: Pico W uses GPIO 25 for SPI communication to the CYW43439. You can enable this by putting something like add_compile_definitions(USE_LED=1) in CMakeLists.txt, for example. */ #if !defined(NO_PICO_LED) && defined(USE_LED) && USE_LED && defined(PICO_DEFAULT_LED_PIN) # define LED_INIT() \ { \ gpio_init(PICO_DEFAULT_LED_PIN); \ gpio_set_dir(PICO_DEFAULT_LED_PIN, GPIO_OUT); \ } # define LED_ON() gpio_put(PICO_DEFAULT_LED_PIN, 1) # define LED_OFF() gpio_put(PICO_DEFAULT_LED_PIN, 0) #else # define LED_ON() # define LED_OFF() # define LED_INIT() #endif #ifdef __cplusplus } #endif /* [] END OF FILE */