/* SPDX-License-Identifier: GPL-2.0-only */ #ifndef __SOC_SIFIVE_HIFIVE_U_SPI_INTERNAL_H__ #define __SOC_SIFIVE_HIFIVE_U_SPI_INTERNAL_H__ #include #define _ASSERT_SIZEOF(type, size) _Static_assert( \ sizeof(type) == (size), \ #type " must be " #size " bytes wide") // Chip Select Mode Register (csmode) #define FU740_SPI_CSMODE_AUTO 0 #define FU740_SPI_CSMODE_HOLD 2 #define FU740_SPI_CSMODE_OFF 3 union fu740_spi_reg_sckmode { struct { uint32_t pha : 1; uint32_t pol : 1; uint32_t reserved : 30; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_sckmode, 4); union fu740_spi_reg_csmode { struct { uint32_t mode : 2; uint32_t reserved : 30; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_csmode, 4); union fu740_spi_reg_delay0 { struct { uint32_t cssck : 8; uint32_t reserved0 : 8; uint32_t sckcs : 8; uint32_t reserved1 : 8; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_delay0, 4); union fu740_spi_reg_delay1 { struct { uint32_t intercs : 8; uint32_t reserved0 : 8; uint32_t interxfr : 8; uint32_t reserved1 : 8; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_delay1, 4); union fu740_spi_reg_fmt { struct { uint32_t proto : 2; uint32_t endian : 1; uint32_t dir : 1; uint32_t reserved0 : 12; uint32_t len : 4; uint32_t reserved1 : 12; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_fmt, 4); union fu740_spi_reg_txdata { struct { uint32_t data : 8; uint32_t reserved : 23; uint32_t full : 1; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_txdata, 4); union fu740_spi_reg_rxdata { struct { uint32_t data : 8; uint32_t reserved : 23; uint32_t empty : 1; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_rxdata, 4); union fu740_spi_reg_txmark { struct { uint32_t txmark : 3; uint32_t reserved : 29; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_txmark, 4); union fu740_spi_reg_rxmark { struct { uint32_t rxmark : 3; uint32_t reserved : 29; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_rxmark, 4); union fu740_spi_reg_fctrl { struct { uint32_t en : 1; uint32_t reserved : 31; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_fctrl, 4); union fu740_spi_reg_ffmt { struct { uint32_t cmd_en : 1; uint32_t addr_len : 3; uint32_t pad_cnt : 4; uint32_t cmd_proto : 2; uint32_t addr_proto : 2; uint32_t data_proto : 2; uint32_t reserved : 2; uint32_t cmd_code : 8; uint32_t pad_code : 8; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_ffmt, 4); union fu740_spi_reg_ie { struct { uint32_t txwm : 1; uint32_t rxwm : 1; uint32_t reserved : 30; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_ie, 4); union fu740_spi_reg_ip { struct { uint32_t txwm : 1; uint32_t rxwm : 1; uint32_t reserved : 30; }; uint32_t raw_bits; }; _ASSERT_SIZEOF(union fu740_spi_reg_ip, 4); #undef _ASSERT_SIZEOF /** * SPI control register memory map. * * All functions take a pointer to a SPI device's control registers. */ struct fu740_spi_ctrl { uint32_t sckdiv; union fu740_spi_reg_sckmode sckmode; uint32_t reserved08; uint32_t reserved0c; uint32_t csid; uint32_t csdef; union fu740_spi_reg_csmode csmode; uint32_t reserved1c; uint32_t reserved20; uint32_t reserved24; union fu740_spi_reg_delay0 delay0; union fu740_spi_reg_delay1 delay1; uint32_t reserved30; uint32_t reserved34; uint32_t reserved38; uint32_t reserved3c; union fu740_spi_reg_fmt fmt; uint32_t reserved44; union fu740_spi_reg_txdata txdata; union fu740_spi_reg_rxdata rxdata; union fu740_spi_reg_txmark txmark; union fu740_spi_reg_rxmark rxmark; uint32_t reserved58; uint32_t reserved5c; union fu740_spi_reg_fctrl fctrl; union fu740_spi_reg_ffmt ffmt; uint32_t reserved68; uint32_t reserved6c; union fu740_spi_reg_ie ie; union fu740_spi_reg_ip ip; }; /** * Get smallest clock divisor that divides input_khz to a quotient less than or * equal to max_target_khz; */ static inline unsigned int fu740_spi_min_clk_divisor(unsigned int input_khz, unsigned int max_target_khz) { // f_sck = f_in / (2 * (div + 1)) => div = (f_in / (2*f_sck)) - 1 // // The nearest integer solution for div requires rounding up as to not // exceed max_target_khz. // // div = ceil(f_in / (2*f_sck)) - 1 // = floor((f_in - 1 + 2*f_sck) / (2*f_sck)) - 1 // // This should not overflow as long as (f_in - 1 + 2*f_sck) does not // exceed 2^32 - 1, which is unlikely since we represent frequencies // in kHz. unsigned int quotient = (input_khz + 2 * max_target_khz - 1) / (2 * max_target_khz); // Avoid underflow if (quotient == 0) return 0; return quotient - 1; } #endif /* __SOC_SIFIVE_HIFIVE_U_SPI_INTERNAL_H__ */