2019-05-29 07:18:02 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2014-07-01 09:03:59 +08:00
/*
* Copyright ( c ) 2014 , Fuzhou Rockchip Electronics Co . , Ltd
2014-07-11 10:07:56 +08:00
* Author : Addy Ke < addy . ke @ rock - chips . com >
2014-07-01 09:03:59 +08:00
*/
# include <linux/clk.h>
2016-03-10 14:51:48 +08:00
# include <linux/dmaengine.h>
2019-05-14 15:41:12 -07:00
# include <linux/interrupt.h>
2016-03-10 14:51:48 +08:00
# include <linux/module.h>
# include <linux/of.h>
2016-12-16 16:59:16 -08:00
# include <linux/pinctrl/consumer.h>
2014-07-01 09:03:59 +08:00
# include <linux/platform_device.h>
# include <linux/spi/spi.h>
# include <linux/pm_runtime.h>
2016-03-10 14:51:48 +08:00
# include <linux/scatterlist.h>
2014-07-01 09:03:59 +08:00
# define DRIVER_NAME "rockchip-spi"
2017-06-28 12:38:43 +08:00
# define ROCKCHIP_SPI_CLR_BITS(reg, bits) \
writel_relaxed ( readl_relaxed ( reg ) & ~ ( bits ) , reg )
# define ROCKCHIP_SPI_SET_BITS(reg, bits) \
writel_relaxed ( readl_relaxed ( reg ) | ( bits ) , reg )
2014-07-01 09:03:59 +08:00
/* SPI register offsets */
# define ROCKCHIP_SPI_CTRLR0 0x0000
# define ROCKCHIP_SPI_CTRLR1 0x0004
# define ROCKCHIP_SPI_SSIENR 0x0008
# define ROCKCHIP_SPI_SER 0x000c
# define ROCKCHIP_SPI_BAUDR 0x0010
# define ROCKCHIP_SPI_TXFTLR 0x0014
# define ROCKCHIP_SPI_RXFTLR 0x0018
# define ROCKCHIP_SPI_TXFLR 0x001c
# define ROCKCHIP_SPI_RXFLR 0x0020
# define ROCKCHIP_SPI_SR 0x0024
# define ROCKCHIP_SPI_IPR 0x0028
# define ROCKCHIP_SPI_IMR 0x002c
# define ROCKCHIP_SPI_ISR 0x0030
# define ROCKCHIP_SPI_RISR 0x0034
# define ROCKCHIP_SPI_ICR 0x0038
# define ROCKCHIP_SPI_DMACR 0x003c
2020-07-23 08:43:55 +08:00
# define ROCKCHIP_SPI_DMATDLR 0x0040
# define ROCKCHIP_SPI_DMARDLR 0x0044
# define ROCKCHIP_SPI_VERSION 0x0048
2014-07-01 09:03:59 +08:00
# define ROCKCHIP_SPI_TXDR 0x0400
# define ROCKCHIP_SPI_RXDR 0x0800
/* Bit fields in CTRLR0 */
# define CR0_DFS_OFFSET 0
2018-10-31 11:57:10 +01:00
# define CR0_DFS_4BIT 0x0
# define CR0_DFS_8BIT 0x1
# define CR0_DFS_16BIT 0x2
2014-07-01 09:03:59 +08:00
# define CR0_CFS_OFFSET 2
# define CR0_SCPH_OFFSET 6
# define CR0_SCPOL_OFFSET 7
# define CR0_CSM_OFFSET 8
# define CR0_CSM_KEEP 0x0
/* ss_n be high for half sclk_out cycles */
# define CR0_CSM_HALF 0X1
/* ss_n be high for one sclk_out cycle */
# define CR0_CSM_ONE 0x2
/* ss_n to sclk_out delay */
# define CR0_SSD_OFFSET 10
/*
* The period between ss_n active and
* sclk_out active is half sclk_out cycles
*/
# define CR0_SSD_HALF 0x0
/*
* The period between ss_n active and
* sclk_out active is one sclk_out cycle
*/
# define CR0_SSD_ONE 0x1
# define CR0_EM_OFFSET 11
# define CR0_EM_LITTLE 0x0
# define CR0_EM_BIG 0x1
# define CR0_FBM_OFFSET 12
# define CR0_FBM_MSB 0x0
# define CR0_FBM_LSB 0x1
# define CR0_BHT_OFFSET 13
# define CR0_BHT_16BIT 0x0
# define CR0_BHT_8BIT 0x1
# define CR0_RSD_OFFSET 14
2018-10-31 11:57:08 +01:00
# define CR0_RSD_MAX 0x3
2014-07-01 09:03:59 +08:00
# define CR0_FRF_OFFSET 16
# define CR0_FRF_SPI 0x0
# define CR0_FRF_SSP 0x1
# define CR0_FRF_MICROWIRE 0x2
# define CR0_XFM_OFFSET 18
# define CR0_XFM_MASK (0x03 << SPI_XFM_OFFSET)
# define CR0_XFM_TR 0x0
# define CR0_XFM_TO 0x1
# define CR0_XFM_RO 0x2
# define CR0_OPM_OFFSET 20
2023-08-18 17:31:44 +08:00
# define CR0_OPM_HOST 0x0
# define CR0_OPM_TARGET 0x1
2014-07-01 09:03:59 +08:00
2021-06-21 18:48:48 +08:00
# define CR0_SOI_OFFSET 23
2014-07-01 09:03:59 +08:00
# define CR0_MTM_OFFSET 0x21
/* Bit fields in SER, 2bit */
# define SER_MASK 0x3
2018-10-31 11:57:07 +01:00
/* Bit fields in BAUDR */
# define BAUDR_SCKDV_MIN 2
# define BAUDR_SCKDV_MAX 65534
2021-06-21 18:47:58 +08:00
/* Bit fields in SR, 6bit */
# define SR_MASK 0x3f
2014-07-01 09:03:59 +08:00
# define SR_BUSY (1 << 0)
# define SR_TF_FULL (1 << 1)
# define SR_TF_EMPTY (1 << 2)
# define SR_RF_EMPTY (1 << 3)
# define SR_RF_FULL (1 << 4)
2023-08-18 17:31:44 +08:00
# define SR_TARGET_TX_BUSY (1 << 5)
2014-07-01 09:03:59 +08:00
/* Bit fields in ISR, IMR, ISR, RISR, 5bit */
# define INT_MASK 0x1f
# define INT_TF_EMPTY (1 << 0)
# define INT_TF_OVERFLOW (1 << 1)
# define INT_RF_UNDERFLOW (1 << 2)
# define INT_RF_OVERFLOW (1 << 3)
2022-02-16 09:40:25 +08:00
# define INT_RF_FULL (1 << 4)
# define INT_CS_INACTIVE (1 << 6)
2014-07-01 09:03:59 +08:00
/* Bit fields in ICR, 4bit */
# define ICR_MASK 0x0f
# define ICR_ALL (1 << 0)
# define ICR_RF_UNDERFLOW (1 << 1)
# define ICR_RF_OVERFLOW (1 << 2)
# define ICR_TF_OVERFLOW (1 << 3)
/* Bit fields in DMACR */
# define RF_DMA_EN (1 << 0)
# define TF_DMA_EN (1 << 1)
2018-10-31 11:57:01 +01:00
/* Driver state flags */
# define RXDMA (1 << 0)
# define TXDMA (1 << 1)
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
/* sclk_out: spi host internal logic in rk3x can support 50Mhz */
2018-10-31 11:57:07 +01:00
# define MAX_SCLK_OUT 50000000U
2014-10-15 19:25:49 +08:00
2016-07-14 18:30:59 -07:00
/*
* SPI_CTRLR1 is 16 - bits , so we should support lengths of 0xffff + 1. However ,
* the controller seems to hang when given 0x10000 , so stick with this for now .
*/
# define ROCKCHIP_SPI_MAX_TRANLEN 0xffff
2024-01-26 20:46:44 +01:00
# define ROCKCHIP_SPI_MAX_NATIVE_CS_NUM 2
2020-07-23 08:43:55 +08:00
# define ROCKCHIP_SPI_VER2_TYPE1 0x05EC0002
# define ROCKCHIP_SPI_VER2_TYPE2 0x00110002
2017-06-28 12:38:43 +08:00
2020-10-16 11:50:14 +03:00
# define ROCKCHIP_AUTOSUSPEND_TIMEOUT 2000
2014-07-01 09:03:59 +08:00
struct rockchip_spi {
struct device * dev ;
struct clk * spiclk ;
struct clk * apb_pclk ;
void __iomem * regs ;
2018-10-31 11:57:04 +01:00
dma_addr_t dma_addr_rx ;
dma_addr_t dma_addr_tx ;
2018-10-31 11:57:01 +01:00
2018-10-31 11:57:09 +01:00
const void * tx ;
void * rx ;
unsigned int tx_left ;
unsigned int rx_left ;
2018-10-31 11:57:01 +01:00
atomic_t state ;
2014-07-01 09:03:59 +08:00
/*depth of the FIFO buffer */
u32 fifo_len ;
2018-10-31 11:57:07 +01:00
/* frequency of spiclk */
u32 freq ;
2014-07-01 09:03:59 +08:00
u8 n_bytes ;
2018-10-31 11:57:08 +01:00
u8 rsd ;
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
bool target_abort ;
bool cs_inactive ; /* spi target tansmition stop when cs inactive */
2022-04-21 23:32:51 +02:00
bool cs_high_supported ; /* native CS supports active-high polarity */
2022-02-16 09:40:25 +08:00
struct spi_transfer * xfer ; /* Store xfer temporarily */
2014-07-01 09:03:59 +08:00
} ;
2018-10-31 11:56:58 +01:00
static inline void spi_enable_chip ( struct rockchip_spi * rs , bool enable )
2014-07-01 09:03:59 +08:00
{
2018-10-31 11:56:58 +01:00
writel_relaxed ( ( enable ? 1U : 0U ) , rs - > regs + ROCKCHIP_SPI_SSIENR ) ;
2014-07-01 09:03:59 +08:00
}
2023-08-18 17:31:44 +08:00
static inline void wait_for_tx_idle ( struct rockchip_spi * rs , bool target_mode )
2014-07-11 10:08:24 +08:00
{
unsigned long timeout = jiffies + msecs_to_jiffies ( 5 ) ;
do {
2023-08-18 17:31:44 +08:00
if ( target_mode ) {
if ( ! ( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_SR ) & SR_TARGET_TX_BUSY ) & &
2021-06-21 18:47:58 +08:00
! ( ( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_SR ) & SR_BUSY ) ) )
return ;
} else {
if ( ! ( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_SR ) & SR_BUSY ) )
return ;
}
2014-09-03 13:44:25 -07:00
} while ( ! time_after ( jiffies , timeout ) ) ;
2014-07-11 10:08:24 +08:00
dev_warn ( rs - > dev , " spi controller is in busy state! \n " ) ;
}
2014-07-01 09:03:59 +08:00
static u32 get_fifo_len ( struct rockchip_spi * rs )
{
2020-07-23 08:43:55 +08:00
u32 ver ;
2014-07-01 09:03:59 +08:00
2020-07-23 08:43:55 +08:00
ver = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_VERSION ) ;
2014-07-01 09:03:59 +08:00
2020-07-23 08:43:55 +08:00
switch ( ver ) {
case ROCKCHIP_SPI_VER2_TYPE1 :
case ROCKCHIP_SPI_VER2_TYPE2 :
return 64 ;
default :
return 32 ;
}
2014-07-01 09:03:59 +08:00
}
static void rockchip_spi_set_cs ( struct spi_device * spi , bool enable )
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = spi - > controller ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2021-06-21 18:48:48 +08:00
bool cs_asserted = spi - > mode & SPI_CS_HIGH ? enable : ! enable ;
2016-02-24 18:00:04 +08:00
2017-06-28 12:38:43 +08:00
if ( cs_asserted ) {
/* Keep things powered as long as CS is asserted */
pm_runtime_get_sync ( rs - > dev ) ;
2014-07-01 09:03:59 +08:00
2023-03-10 23:02:03 +05:30
if ( spi_get_csgpiod ( spi , 0 ) )
2021-06-21 18:48:47 +08:00
ROCKCHIP_SPI_SET_BITS ( rs - > regs + ROCKCHIP_SPI_SER , 1 ) ;
else
2023-03-10 23:02:03 +05:30
ROCKCHIP_SPI_SET_BITS ( rs - > regs + ROCKCHIP_SPI_SER ,
BIT ( spi_get_chipselect ( spi , 0 ) ) ) ;
2017-06-28 12:38:43 +08:00
} else {
2023-03-10 23:02:03 +05:30
if ( spi_get_csgpiod ( spi , 0 ) )
2021-06-21 18:48:47 +08:00
ROCKCHIP_SPI_CLR_BITS ( rs - > regs + ROCKCHIP_SPI_SER , 1 ) ;
else
2023-03-10 23:02:03 +05:30
ROCKCHIP_SPI_CLR_BITS ( rs - > regs + ROCKCHIP_SPI_SER ,
BIT ( spi_get_chipselect ( spi , 0 ) ) ) ;
2014-07-01 09:03:59 +08:00
2017-06-28 12:38:43 +08:00
/* Drop reference from when we first asserted CS */
pm_runtime_put ( rs - > dev ) ;
}
2014-07-01 09:03:59 +08:00
}
2020-05-11 16:30:20 +08:00
static void rockchip_spi_handle_err ( struct spi_controller * ctlr ,
2015-02-27 17:34:16 +02:00
struct spi_message * msg )
2014-07-01 09:03:59 +08:00
{
2020-05-11 16:30:20 +08:00
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:02 +01:00
/* stop running spi transfer
* this also flushes both rx and tx fifos
2014-07-11 10:07:56 +08:00
*/
2018-10-31 11:57:02 +01:00
spi_enable_chip ( rs , false ) ;
2022-02-16 09:40:28 +08:00
/* make sure all interrupts are masked and status cleared */
2018-10-31 11:57:09 +01:00
writel_relaxed ( 0 , rs - > regs + ROCKCHIP_SPI_IMR ) ;
2022-02-16 09:40:28 +08:00
writel_relaxed ( 0xffffffff , rs - > regs + ROCKCHIP_SPI_ICR ) ;
2018-10-31 11:57:09 +01:00
2018-10-31 11:57:01 +01:00
if ( atomic_read ( & rs - > state ) & TXDMA )
2020-05-11 16:30:20 +08:00
dmaengine_terminate_async ( ctlr - > dma_tx ) ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:02 +01:00
if ( atomic_read ( & rs - > state ) & RXDMA )
2020-05-11 16:30:20 +08:00
dmaengine_terminate_async ( ctlr - > dma_rx ) ;
2014-07-01 09:03:59 +08:00
}
static void rockchip_spi_pio_writer ( struct rockchip_spi * rs )
{
2018-10-31 11:57:09 +01:00
u32 tx_free = rs - > fifo_len - readl_relaxed ( rs - > regs + ROCKCHIP_SPI_TXFLR ) ;
u32 words = min ( rs - > tx_left , tx_free ) ;
rs - > tx_left - = words ;
for ( ; words ; words - - ) {
u32 txw ;
2014-07-01 09:03:59 +08:00
if ( rs - > n_bytes = = 1 )
2018-10-31 11:57:09 +01:00
txw = * ( u8 * ) rs - > tx ;
2014-07-01 09:03:59 +08:00
else
2018-10-31 11:57:09 +01:00
txw = * ( u16 * ) rs - > tx ;
2014-07-01 09:03:59 +08:00
writel_relaxed ( txw , rs - > regs + ROCKCHIP_SPI_TXDR ) ;
rs - > tx + = rs - > n_bytes ;
}
}
static void rockchip_spi_pio_reader ( struct rockchip_spi * rs )
{
2018-10-31 11:57:09 +01:00
u32 words = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXFLR ) ;
2020-07-23 08:43:56 +08:00
u32 rx_left = ( rs - > rx_left > words ) ? rs - > rx_left - words : 0 ;
2018-10-31 11:57:09 +01:00
/* the hardware doesn't allow us to change fifo threshold
* level while spi is enabled , so instead make sure to leave
* enough words in the rx fifo to get the last interrupt
* exactly when all words have been received
*/
if ( rx_left ) {
u32 ftl = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXFTLR ) + 1 ;
if ( rx_left < ftl ) {
rx_left = ftl ;
words = rs - > rx_left - rx_left ;
}
}
rs - > rx_left = rx_left ;
for ( ; words ; words - - ) {
u32 rxw = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXDR ) ;
if ( ! rs - > rx )
continue ;
2014-07-01 09:03:59 +08:00
if ( rs - > n_bytes = = 1 )
2018-10-31 11:57:09 +01:00
* ( u8 * ) rs - > rx = ( u8 ) rxw ;
2014-07-01 09:03:59 +08:00
else
2018-10-31 11:57:09 +01:00
* ( u16 * ) rs - > rx = ( u16 ) rxw ;
2014-07-01 09:03:59 +08:00
rs - > rx + = rs - > n_bytes ;
2014-07-11 10:07:56 +08:00
}
2014-07-01 09:03:59 +08:00
}
2018-10-31 11:57:09 +01:00
static irqreturn_t rockchip_spi_isr ( int irq , void * dev_id )
2014-07-01 09:03:59 +08:00
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = dev_id ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
/* When int_cs_inactive comes, spi target abort */
2022-02-16 09:40:25 +08:00
if ( rs - > cs_inactive & & readl_relaxed ( rs - > regs + ROCKCHIP_SPI_IMR ) & INT_CS_INACTIVE ) {
2023-08-18 17:31:44 +08:00
ctlr - > target_abort ( ctlr ) ;
2022-02-16 09:40:25 +08:00
writel_relaxed ( 0 , rs - > regs + ROCKCHIP_SPI_IMR ) ;
writel_relaxed ( 0xffffffff , rs - > regs + ROCKCHIP_SPI_ICR ) ;
return IRQ_HANDLED ;
}
2018-10-31 11:57:09 +01:00
if ( rs - > tx_left )
rockchip_spi_pio_writer ( rs ) ;
2018-10-10 11:00:38 +02:00
2018-10-31 11:57:09 +01:00
rockchip_spi_pio_reader ( rs ) ;
if ( ! rs - > rx_left ) {
spi_enable_chip ( rs , false ) ;
writel_relaxed ( 0 , rs - > regs + ROCKCHIP_SPI_IMR ) ;
2022-02-16 09:40:25 +08:00
writel_relaxed ( 0xffffffff , rs - > regs + ROCKCHIP_SPI_ICR ) ;
2020-05-11 16:30:20 +08:00
spi_finalize_current_transfer ( ctlr ) ;
2018-10-31 11:57:09 +01:00
}
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:09 +01:00
return IRQ_HANDLED ;
}
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:09 +01:00
static int rockchip_spi_prepare_irq ( struct rockchip_spi * rs ,
2022-02-16 09:40:25 +08:00
struct spi_controller * ctlr ,
struct spi_transfer * xfer )
2018-10-31 11:57:09 +01:00
{
rs - > tx = xfer - > tx_buf ;
rs - > rx = xfer - > rx_buf ;
rs - > tx_left = rs - > tx ? xfer - > len / rs - > n_bytes : 0 ;
rs - > rx_left = xfer - > len / rs - > n_bytes ;
2014-07-01 09:03:59 +08:00
2022-06-17 20:42:51 +08:00
writel_relaxed ( 0xffffffff , rs - > regs + ROCKCHIP_SPI_ICR ) ;
2018-10-31 11:57:09 +01:00
spi_enable_chip ( rs , true ) ;
2014-07-11 10:08:24 +08:00
2018-10-31 11:57:09 +01:00
if ( rs - > tx_left )
rockchip_spi_pio_writer ( rs ) ;
2014-10-15 19:26:18 +08:00
2022-06-17 20:42:51 +08:00
if ( rs - > cs_inactive )
writel_relaxed ( INT_RF_FULL | INT_CS_INACTIVE , rs - > regs + ROCKCHIP_SPI_IMR ) ;
else
writel_relaxed ( INT_RF_FULL , rs - > regs + ROCKCHIP_SPI_IMR ) ;
2018-10-31 11:57:09 +01:00
/* 1 means the transfer is in progress */
return 1 ;
2014-07-01 09:03:59 +08:00
}
static void rockchip_spi_dma_rxcb ( void * data )
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = data ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2018-10-31 11:57:01 +01:00
int state = atomic_fetch_andnot ( RXDMA , & rs - > state ) ;
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
if ( state & TXDMA & & ! rs - > target_abort )
2018-10-31 11:57:01 +01:00
return ;
2014-07-01 09:03:59 +08:00
2022-02-16 09:40:25 +08:00
if ( rs - > cs_inactive )
writel_relaxed ( 0 , rs - > regs + ROCKCHIP_SPI_IMR ) ;
2018-10-31 11:57:01 +01:00
spi_enable_chip ( rs , false ) ;
2020-05-11 16:30:20 +08:00
spi_finalize_current_transfer ( ctlr ) ;
2014-07-01 09:03:59 +08:00
}
static void rockchip_spi_dma_txcb ( void * data )
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = data ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2018-10-31 11:57:01 +01:00
int state = atomic_fetch_andnot ( TXDMA , & rs - > state ) ;
2023-08-18 17:31:44 +08:00
if ( state & RXDMA & & ! rs - > target_abort )
2018-10-31 11:57:01 +01:00
return ;
2014-07-01 09:03:59 +08:00
2014-07-11 10:08:24 +08:00
/* Wait until the FIFO data completely. */
2023-08-18 17:31:44 +08:00
wait_for_tx_idle ( rs , ctlr - > target ) ;
2014-07-11 10:08:24 +08:00
2018-10-31 11:57:01 +01:00
spi_enable_chip ( rs , false ) ;
2020-05-11 16:30:20 +08:00
spi_finalize_current_transfer ( ctlr ) ;
2014-07-01 09:03:59 +08:00
}
2020-07-23 08:43:54 +08:00
static u32 rockchip_spi_calc_burst_size ( u32 data_len )
{
u32 i ;
/* burst size: 1, 2, 4, 8 */
for ( i = 1 ; i < 8 ; i < < = 1 ) {
if ( data_len & i )
break ;
}
return i ;
}
2018-10-31 11:57:03 +01:00
static int rockchip_spi_prepare_dma ( struct rockchip_spi * rs ,
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr , struct spi_transfer * xfer )
2014-07-01 09:03:59 +08:00
{
struct dma_async_tx_descriptor * rxdesc , * txdesc ;
2018-10-31 11:57:01 +01:00
atomic_set ( & rs - > state , 0 ) ;
2014-07-01 09:03:59 +08:00
2022-02-16 09:40:25 +08:00
rs - > tx = xfer - > tx_buf ;
rs - > rx = xfer - > rx_buf ;
2015-01-28 14:25:10 +01:00
rxdesc = NULL ;
2018-10-31 11:57:03 +01:00
if ( xfer - > rx_buf ) {
2018-10-31 11:56:59 +01:00
struct dma_slave_config rxconf = {
. direction = DMA_DEV_TO_MEM ,
2018-10-31 11:57:04 +01:00
. src_addr = rs - > dma_addr_rx ,
2018-10-31 11:56:59 +01:00
. src_addr_width = rs - > n_bytes ,
2022-02-16 09:40:25 +08:00
. src_maxburst = rockchip_spi_calc_burst_size ( xfer - > len / rs - > n_bytes ) ,
2018-10-31 11:56:59 +01:00
} ;
2020-05-11 16:30:20 +08:00
dmaengine_slave_config ( ctlr - > dma_rx , & rxconf ) ;
2014-07-01 09:03:59 +08:00
2014-07-11 10:07:56 +08:00
rxdesc = dmaengine_prep_slave_sg (
2020-05-11 16:30:20 +08:00
ctlr - > dma_rx ,
2018-10-31 11:57:03 +01:00
xfer - > rx_sg . sgl , xfer - > rx_sg . nents ,
2018-10-10 11:00:37 +02:00
DMA_DEV_TO_MEM , DMA_PREP_INTERRUPT ) ;
2016-03-09 16:11:15 +08:00
if ( ! rxdesc )
return - EINVAL ;
2014-07-01 09:03:59 +08:00
rxdesc - > callback = rockchip_spi_dma_rxcb ;
2020-05-11 16:30:20 +08:00
rxdesc - > callback_param = ctlr ;
2014-07-01 09:03:59 +08:00
}
2015-01-28 14:25:10 +01:00
txdesc = NULL ;
2018-10-31 11:57:03 +01:00
if ( xfer - > tx_buf ) {
2018-10-31 11:56:59 +01:00
struct dma_slave_config txconf = {
. direction = DMA_MEM_TO_DEV ,
2018-10-31 11:57:04 +01:00
. dst_addr = rs - > dma_addr_tx ,
2018-10-31 11:56:59 +01:00
. dst_addr_width = rs - > n_bytes ,
2019-04-12 12:53:20 +02:00
. dst_maxburst = rs - > fifo_len / 4 ,
2018-10-31 11:56:59 +01:00
} ;
2020-05-11 16:30:20 +08:00
dmaengine_slave_config ( ctlr - > dma_tx , & txconf ) ;
2014-07-01 09:03:59 +08:00
2014-07-11 10:07:56 +08:00
txdesc = dmaengine_prep_slave_sg (
2020-05-11 16:30:20 +08:00
ctlr - > dma_tx ,
2018-10-31 11:57:03 +01:00
xfer - > tx_sg . sgl , xfer - > tx_sg . nents ,
2018-10-10 11:00:37 +02:00
DMA_MEM_TO_DEV , DMA_PREP_INTERRUPT ) ;
2016-03-09 16:11:15 +08:00
if ( ! txdesc ) {
if ( rxdesc )
2020-05-11 16:30:20 +08:00
dmaengine_terminate_sync ( ctlr - > dma_rx ) ;
2016-03-09 16:11:15 +08:00
return - EINVAL ;
}
2014-07-01 09:03:59 +08:00
txdesc - > callback = rockchip_spi_dma_txcb ;
2020-05-11 16:30:20 +08:00
txdesc - > callback_param = ctlr ;
2014-07-01 09:03:59 +08:00
}
/* rx must be started before tx due to spi instinct */
2015-01-28 14:25:10 +01:00
if ( rxdesc ) {
2018-10-31 11:57:01 +01:00
atomic_or ( RXDMA , & rs - > state ) ;
2022-02-16 09:40:25 +08:00
ctlr - > dma_rx - > cookie = dmaengine_submit ( rxdesc ) ;
2020-05-11 16:30:20 +08:00
dma_async_issue_pending ( ctlr - > dma_rx ) ;
2014-07-01 09:03:59 +08:00
}
2022-02-16 09:40:25 +08:00
if ( rs - > cs_inactive )
writel_relaxed ( INT_CS_INACTIVE , rs - > regs + ROCKCHIP_SPI_IMR ) ;
2018-10-31 11:56:58 +01:00
spi_enable_chip ( rs , true ) ;
2018-10-10 11:00:38 +02:00
2015-01-28 14:25:10 +01:00
if ( txdesc ) {
2018-10-31 11:57:01 +01:00
atomic_or ( TXDMA , & rs - > state ) ;
2014-07-01 09:03:59 +08:00
dmaengine_submit ( txdesc ) ;
2020-05-11 16:30:20 +08:00
dma_async_issue_pending ( ctlr - > dma_tx ) ;
2014-07-01 09:03:59 +08:00
}
2016-03-09 16:11:15 +08:00
2018-10-10 11:00:38 +02:00
/* 1 means the transfer is in progress */
return 1 ;
2014-07-01 09:03:59 +08:00
}
2021-02-26 15:00:48 +01:00
static int rockchip_spi_config ( struct rockchip_spi * rs ,
2018-10-31 11:57:06 +01:00
struct spi_device * spi , struct spi_transfer * xfer ,
2023-08-18 17:31:44 +08:00
bool use_dma , bool target_mode )
2014-07-01 09:03:59 +08:00
{
2018-10-31 11:57:00 +01:00
u32 cr0 = CR0_FRF_SPI < < CR0_FRF_OFFSET
2021-03-24 14:16:41 +08:00
| CR0_BHT_8BIT < < CR0_BHT_OFFSET
| CR0_SSD_ONE < < CR0_SSD_OFFSET
| CR0_EM_BIG < < CR0_EM_OFFSET ;
2018-10-31 11:57:10 +01:00
u32 cr1 ;
u32 dmacr = 0 ;
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
if ( target_mode )
cr0 | = CR0_OPM_TARGET < < CR0_OPM_OFFSET ;
rs - > target_abort = false ;
2020-05-11 16:30:21 +08:00
2018-10-31 11:57:08 +01:00
cr0 | = rs - > rsd < < CR0_RSD_OFFSET ;
2018-10-31 11:57:03 +01:00
cr0 | = ( spi - > mode & 0x3U ) < < CR0_SCPH_OFFSET ;
2018-10-31 11:57:11 +01:00
if ( spi - > mode & SPI_LSB_FIRST )
cr0 | = CR0_FBM_LSB < < CR0_FBM_OFFSET ;
2021-06-21 18:48:48 +08:00
if ( spi - > mode & SPI_CS_HIGH )
2023-03-10 23:02:03 +05:30
cr0 | = BIT ( spi_get_chipselect ( spi , 0 ) ) < < CR0_SOI_OFFSET ;
2018-10-31 11:57:03 +01:00
if ( xfer - > rx_buf & & xfer - > tx_buf )
cr0 | = CR0_XFM_TR < < CR0_XFM_OFFSET ;
else if ( xfer - > rx_buf )
cr0 | = CR0_XFM_RO < < CR0_XFM_OFFSET ;
2018-10-31 11:57:09 +01:00
else if ( use_dma )
2018-10-31 11:57:03 +01:00
cr0 | = CR0_XFM_TO < < CR0_XFM_OFFSET ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:10 +01:00
switch ( xfer - > bits_per_word ) {
case 4 :
cr0 | = CR0_DFS_4BIT < < CR0_DFS_OFFSET ;
cr1 = xfer - > len - 1 ;
break ;
case 8 :
cr0 | = CR0_DFS_8BIT < < CR0_DFS_OFFSET ;
cr1 = xfer - > len - 1 ;
break ;
case 16 :
cr0 | = CR0_DFS_16BIT < < CR0_DFS_OFFSET ;
cr1 = xfer - > len / 2 - 1 ;
break ;
default :
/* we only whitelist 4, 8 and 16 bit words in
2020-05-11 16:30:20 +08:00
* ctlr - > bits_per_word_mask , so this shouldn ' t
2018-10-31 11:57:10 +01:00
* happen
*/
2021-02-26 15:00:48 +01:00
dev_err ( rs - > dev , " unknown bits per word: %d \n " ,
xfer - > bits_per_word ) ;
return - EINVAL ;
2018-10-31 11:57:10 +01:00
}
2018-10-31 11:57:06 +01:00
if ( use_dma ) {
2018-10-31 11:57:03 +01:00
if ( xfer - > tx_buf )
2014-07-01 09:03:59 +08:00
dmacr | = TF_DMA_EN ;
2018-10-31 11:57:03 +01:00
if ( xfer - > rx_buf )
2014-07-01 09:03:59 +08:00
dmacr | = RF_DMA_EN ;
}
writel_relaxed ( cr0 , rs - > regs + ROCKCHIP_SPI_CTRLR0 ) ;
2018-10-31 11:57:10 +01:00
writel_relaxed ( cr1 , rs - > regs + ROCKCHIP_SPI_CTRLR1 ) ;
2017-08-16 10:12:02 +08:00
2018-10-31 11:57:09 +01:00
/* unfortunately setting the fifo threshold level to generate an
* interrupt exactly when the fifo is full doesn ' t seem to work ,
* so we need the strict inequality here
*/
2021-06-21 18:47:57 +08:00
if ( ( xfer - > len / rs - > n_bytes ) < rs - > fifo_len )
writel_relaxed ( xfer - > len / rs - > n_bytes - 1 , rs - > regs + ROCKCHIP_SPI_RXFTLR ) ;
2018-10-31 11:57:09 +01:00
else
writel_relaxed ( rs - > fifo_len / 2 - 1 , rs - > regs + ROCKCHIP_SPI_RXFTLR ) ;
2014-07-01 09:03:59 +08:00
2021-06-21 18:47:58 +08:00
writel_relaxed ( rs - > fifo_len / 2 - 1 , rs - > regs + ROCKCHIP_SPI_DMATDLR ) ;
2020-07-23 08:43:54 +08:00
writel_relaxed ( rockchip_spi_calc_burst_size ( xfer - > len / rs - > n_bytes ) - 1 ,
rs - > regs + ROCKCHIP_SPI_DMARDLR ) ;
2014-07-01 09:03:59 +08:00
writel_relaxed ( dmacr , rs - > regs + ROCKCHIP_SPI_DMACR ) ;
2018-10-31 11:57:07 +01:00
/* the hardware only supports an even clock divisor, so
* round divisor = spiclk / speed up to nearest even number
* so that the resulting speed is < = the requested speed
*/
writel_relaxed ( 2 * DIV_ROUND_UP ( rs - > freq , 2 * xfer - > speed_hz ) ,
rs - > regs + ROCKCHIP_SPI_BAUDR ) ;
2021-02-26 15:00:48 +01:00
return 0 ;
2014-07-01 09:03:59 +08:00
}
2016-07-14 18:30:59 -07:00
static size_t rockchip_spi_max_transfer_size ( struct spi_device * spi )
{
return ROCKCHIP_SPI_MAX_TRANLEN ;
}
2023-08-18 17:31:44 +08:00
static int rockchip_spi_target_abort ( struct spi_controller * ctlr )
2020-05-11 16:30:21 +08:00
{
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2022-02-16 09:40:25 +08:00
u32 rx_fifo_left ;
struct dma_tx_state state ;
enum dma_status status ;
/* Get current dma rx point */
if ( atomic_read ( & rs - > state ) & RXDMA ) {
dmaengine_pause ( ctlr - > dma_rx ) ;
status = dmaengine_tx_status ( ctlr - > dma_rx , ctlr - > dma_rx - > cookie , & state ) ;
if ( status = = DMA_ERROR ) {
rs - > rx = rs - > xfer - > rx_buf ;
rs - > xfer - > len = 0 ;
rx_fifo_left = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXFLR ) ;
for ( ; rx_fifo_left ; rx_fifo_left - - )
readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXDR ) ;
goto out ;
} else {
rs - > rx + = rs - > xfer - > len - rs - > n_bytes * state . residue ;
}
}
2020-05-11 16:30:21 +08:00
2022-02-16 09:40:25 +08:00
/* Get the valid data left in rx fifo and set rs->xfer->len real rx size */
if ( rs - > rx ) {
rx_fifo_left = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXFLR ) ;
for ( ; rx_fifo_left ; rx_fifo_left - - ) {
u32 rxw = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_RXDR ) ;
if ( rs - > n_bytes = = 1 )
* ( u8 * ) rs - > rx = ( u8 ) rxw ;
else
* ( u16 * ) rs - > rx = ( u16 ) rxw ;
rs - > rx + = rs - > n_bytes ;
}
rs - > xfer - > len = ( unsigned int ) ( rs - > rx - rs - > xfer - > rx_buf ) ;
}
out :
2022-02-16 09:40:24 +08:00
if ( atomic_read ( & rs - > state ) & RXDMA )
dmaengine_terminate_sync ( ctlr - > dma_rx ) ;
if ( atomic_read ( & rs - > state ) & TXDMA )
dmaengine_terminate_sync ( ctlr - > dma_tx ) ;
atomic_set ( & rs - > state , 0 ) ;
spi_enable_chip ( rs , false ) ;
2023-08-18 17:31:44 +08:00
rs - > target_abort = true ;
2021-01-07 23:58:32 +00:00
spi_finalize_current_transfer ( ctlr ) ;
2020-05-11 16:30:21 +08:00
return 0 ;
}
2014-07-11 10:07:56 +08:00
static int rockchip_spi_transfer_one (
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr ,
2014-07-01 09:03:59 +08:00
struct spi_device * spi ,
struct spi_transfer * xfer )
{
2020-05-11 16:30:20 +08:00
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2021-02-26 15:00:48 +01:00
int ret ;
2018-10-31 11:57:06 +01:00
bool use_dma ;
2014-07-01 09:03:59 +08:00
2021-08-27 07:03:57 +02:00
/* Zero length transfers won't trigger an interrupt on completion */
if ( ! xfer - > len ) {
spi_finalize_current_transfer ( ctlr ) ;
return 1 ;
}
2014-09-03 13:44:26 -07:00
WARN_ON ( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_SSIENR ) & &
( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_SR ) & SR_BUSY ) ) ;
2014-07-01 09:03:59 +08:00
if ( ! xfer - > tx_buf & & ! xfer - > rx_buf ) {
dev_err ( rs - > dev , " No buffer for transfer \n " ) ;
return - EINVAL ;
}
2016-07-14 18:30:59 -07:00
if ( xfer - > len > ROCKCHIP_SPI_MAX_TRANLEN ) {
dev_err ( rs - > dev , " Transfer is too long (%d) \n " , xfer - > len ) ;
return - EINVAL ;
}
2018-10-31 11:57:10 +01:00
rs - > n_bytes = xfer - > bits_per_word < = 8 ? 1 : 2 ;
2022-02-16 09:40:25 +08:00
rs - > xfer = xfer ;
2020-05-11 16:30:20 +08:00
use_dma = ctlr - > can_dma ? ctlr - > can_dma ( ctlr , spi , xfer ) : false ;
2014-07-01 09:03:59 +08:00
2023-08-18 17:31:44 +08:00
ret = rockchip_spi_config ( rs , spi , xfer , use_dma , ctlr - > target ) ;
2021-02-26 15:00:48 +01:00
if ( ret )
return ret ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:06 +01:00
if ( use_dma )
2020-05-11 16:30:20 +08:00
return rockchip_spi_prepare_dma ( rs , ctlr , xfer ) ;
2014-07-01 09:03:59 +08:00
2022-02-16 09:40:25 +08:00
return rockchip_spi_prepare_irq ( rs , ctlr , xfer ) ;
2014-07-01 09:03:59 +08:00
}
2020-05-11 16:30:20 +08:00
static bool rockchip_spi_can_dma ( struct spi_controller * ctlr ,
2014-07-11 10:07:56 +08:00
struct spi_device * spi ,
struct spi_transfer * xfer )
2014-07-01 09:03:59 +08:00
{
2020-05-11 16:30:20 +08:00
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2018-10-31 11:57:09 +01:00
unsigned int bytes_per_word = xfer - > bits_per_word < = 8 ? 1 : 2 ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:09 +01:00
/* if the numbor of spi words to transfer is less than the fifo
* length we can just fill the fifo and wait for a single irq ,
* so don ' t bother setting up dma
*/
return xfer - > len / bytes_per_word > = rs - > fifo_len ;
2014-07-01 09:03:59 +08:00
}
2022-02-16 09:40:26 +08:00
static int rockchip_spi_setup ( struct spi_device * spi )
{
struct rockchip_spi * rs = spi_controller_get_devdata ( spi - > controller ) ;
u32 cr0 ;
2023-03-10 23:02:03 +05:30
if ( ! spi_get_csgpiod ( spi , 0 ) & & ( spi - > mode & SPI_CS_HIGH ) & & ! rs - > cs_high_supported ) {
2022-04-21 23:32:51 +02:00
dev_warn ( & spi - > dev , " setup: non GPIO CS can't be active-high \n " ) ;
return - EINVAL ;
}
2022-02-16 09:40:26 +08:00
pm_runtime_get_sync ( rs - > dev ) ;
cr0 = readl_relaxed ( rs - > regs + ROCKCHIP_SPI_CTRLR0 ) ;
cr0 & = ~ ( 0x3 < < CR0_SCPH_OFFSET ) ;
cr0 | = ( ( spi - > mode & 0x3 ) < < CR0_SCPH_OFFSET ) ;
2023-03-10 23:02:03 +05:30
if ( spi - > mode & SPI_CS_HIGH & & spi_get_chipselect ( spi , 0 ) < = 1 )
cr0 | = BIT ( spi_get_chipselect ( spi , 0 ) ) < < CR0_SOI_OFFSET ;
else if ( spi_get_chipselect ( spi , 0 ) < = 1 )
cr0 & = ~ ( BIT ( spi_get_chipselect ( spi , 0 ) ) < < CR0_SOI_OFFSET ) ;
2022-02-16 09:40:26 +08:00
writel_relaxed ( cr0 , rs - > regs + ROCKCHIP_SPI_CTRLR0 ) ;
pm_runtime_put ( rs - > dev ) ;
return 0 ;
}
2014-07-01 09:03:59 +08:00
static int rockchip_spi_probe ( struct platform_device * pdev )
{
2017-08-07 20:40:18 +08:00
int ret ;
2014-07-01 09:03:59 +08:00
struct rockchip_spi * rs ;
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr ;
2014-07-01 09:03:59 +08:00
struct resource * mem ;
2020-05-11 16:30:21 +08:00
struct device_node * np = pdev - > dev . of_node ;
2022-02-16 09:40:23 +08:00
u32 rsd_nsecs , num_cs ;
2023-08-18 17:31:44 +08:00
bool target_mode ;
2020-05-11 16:30:21 +08:00
2023-08-18 17:31:44 +08:00
target_mode = of_property_read_bool ( np , " spi-slave " ) ;
2020-05-11 16:30:21 +08:00
2023-08-18 17:31:44 +08:00
if ( target_mode )
ctlr = spi_alloc_target ( & pdev - > dev ,
2020-05-11 16:30:21 +08:00
sizeof ( struct rockchip_spi ) ) ;
else
2023-08-18 17:31:44 +08:00
ctlr = spi_alloc_host ( & pdev - > dev ,
2020-05-11 16:30:21 +08:00
sizeof ( struct rockchip_spi ) ) ;
2014-07-01 09:03:59 +08:00
2020-05-11 16:30:20 +08:00
if ( ! ctlr )
2014-07-01 09:03:59 +08:00
return - ENOMEM ;
2014-07-11 10:07:56 +08:00
2020-05-11 16:30:20 +08:00
platform_set_drvdata ( pdev , ctlr ) ;
2014-07-01 09:03:59 +08:00
2020-05-11 16:30:20 +08:00
rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
/* Get basic io resource and map it */
2023-02-26 14:51:25 +08:00
rs - > regs = devm_platform_get_and_ioremap_resource ( pdev , 0 , & mem ) ;
2014-07-01 09:03:59 +08:00
if ( IS_ERR ( rs - > regs ) ) {
ret = PTR_ERR ( rs - > regs ) ;
2020-05-11 16:30:20 +08:00
goto err_put_ctlr ;
2014-07-01 09:03:59 +08:00
}
2023-08-23 21:39:38 +08:00
rs - > apb_pclk = devm_clk_get_enabled ( & pdev - > dev , " apb_pclk " ) ;
2014-07-01 09:03:59 +08:00
if ( IS_ERR ( rs - > apb_pclk ) ) {
dev_err ( & pdev - > dev , " Failed to get apb_pclk \n " ) ;
ret = PTR_ERR ( rs - > apb_pclk ) ;
2020-05-11 16:30:20 +08:00
goto err_put_ctlr ;
2014-07-01 09:03:59 +08:00
}
2023-08-23 21:39:38 +08:00
rs - > spiclk = devm_clk_get_enabled ( & pdev - > dev , " spiclk " ) ;
2014-07-01 09:03:59 +08:00
if ( IS_ERR ( rs - > spiclk ) ) {
dev_err ( & pdev - > dev , " Failed to get spi_pclk \n " ) ;
ret = PTR_ERR ( rs - > spiclk ) ;
2020-05-11 16:30:20 +08:00
goto err_put_ctlr ;
2014-07-01 09:03:59 +08:00
}
2018-10-31 11:56:58 +01:00
spi_enable_chip ( rs , false ) ;
2014-07-01 09:03:59 +08:00
2018-10-31 11:57:09 +01:00
ret = platform_get_irq ( pdev , 0 ) ;
if ( ret < 0 )
2023-08-23 21:39:38 +08:00
goto err_put_ctlr ;
2018-10-31 11:57:09 +01:00
ret = devm_request_threaded_irq ( & pdev - > dev , ret , rockchip_spi_isr , NULL ,
2020-05-11 16:30:20 +08:00
IRQF_ONESHOT , dev_name ( & pdev - > dev ) , ctlr ) ;
2018-10-31 11:57:09 +01:00
if ( ret )
2023-08-23 21:39:38 +08:00
goto err_put_ctlr ;
2018-10-31 11:57:09 +01:00
2014-07-01 09:03:59 +08:00
rs - > dev = & pdev - > dev ;
2018-10-31 11:57:07 +01:00
rs - > freq = clk_get_rate ( rs - > spiclk ) ;
2014-07-01 09:03:59 +08:00
2015-03-26 16:30:25 -07:00
if ( ! of_property_read_u32 ( pdev - > dev . of_node , " rx-sample-delay-ns " ,
2018-10-31 11:57:08 +01:00
& rsd_nsecs ) ) {
/* rx sample delay is expressed in parent clock cycles (max 3) */
u32 rsd = DIV_ROUND_CLOSEST ( rsd_nsecs * ( rs - > freq > > 8 ) ,
1000000000 > > 8 ) ;
if ( ! rsd ) {
dev_warn ( rs - > dev , " %u Hz are too slow to express %u ns delay \n " ,
rs - > freq , rsd_nsecs ) ;
} else if ( rsd > CR0_RSD_MAX ) {
rsd = CR0_RSD_MAX ;
dev_warn ( rs - > dev , " %u Hz are too fast to express %u ns delay, clamping at %u ns \n " ,
rs - > freq , rsd_nsecs ,
CR0_RSD_MAX * 1000000000U / rs - > freq ) ;
}
rs - > rsd = rsd ;
}
2015-03-26 16:30:25 -07:00
2014-07-01 09:03:59 +08:00
rs - > fifo_len = get_fifo_len ( rs ) ;
if ( ! rs - > fifo_len ) {
dev_err ( & pdev - > dev , " Failed to get fifo length \n " ) ;
2014-07-20 22:02:04 +08:00
ret = - EINVAL ;
2023-08-23 21:39:38 +08:00
goto err_put_ctlr ;
2014-07-01 09:03:59 +08:00
}
2020-10-16 11:50:14 +03:00
pm_runtime_set_autosuspend_delay ( & pdev - > dev , ROCKCHIP_AUTOSUSPEND_TIMEOUT ) ;
pm_runtime_use_autosuspend ( & pdev - > dev ) ;
2014-07-01 09:03:59 +08:00
pm_runtime_set_active ( & pdev - > dev ) ;
pm_runtime_enable ( & pdev - > dev ) ;
2020-05-11 16:30:20 +08:00
ctlr - > auto_runtime_pm = true ;
ctlr - > bus_num = pdev - > id ;
ctlr - > mode_bits = SPI_CPOL | SPI_CPHA | SPI_LOOP | SPI_LSB_FIRST ;
2023-08-18 17:31:44 +08:00
if ( target_mode ) {
2020-05-11 16:30:21 +08:00
ctlr - > mode_bits | = SPI_NO_CS ;
2023-08-18 17:31:44 +08:00
ctlr - > target_abort = rockchip_spi_target_abort ;
2020-05-11 16:30:21 +08:00
} else {
2023-07-10 18:49:30 +03:00
ctlr - > flags = SPI_CONTROLLER_GPIO_SS ;
2024-01-26 20:46:44 +01:00
ctlr - > max_native_cs = ROCKCHIP_SPI_MAX_NATIVE_CS_NUM ;
2020-05-11 16:30:22 +08:00
/*
* rk spi0 has two native cs , spi1 . .5 one cs only
* if num - cs is missing in the dts , default to 1
*/
2022-02-16 09:40:23 +08:00
if ( of_property_read_u32 ( np , " num-cs " , & num_cs ) )
num_cs = 1 ;
ctlr - > num_chipselect = num_cs ;
2020-05-11 16:30:22 +08:00
ctlr - > use_gpio_descriptors = true ;
2020-05-11 16:30:21 +08:00
}
2020-05-11 16:30:20 +08:00
ctlr - > dev . of_node = pdev - > dev . of_node ;
ctlr - > bits_per_word_mask = SPI_BPW_MASK ( 16 ) | SPI_BPW_MASK ( 8 ) | SPI_BPW_MASK ( 4 ) ;
ctlr - > min_speed_hz = rs - > freq / BAUDR_SCKDV_MAX ;
ctlr - > max_speed_hz = min ( rs - > freq / BAUDR_SCKDV_MIN , MAX_SCLK_OUT ) ;
2022-02-16 09:40:26 +08:00
ctlr - > setup = rockchip_spi_setup ;
2020-05-11 16:30:20 +08:00
ctlr - > set_cs = rockchip_spi_set_cs ;
ctlr - > transfer_one = rockchip_spi_transfer_one ;
ctlr - > max_transfer_size = rockchip_spi_max_transfer_size ;
ctlr - > handle_err = rockchip_spi_handle_err ;
ctlr - > dma_tx = dma_request_chan ( rs - > dev , " tx " ) ;
if ( IS_ERR ( ctlr - > dma_tx ) ) {
2016-03-09 16:11:32 +08:00
/* Check tx to see if we need defer probing driver */
2020-05-11 16:30:20 +08:00
if ( PTR_ERR ( ctlr - > dma_tx ) = = - EPROBE_DEFER ) {
2016-03-09 16:11:32 +08:00
ret = - EPROBE_DEFER ;
2017-06-13 13:25:40 +08:00
goto err_disable_pm_runtime ;
2016-03-09 16:11:32 +08:00
}
2014-07-01 09:03:59 +08:00
dev_warn ( rs - > dev , " Failed to request TX DMA channel \n " ) ;
2020-05-11 16:30:20 +08:00
ctlr - > dma_tx = NULL ;
2016-03-09 16:11:32 +08:00
}
2014-07-01 09:03:59 +08:00
2020-05-11 16:30:20 +08:00
ctlr - > dma_rx = dma_request_chan ( rs - > dev , " rx " ) ;
if ( IS_ERR ( ctlr - > dma_rx ) ) {
if ( PTR_ERR ( ctlr - > dma_rx ) = = - EPROBE_DEFER ) {
2016-03-31 11:11:41 +08:00
ret = - EPROBE_DEFER ;
2016-05-04 09:25:46 +03:00
goto err_free_dma_tx ;
2014-07-01 09:03:59 +08:00
}
dev_warn ( rs - > dev , " Failed to request RX DMA channel \n " ) ;
2020-05-11 16:30:20 +08:00
ctlr - > dma_rx = NULL ;
2014-07-01 09:03:59 +08:00
}
2020-05-11 16:30:20 +08:00
if ( ctlr - > dma_tx & & ctlr - > dma_rx ) {
2018-10-31 11:57:04 +01:00
rs - > dma_addr_tx = mem - > start + ROCKCHIP_SPI_TXDR ;
rs - > dma_addr_rx = mem - > start + ROCKCHIP_SPI_RXDR ;
2020-05-11 16:30:20 +08:00
ctlr - > can_dma = rockchip_spi_can_dma ;
2014-07-01 09:03:59 +08:00
}
2021-06-21 18:48:48 +08:00
switch ( readl_relaxed ( rs - > regs + ROCKCHIP_SPI_VERSION ) ) {
case ROCKCHIP_SPI_VER2_TYPE2 :
2022-04-21 23:32:51 +02:00
rs - > cs_high_supported = true ;
2021-06-21 18:48:48 +08:00
ctlr - > mode_bits | = SPI_CS_HIGH ;
2023-08-18 17:31:44 +08:00
if ( ctlr - > can_dma & & target_mode )
2022-02-16 09:40:25 +08:00
rs - > cs_inactive = true ;
else
rs - > cs_inactive = false ;
2021-06-21 18:48:48 +08:00
break ;
default :
2022-02-16 09:40:25 +08:00
rs - > cs_inactive = false ;
2021-06-21 18:48:48 +08:00
break ;
}
2020-05-11 16:30:20 +08:00
ret = devm_spi_register_controller ( & pdev - > dev , ctlr ) ;
2017-08-07 20:40:18 +08:00
if ( ret < 0 ) {
2020-05-11 16:30:20 +08:00
dev_err ( & pdev - > dev , " Failed to register controller \n " ) ;
2017-06-13 13:25:40 +08:00
goto err_free_dma_rx ;
2014-07-01 09:03:59 +08:00
}
return 0 ;
2017-06-13 13:25:40 +08:00
err_free_dma_rx :
2020-05-11 16:30:20 +08:00
if ( ctlr - > dma_rx )
dma_release_channel ( ctlr - > dma_rx ) ;
2016-05-04 09:25:46 +03:00
err_free_dma_tx :
2020-05-11 16:30:20 +08:00
if ( ctlr - > dma_tx )
dma_release_channel ( ctlr - > dma_tx ) ;
2017-06-13 13:25:40 +08:00
err_disable_pm_runtime :
pm_runtime_disable ( & pdev - > dev ) ;
2020-05-11 16:30:20 +08:00
err_put_ctlr :
spi_controller_put ( ctlr ) ;
2014-07-01 09:03:59 +08:00
return ret ;
}
2023-03-03 18:20:13 +01:00
static void rockchip_spi_remove ( struct platform_device * pdev )
2014-07-01 09:03:59 +08:00
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = spi_controller_get ( platform_get_drvdata ( pdev ) ) ;
2014-07-01 09:03:59 +08:00
2017-08-07 20:40:19 +08:00
pm_runtime_get_sync ( & pdev - > dev ) ;
2014-07-01 09:03:59 +08:00
2017-08-07 20:40:19 +08:00
pm_runtime_put_noidle ( & pdev - > dev ) ;
pm_runtime_disable ( & pdev - > dev ) ;
pm_runtime_set_suspended ( & pdev - > dev ) ;
2020-05-11 16:30:20 +08:00
if ( ctlr - > dma_tx )
dma_release_channel ( ctlr - > dma_tx ) ;
if ( ctlr - > dma_rx )
dma_release_channel ( ctlr - > dma_rx ) ;
2014-07-01 09:03:59 +08:00
2020-05-11 16:30:20 +08:00
spi_controller_put ( ctlr ) ;
2014-07-01 09:03:59 +08:00
}
# ifdef CONFIG_PM_SLEEP
static int rockchip_spi_suspend ( struct device * dev )
{
2017-08-07 20:40:18 +08:00
int ret ;
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = dev_get_drvdata ( dev ) ;
2022-02-16 09:40:27 +08:00
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
2020-05-11 16:30:20 +08:00
ret = spi_controller_suspend ( ctlr ) ;
2017-08-07 20:40:18 +08:00
if ( ret < 0 )
2014-07-01 09:03:59 +08:00
return ret ;
2022-02-16 09:40:27 +08:00
clk_disable_unprepare ( rs - > spiclk ) ;
clk_disable_unprepare ( rs - > apb_pclk ) ;
2014-07-01 09:03:59 +08:00
2016-12-16 16:59:16 -08:00
pinctrl_pm_select_sleep_state ( dev ) ;
2017-08-07 20:40:18 +08:00
return 0 ;
2014-07-01 09:03:59 +08:00
}
static int rockchip_spi_resume ( struct device * dev )
{
2017-08-07 20:40:18 +08:00
int ret ;
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = dev_get_drvdata ( dev ) ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
2016-12-16 16:59:16 -08:00
pinctrl_pm_select_default_state ( dev ) ;
2022-02-16 09:40:27 +08:00
ret = clk_prepare_enable ( rs - > apb_pclk ) ;
2017-08-07 20:40:20 +08:00
if ( ret < 0 )
return ret ;
2014-07-01 09:03:59 +08:00
2022-02-16 09:40:27 +08:00
ret = clk_prepare_enable ( rs - > spiclk ) ;
if ( ret < 0 )
clk_disable_unprepare ( rs - > apb_pclk ) ;
2020-05-11 16:30:20 +08:00
ret = spi_controller_resume ( ctlr ) ;
2014-07-01 09:03:59 +08:00
if ( ret < 0 ) {
clk_disable_unprepare ( rs - > spiclk ) ;
clk_disable_unprepare ( rs - > apb_pclk ) ;
}
2017-08-07 20:40:18 +08:00
return 0 ;
2014-07-01 09:03:59 +08:00
}
# endif /* CONFIG_PM_SLEEP */
2014-12-13 00:41:15 +01:00
# ifdef CONFIG_PM
2014-07-01 09:03:59 +08:00
static int rockchip_spi_runtime_suspend ( struct device * dev )
{
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = dev_get_drvdata ( dev ) ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
clk_disable_unprepare ( rs - > spiclk ) ;
clk_disable_unprepare ( rs - > apb_pclk ) ;
return 0 ;
}
static int rockchip_spi_runtime_resume ( struct device * dev )
{
int ret ;
2020-05-11 16:30:20 +08:00
struct spi_controller * ctlr = dev_get_drvdata ( dev ) ;
struct rockchip_spi * rs = spi_controller_get_devdata ( ctlr ) ;
2014-07-01 09:03:59 +08:00
ret = clk_prepare_enable ( rs - > apb_pclk ) ;
2017-08-07 20:40:18 +08:00
if ( ret < 0 )
2014-07-01 09:03:59 +08:00
return ret ;
ret = clk_prepare_enable ( rs - > spiclk ) ;
2017-08-07 20:40:18 +08:00
if ( ret < 0 )
2014-07-01 09:03:59 +08:00
clk_disable_unprepare ( rs - > apb_pclk ) ;
2017-08-07 20:40:18 +08:00
return 0 ;
2014-07-01 09:03:59 +08:00
}
2014-12-13 00:41:15 +01:00
# endif /* CONFIG_PM */
2014-07-01 09:03:59 +08:00
static const struct dev_pm_ops rockchip_spi_pm = {
2022-02-16 09:40:27 +08:00
SET_NOIRQ_SYSTEM_SLEEP_PM_OPS ( rockchip_spi_suspend , rockchip_spi_resume )
2014-07-01 09:03:59 +08:00
SET_RUNTIME_PM_OPS ( rockchip_spi_runtime_suspend ,
rockchip_spi_runtime_resume , NULL )
} ;
static const struct of_device_id rockchip_spi_dt_match [ ] = {
2020-03-09 16:10:03 +01:00
{ . compatible = " rockchip,px30-spi " , } ,
2016-05-20 07:56:21 +08:00
{ . compatible = " rockchip,rk3036-spi " , } ,
2014-07-01 09:03:59 +08:00
{ . compatible = " rockchip,rk3066-spi " , } ,
2014-07-11 10:09:19 +08:00
{ . compatible = " rockchip,rk3188-spi " , } ,
2016-05-20 07:56:21 +08:00
{ . compatible = " rockchip,rk3228-spi " , } ,
2014-07-11 10:09:19 +08:00
{ . compatible = " rockchip,rk3288-spi " , } ,
2020-03-09 16:10:03 +01:00
{ . compatible = " rockchip,rk3308-spi " , } ,
{ . compatible = " rockchip,rk3328-spi " , } ,
2016-05-20 07:56:21 +08:00
{ . compatible = " rockchip,rk3368-spi " , } ,
2016-02-18 19:16:31 +08:00
{ . compatible = " rockchip,rk3399-spi " , } ,
2020-03-09 16:10:03 +01:00
{ . compatible = " rockchip,rv1108-spi " , } ,
2021-06-21 18:47:56 +08:00
{ . compatible = " rockchip,rv1126-spi " , } ,
2014-07-01 09:03:59 +08:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , rockchip_spi_dt_match ) ;
static struct platform_driver rockchip_spi_driver = {
. driver = {
. name = DRIVER_NAME ,
. pm = & rockchip_spi_pm ,
. of_match_table = of_match_ptr ( rockchip_spi_dt_match ) ,
} ,
. probe = rockchip_spi_probe ,
2023-03-03 18:20:13 +01:00
. remove_new = rockchip_spi_remove ,
2014-07-01 09:03:59 +08:00
} ;
module_platform_driver ( rockchip_spi_driver ) ;
2014-07-11 10:07:56 +08:00
MODULE_AUTHOR ( " Addy Ke <addy.ke@rock-chips.com> " ) ;
2014-07-01 09:03:59 +08:00
MODULE_DESCRIPTION ( " ROCKCHIP SPI Controller Driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;