2019-05-28 20:10:04 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2013-02-22 16:37:39 +04:00
/*
* SPI driver for NVIDIA ' s Tegra114 SPI Controller .
*
* Copyright ( c ) 2013 , NVIDIA CORPORATION . All rights reserved .
*/
# include <linux/clk.h>
# include <linux/completion.h>
# include <linux/delay.h>
# include <linux/dmaengine.h>
# include <linux/dma-mapping.h>
# include <linux/dmapool.h>
# include <linux/err.h>
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/kthread.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/pm_runtime.h>
# include <linux/of.h>
# include <linux/of_device.h>
2013-11-07 03:31:24 +04:00
# include <linux/reset.h>
2013-02-22 16:37:39 +04:00
# include <linux/spi/spi.h>
# define SPI_COMMAND1 0x000
# define SPI_BIT_LENGTH(x) (((x) & 0x1f) << 0)
# define SPI_PACKED (1 << 5)
# define SPI_TX_EN (1 << 11)
# define SPI_RX_EN (1 << 12)
# define SPI_BOTH_EN_BYTE (1 << 13)
# define SPI_BOTH_EN_BIT (1 << 14)
# define SPI_LSBYTE_FE (1 << 15)
# define SPI_LSBIT_FE (1 << 16)
# define SPI_BIDIROE (1 << 17)
# define SPI_IDLE_SDA_DRIVE_LOW (0 << 18)
# define SPI_IDLE_SDA_DRIVE_HIGH (1 << 18)
# define SPI_IDLE_SDA_PULL_LOW (2 << 18)
# define SPI_IDLE_SDA_PULL_HIGH (3 << 18)
# define SPI_IDLE_SDA_MASK (3 << 18)
2017-10-05 14:22:36 +03:00
# define SPI_CS_SW_VAL (1 << 20)
2013-02-22 16:37:39 +04:00
# define SPI_CS_SW_HW (1 << 21)
/* SPI_CS_POL_INACTIVE bits are default high */
2013-12-08 19:35:09 +04:00
/* n from 0 to 3 */
# define SPI_CS_POL_INACTIVE(n) (1 << (22 + (n)))
2013-02-22 16:37:39 +04:00
# define SPI_CS_POL_INACTIVE_MASK (0xF << 22)
# define SPI_CS_SEL_0 (0 << 26)
# define SPI_CS_SEL_1 (1 << 26)
# define SPI_CS_SEL_2 (2 << 26)
# define SPI_CS_SEL_3 (3 << 26)
# define SPI_CS_SEL_MASK (3 << 26)
# define SPI_CS_SEL(x) (((x) & 0x3) << 26)
# define SPI_CONTROL_MODE_0 (0 << 28)
# define SPI_CONTROL_MODE_1 (1 << 28)
# define SPI_CONTROL_MODE_2 (2 << 28)
# define SPI_CONTROL_MODE_3 (3 << 28)
# define SPI_CONTROL_MODE_MASK (3 << 28)
# define SPI_MODE_SEL(x) (((x) & 0x3) << 28)
# define SPI_M_S (1 << 30)
# define SPI_PIO (1 << 31)
# define SPI_COMMAND2 0x004
# define SPI_TX_TAP_DELAY(x) (((x) & 0x3F) << 6)
# define SPI_RX_TAP_DELAY(x) (((x) & 0x3F) << 0)
# define SPI_CS_TIMING1 0x008
# define SPI_SETUP_HOLD(setup, hold) (((setup) << 4) | (hold))
# define SPI_CS_SETUP_HOLD(reg, cs, val) \
( ( ( ( val ) & 0xFFu ) < < ( ( cs ) * 8 ) ) | \
( ( reg ) & ~ ( 0xFFu < < ( ( cs ) * 8 ) ) ) )
# define SPI_CS_TIMING2 0x00C
# define CYCLES_BETWEEN_PACKETS_0(x) (((x) & 0x1F) << 0)
# define CS_ACTIVE_BETWEEN_PACKETS_0 (1 << 5)
# define CYCLES_BETWEEN_PACKETS_1(x) (((x) & 0x1F) << 8)
# define CS_ACTIVE_BETWEEN_PACKETS_1 (1 << 13)
# define CYCLES_BETWEEN_PACKETS_2(x) (((x) & 0x1F) << 16)
# define CS_ACTIVE_BETWEEN_PACKETS_2 (1 << 21)
# define CYCLES_BETWEEN_PACKETS_3(x) (((x) & 0x1F) << 24)
# define CS_ACTIVE_BETWEEN_PACKETS_3 (1 << 29)
# define SPI_SET_CS_ACTIVE_BETWEEN_PACKETS(reg, cs, val) \
( reg = ( ( ( val ) & 0x1 ) < < ( ( cs ) * 8 + 5 ) ) | \
( ( reg ) & ~ ( 1 < < ( ( cs ) * 8 + 5 ) ) ) )
# define SPI_SET_CYCLES_BETWEEN_PACKETS(reg, cs, val) \
2019-05-14 08:03:54 +03:00
( reg = ( ( ( val ) & 0x1F ) < < ( ( cs ) * 8 ) ) | \
( ( reg ) & ~ ( 0x1F < < ( ( cs ) * 8 ) ) ) )
# define MAX_SETUP_HOLD_CYCLES 16
# define MAX_INACTIVE_CYCLES 32
2013-02-22 16:37:39 +04:00
# define SPI_TRANS_STATUS 0x010
# define SPI_BLK_CNT(val) (((val) >> 0) & 0xFFFF)
# define SPI_SLV_IDLE_COUNT(val) (((val) >> 16) & 0xFF)
# define SPI_RDY (1 << 30)
# define SPI_FIFO_STATUS 0x014
# define SPI_RX_FIFO_EMPTY (1 << 0)
# define SPI_RX_FIFO_FULL (1 << 1)
# define SPI_TX_FIFO_EMPTY (1 << 2)
# define SPI_TX_FIFO_FULL (1 << 3)
# define SPI_RX_FIFO_UNF (1 << 4)
# define SPI_RX_FIFO_OVF (1 << 5)
# define SPI_TX_FIFO_UNF (1 << 6)
# define SPI_TX_FIFO_OVF (1 << 7)
# define SPI_ERR (1 << 8)
# define SPI_TX_FIFO_FLUSH (1 << 14)
# define SPI_RX_FIFO_FLUSH (1 << 15)
# define SPI_TX_FIFO_EMPTY_COUNT(val) (((val) >> 16) & 0x7F)
# define SPI_RX_FIFO_FULL_COUNT(val) (((val) >> 23) & 0x7F)
# define SPI_FRAME_END (1 << 30)
# define SPI_CS_INACTIVE (1 << 31)
# define SPI_FIFO_ERROR (SPI_RX_FIFO_UNF | \
SPI_RX_FIFO_OVF | SPI_TX_FIFO_UNF | SPI_TX_FIFO_OVF )
# define SPI_FIFO_EMPTY (SPI_RX_FIFO_EMPTY | SPI_TX_FIFO_EMPTY)
# define SPI_TX_DATA 0x018
# define SPI_RX_DATA 0x01C
# define SPI_DMA_CTL 0x020
# define SPI_TX_TRIG_1 (0 << 15)
# define SPI_TX_TRIG_4 (1 << 15)
# define SPI_TX_TRIG_8 (2 << 15)
# define SPI_TX_TRIG_16 (3 << 15)
# define SPI_TX_TRIG_MASK (3 << 15)
# define SPI_RX_TRIG_1 (0 << 19)
# define SPI_RX_TRIG_4 (1 << 19)
# define SPI_RX_TRIG_8 (2 << 19)
# define SPI_RX_TRIG_16 (3 << 19)
# define SPI_RX_TRIG_MASK (3 << 19)
# define SPI_IE_TX (1 << 28)
# define SPI_IE_RX (1 << 29)
# define SPI_CONT (1 << 30)
# define SPI_DMA (1 << 31)
# define SPI_DMA_EN SPI_DMA
# define SPI_DMA_BLK 0x024
# define SPI_DMA_BLK_SET(x) (((x) & 0xFFFF) << 0)
# define SPI_TX_FIFO 0x108
# define SPI_RX_FIFO 0x188
2019-04-05 03:14:12 +03:00
# define SPI_INTR_MASK 0x18c
# define SPI_INTR_ALL_MASK (0x1fUL << 25)
2013-02-22 16:37:39 +04:00
# define MAX_CHIP_SELECT 4
# define SPI_FIFO_DEPTH 64
# define DATA_DIR_TX (1 << 0)
# define DATA_DIR_RX (1 << 1)
# define SPI_DMA_TIMEOUT (msecs_to_jiffies(1000))
# define DEFAULT_SPI_DMA_BUF_LEN (16*1024)
# define TX_FIFO_EMPTY_COUNT_MAX SPI_TX_FIFO_EMPTY_COUNT(0x40)
# define RX_FIFO_FULL_COUNT_ZERO SPI_RX_FIFO_FULL_COUNT(0)
# define MAX_HOLD_CYCLES 16
# define SPI_DEFAULT_SPEED 25000000
2019-04-05 03:14:12 +03:00
struct tegra_spi_soc_data {
bool has_intr_mask_reg ;
} ;
2019-05-14 08:03:55 +03:00
struct tegra_spi_client_data {
int tx_clk_tap_delay ;
int rx_clk_tap_delay ;
} ;
2013-02-22 16:37:39 +04:00
struct tegra_spi_data {
struct device * dev ;
struct spi_master * master ;
spinlock_t lock ;
struct clk * clk ;
2013-11-07 03:31:24 +04:00
struct reset_control * rst ;
2013-02-22 16:37:39 +04:00
void __iomem * base ;
phys_addr_t phys ;
unsigned irq ;
u32 cur_speed ;
struct spi_device * cur_spi ;
2013-09-26 21:01:43 +04:00
struct spi_device * cs_control ;
2013-02-22 16:37:39 +04:00
unsigned cur_pos ;
unsigned words_per_32bit ;
unsigned bytes_per_word ;
unsigned curr_dma_words ;
unsigned cur_direction ;
unsigned cur_rx_pos ;
unsigned cur_tx_pos ;
unsigned dma_buf_size ;
unsigned max_buf_size ;
bool is_curr_dma_xfer ;
2019-05-14 08:03:53 +03:00
bool use_hw_based_cs ;
2013-02-22 16:37:39 +04:00
struct completion rx_dma_complete ;
struct completion tx_dma_complete ;
u32 tx_status ;
u32 rx_status ;
u32 status_reg ;
bool is_packed ;
u32 command1_reg ;
u32 dma_control_reg ;
u32 def_command1_reg ;
2019-05-14 08:03:55 +03:00
u32 def_command2_reg ;
2019-05-14 08:03:54 +03:00
u32 spi_cs_timing1 ;
u32 spi_cs_timing2 ;
2019-05-14 08:03:55 +03:00
u8 last_used_cs ;
2013-02-22 16:37:39 +04:00
struct completion xfer_completion ;
struct spi_transfer * curr_xfer ;
struct dma_chan * rx_dma_chan ;
u32 * rx_dma_buf ;
dma_addr_t rx_dma_phys ;
struct dma_async_tx_descriptor * rx_dma_desc ;
struct dma_chan * tx_dma_chan ;
u32 * tx_dma_buf ;
dma_addr_t tx_dma_phys ;
struct dma_async_tx_descriptor * tx_dma_desc ;
2019-04-05 03:14:12 +03:00
const struct tegra_spi_soc_data * soc_data ;
2013-02-22 16:37:39 +04:00
} ;
static int tegra_spi_runtime_suspend ( struct device * dev ) ;
static int tegra_spi_runtime_resume ( struct device * dev ) ;
2013-12-08 19:35:09 +04:00
static inline u32 tegra_spi_readl ( struct tegra_spi_data * tspi ,
2013-02-22 16:37:39 +04:00
unsigned long reg )
{
return readl ( tspi - > base + reg ) ;
}
static inline void tegra_spi_writel ( struct tegra_spi_data * tspi ,
2013-12-08 19:35:09 +04:00
u32 val , unsigned long reg )
2013-02-22 16:37:39 +04:00
{
writel ( val , tspi - > base + reg ) ;
/* Read back register to make sure that register writes completed */
if ( reg ! = SPI_TX_FIFO )
readl ( tspi - > base + SPI_COMMAND1 ) ;
}
static void tegra_spi_clear_status ( struct tegra_spi_data * tspi )
{
2013-12-08 19:35:09 +04:00
u32 val ;
2013-02-22 16:37:39 +04:00
/* Write 1 to clear status register */
val = tegra_spi_readl ( tspi , SPI_TRANS_STATUS ) ;
tegra_spi_writel ( tspi , val , SPI_TRANS_STATUS ) ;
/* Clear fifo status error if any */
val = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
if ( val & SPI_ERR )
tegra_spi_writel ( tspi , SPI_ERR | SPI_FIFO_ERROR ,
SPI_FIFO_STATUS ) ;
}
static unsigned tegra_spi_calculate_curr_xfer_param (
struct spi_device * spi , struct tegra_spi_data * tspi ,
struct spi_transfer * t )
{
unsigned remain_len = t - > len - tspi - > cur_pos ;
unsigned max_word ;
unsigned bits_per_word = t - > bits_per_word ;
unsigned max_len ;
unsigned total_fifo_words ;
2013-08-30 07:00:23 +04:00
tspi - > bytes_per_word = DIV_ROUND_UP ( bits_per_word , 8 ) ;
2013-02-22 16:37:39 +04:00
2019-04-05 03:14:01 +03:00
if ( ( bits_per_word = = 8 | | bits_per_word = = 16 | |
bits_per_word = = 32 ) & & t - > len > 3 ) {
2019-12-24 06:52:06 +03:00
tspi - > is_packed = true ;
2013-02-22 16:37:39 +04:00
tspi - > words_per_32bit = 32 / bits_per_word ;
} else {
2019-12-24 06:52:06 +03:00
tspi - > is_packed = false ;
2013-02-22 16:37:39 +04:00
tspi - > words_per_32bit = 1 ;
}
if ( tspi - > is_packed ) {
max_len = min ( remain_len , tspi - > max_buf_size ) ;
tspi - > curr_dma_words = max_len / tspi - > bytes_per_word ;
total_fifo_words = ( max_len + 3 ) / 4 ;
} else {
max_word = ( remain_len - 1 ) / tspi - > bytes_per_word + 1 ;
max_word = min ( max_word , tspi - > max_buf_size / 4 ) ;
tspi - > curr_dma_words = max_word ;
total_fifo_words = max_word ;
}
return total_fifo_words ;
}
static unsigned tegra_spi_fill_tx_fifo_from_client_txbuf (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
unsigned nbytes ;
unsigned tx_empty_count ;
2013-12-08 19:35:09 +04:00
u32 fifo_status ;
2013-02-22 16:37:39 +04:00
unsigned max_n_32bit ;
unsigned i , count ;
unsigned int written_words ;
unsigned fifo_words_left ;
u8 * tx_buf = ( u8 * ) t - > tx_buf + tspi - > cur_tx_pos ;
fifo_status = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
tx_empty_count = SPI_TX_FIFO_EMPTY_COUNT ( fifo_status ) ;
if ( tspi - > is_packed ) {
fifo_words_left = tx_empty_count * tspi - > words_per_32bit ;
written_words = min ( fifo_words_left , tspi - > curr_dma_words ) ;
nbytes = written_words * tspi - > bytes_per_word ;
max_n_32bit = DIV_ROUND_UP ( nbytes , 4 ) ;
for ( count = 0 ; count < max_n_32bit ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = 0 ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
for ( i = 0 ; ( i < 4 ) & & nbytes ; i + + , nbytes - - )
2013-12-08 19:35:09 +04:00
x | = ( u32 ) ( * tx_buf + + ) < < ( i * 8 ) ;
2013-02-22 16:37:39 +04:00
tegra_spi_writel ( tspi , x , SPI_TX_FIFO ) ;
}
2019-03-27 08:56:24 +03:00
tspi - > cur_tx_pos + = written_words * tspi - > bytes_per_word ;
2013-02-22 16:37:39 +04:00
} else {
2019-03-27 08:56:24 +03:00
unsigned int write_bytes ;
2013-02-22 16:37:39 +04:00
max_n_32bit = min ( tspi - > curr_dma_words , tx_empty_count ) ;
written_words = max_n_32bit ;
nbytes = written_words * tspi - > bytes_per_word ;
2019-03-27 08:56:24 +03:00
if ( nbytes > t - > len - tspi - > cur_pos )
nbytes = t - > len - tspi - > cur_pos ;
write_bytes = nbytes ;
2013-02-22 16:37:39 +04:00
for ( count = 0 ; count < max_n_32bit ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = 0 ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
for ( i = 0 ; nbytes & & ( i < tspi - > bytes_per_word ) ;
i + + , nbytes - - )
2013-12-08 19:35:09 +04:00
x | = ( u32 ) ( * tx_buf + + ) < < ( i * 8 ) ;
2013-02-22 16:37:39 +04:00
tegra_spi_writel ( tspi , x , SPI_TX_FIFO ) ;
}
2019-03-27 08:56:24 +03:00
tspi - > cur_tx_pos + = write_bytes ;
2013-02-22 16:37:39 +04:00
}
2019-03-27 08:56:24 +03:00
2013-02-22 16:37:39 +04:00
return written_words ;
}
static unsigned int tegra_spi_read_rx_fifo_to_client_rxbuf (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
unsigned rx_full_count ;
2013-12-08 19:35:09 +04:00
u32 fifo_status ;
2013-02-22 16:37:39 +04:00
unsigned i , count ;
unsigned int read_words = 0 ;
unsigned len ;
u8 * rx_buf = ( u8 * ) t - > rx_buf + tspi - > cur_rx_pos ;
fifo_status = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
rx_full_count = SPI_RX_FIFO_FULL_COUNT ( fifo_status ) ;
if ( tspi - > is_packed ) {
len = tspi - > curr_dma_words * tspi - > bytes_per_word ;
for ( count = 0 ; count < rx_full_count ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = tegra_spi_readl ( tspi , SPI_RX_FIFO ) ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
for ( i = 0 ; len & & ( i < 4 ) ; i + + , len - - )
* rx_buf + + = ( x > > i * 8 ) & 0xFF ;
}
read_words + = tspi - > curr_dma_words ;
2019-03-27 08:56:24 +03:00
tspi - > cur_rx_pos + = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2013-02-22 16:37:39 +04:00
} else {
2013-12-08 19:35:09 +04:00
u32 rx_mask = ( ( u32 ) 1 < < t - > bits_per_word ) - 1 ;
2019-03-27 08:56:24 +03:00
u8 bytes_per_word = tspi - > bytes_per_word ;
unsigned int read_bytes ;
2014-09-02 06:52:23 +04:00
2019-03-27 08:56:24 +03:00
len = rx_full_count * bytes_per_word ;
if ( len > t - > len - tspi - > cur_pos )
len = t - > len - tspi - > cur_pos ;
read_bytes = len ;
2013-02-22 16:37:39 +04:00
for ( count = 0 ; count < rx_full_count ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = tegra_spi_readl ( tspi , SPI_RX_FIFO ) & rx_mask ;
2014-09-02 06:52:23 +04:00
2019-03-27 08:56:24 +03:00
for ( i = 0 ; len & & ( i < bytes_per_word ) ; i + + , len - - )
2013-02-22 16:37:39 +04:00
* rx_buf + + = ( x > > ( i * 8 ) ) & 0xFF ;
}
read_words + = rx_full_count ;
2019-03-27 08:56:24 +03:00
tspi - > cur_rx_pos + = read_bytes ;
2013-02-22 16:37:39 +04:00
}
2019-03-27 08:56:24 +03:00
2013-02-22 16:37:39 +04:00
return read_words ;
}
static void tegra_spi_copy_client_txbuf_to_spi_txbuf (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
/* Make the dma buffer to read by cpu */
dma_sync_single_for_cpu ( tspi - > dev , tspi - > tx_dma_phys ,
tspi - > dma_buf_size , DMA_TO_DEVICE ) ;
if ( tspi - > is_packed ) {
2013-12-08 19:35:09 +04:00
unsigned len = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
memcpy ( tspi - > tx_dma_buf , t - > tx_buf + tspi - > cur_pos , len ) ;
2019-03-27 08:56:24 +03:00
tspi - > cur_tx_pos + = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2013-02-22 16:37:39 +04:00
} else {
unsigned int i ;
unsigned int count ;
u8 * tx_buf = ( u8 * ) t - > tx_buf + tspi - > cur_tx_pos ;
unsigned consume = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2019-03-27 08:56:24 +03:00
unsigned int write_bytes ;
2013-02-22 16:37:39 +04:00
2019-03-27 08:56:24 +03:00
if ( consume > t - > len - tspi - > cur_pos )
consume = t - > len - tspi - > cur_pos ;
write_bytes = consume ;
2013-02-22 16:37:39 +04:00
for ( count = 0 ; count < tspi - > curr_dma_words ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = 0 ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
for ( i = 0 ; consume & & ( i < tspi - > bytes_per_word ) ;
i + + , consume - - )
2013-12-08 19:35:09 +04:00
x | = ( u32 ) ( * tx_buf + + ) < < ( i * 8 ) ;
2013-02-22 16:37:39 +04:00
tspi - > tx_dma_buf [ count ] = x ;
}
2019-03-27 08:56:24 +03:00
tspi - > cur_tx_pos + = write_bytes ;
2013-02-22 16:37:39 +04:00
}
/* Make the dma buffer to read by dma */
dma_sync_single_for_device ( tspi - > dev , tspi - > tx_dma_phys ,
tspi - > dma_buf_size , DMA_TO_DEVICE ) ;
}
static void tegra_spi_copy_spi_rxbuf_to_client_rxbuf (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
/* Make the dma buffer to read by cpu */
dma_sync_single_for_cpu ( tspi - > dev , tspi - > rx_dma_phys ,
tspi - > dma_buf_size , DMA_FROM_DEVICE ) ;
if ( tspi - > is_packed ) {
2013-12-08 19:35:09 +04:00
unsigned len = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2014-09-02 06:52:23 +04:00
2013-02-22 16:37:39 +04:00
memcpy ( t - > rx_buf + tspi - > cur_rx_pos , tspi - > rx_dma_buf , len ) ;
2019-03-27 08:56:24 +03:00
tspi - > cur_rx_pos + = tspi - > curr_dma_words * tspi - > bytes_per_word ;
2013-02-22 16:37:39 +04:00
} else {
unsigned int i ;
unsigned int count ;
unsigned char * rx_buf = t - > rx_buf + tspi - > cur_rx_pos ;
2013-12-08 19:35:09 +04:00
u32 rx_mask = ( ( u32 ) 1 < < t - > bits_per_word ) - 1 ;
2019-03-27 08:56:24 +03:00
unsigned consume = tspi - > curr_dma_words * tspi - > bytes_per_word ;
unsigned int read_bytes ;
2013-02-22 16:37:39 +04:00
2019-03-27 08:56:24 +03:00
if ( consume > t - > len - tspi - > cur_pos )
consume = t - > len - tspi - > cur_pos ;
read_bytes = consume ;
2013-02-22 16:37:39 +04:00
for ( count = 0 ; count < tspi - > curr_dma_words ; count + + ) {
2013-12-08 19:35:09 +04:00
u32 x = tspi - > rx_dma_buf [ count ] & rx_mask ;
2014-09-02 06:52:23 +04:00
2019-03-27 08:56:24 +03:00
for ( i = 0 ; consume & & ( i < tspi - > bytes_per_word ) ;
i + + , consume - - )
2013-02-22 16:37:39 +04:00
* rx_buf + + = ( x > > ( i * 8 ) ) & 0xFF ;
}
2019-03-27 08:56:24 +03:00
tspi - > cur_rx_pos + = read_bytes ;
2013-02-22 16:37:39 +04:00
}
/* Make the dma buffer to read by dma */
dma_sync_single_for_device ( tspi - > dev , tspi - > rx_dma_phys ,
tspi - > dma_buf_size , DMA_FROM_DEVICE ) ;
}
static void tegra_spi_dma_complete ( void * args )
{
struct completion * dma_complete = args ;
complete ( dma_complete ) ;
}
static int tegra_spi_start_tx_dma ( struct tegra_spi_data * tspi , int len )
{
2013-11-15 02:32:02 +04:00
reinit_completion ( & tspi - > tx_dma_complete ) ;
2013-02-22 16:37:39 +04:00
tspi - > tx_dma_desc = dmaengine_prep_slave_single ( tspi - > tx_dma_chan ,
tspi - > tx_dma_phys , len , DMA_MEM_TO_DEV ,
DMA_PREP_INTERRUPT | DMA_CTRL_ACK ) ;
if ( ! tspi - > tx_dma_desc ) {
dev_err ( tspi - > dev , " Not able to get desc for Tx \n " ) ;
return - EIO ;
}
tspi - > tx_dma_desc - > callback = tegra_spi_dma_complete ;
tspi - > tx_dma_desc - > callback_param = & tspi - > tx_dma_complete ;
dmaengine_submit ( tspi - > tx_dma_desc ) ;
dma_async_issue_pending ( tspi - > tx_dma_chan ) ;
return 0 ;
}
static int tegra_spi_start_rx_dma ( struct tegra_spi_data * tspi , int len )
{
2013-11-15 02:32:02 +04:00
reinit_completion ( & tspi - > rx_dma_complete ) ;
2013-02-22 16:37:39 +04:00
tspi - > rx_dma_desc = dmaengine_prep_slave_single ( tspi - > rx_dma_chan ,
tspi - > rx_dma_phys , len , DMA_DEV_TO_MEM ,
DMA_PREP_INTERRUPT | DMA_CTRL_ACK ) ;
if ( ! tspi - > rx_dma_desc ) {
dev_err ( tspi - > dev , " Not able to get desc for Rx \n " ) ;
return - EIO ;
}
tspi - > rx_dma_desc - > callback = tegra_spi_dma_complete ;
tspi - > rx_dma_desc - > callback_param = & tspi - > rx_dma_complete ;
dmaengine_submit ( tspi - > rx_dma_desc ) ;
dma_async_issue_pending ( tspi - > rx_dma_chan ) ;
return 0 ;
}
2019-03-27 08:56:28 +03:00
static int tegra_spi_flush_fifos ( struct tegra_spi_data * tspi )
2013-02-22 16:37:39 +04:00
{
2019-03-27 08:56:28 +03:00
unsigned long timeout = jiffies + HZ ;
2013-12-08 19:35:09 +04:00
u32 status ;
2013-02-22 16:37:39 +04:00
status = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
if ( ( status & SPI_FIFO_EMPTY ) ! = SPI_FIFO_EMPTY ) {
2019-03-27 08:56:28 +03:00
status | = SPI_RX_FIFO_FLUSH | SPI_TX_FIFO_FLUSH ;
tegra_spi_writel ( tspi , status , SPI_FIFO_STATUS ) ;
while ( ( status & SPI_FIFO_EMPTY ) ! = SPI_FIFO_EMPTY ) {
status = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
if ( time_after ( jiffies , timeout ) ) {
dev_err ( tspi - > dev ,
" timeout waiting for fifo flush \n " ) ;
return - EIO ;
}
udelay ( 1 ) ;
}
2013-02-22 16:37:39 +04:00
}
2019-03-27 08:56:28 +03:00
return 0 ;
}
static int tegra_spi_start_dma_based_transfer (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
u32 val ;
unsigned int len ;
int ret = 0 ;
2019-03-27 08:56:29 +03:00
u8 dma_burst ;
struct dma_slave_config dma_sconfig = { 0 } ;
2019-03-27 08:56:28 +03:00
2013-02-22 16:37:39 +04:00
val = SPI_DMA_BLK_SET ( tspi - > curr_dma_words - 1 ) ;
tegra_spi_writel ( tspi , val , SPI_DMA_BLK ) ;
if ( tspi - > is_packed )
len = DIV_ROUND_UP ( tspi - > curr_dma_words * tspi - > bytes_per_word ,
4 ) * 4 ;
else
len = tspi - > curr_dma_words * 4 ;
/* Set attention level based on length of transfer */
2019-03-27 08:56:29 +03:00
if ( len & 0xF ) {
2013-02-22 16:37:39 +04:00
val | = SPI_TX_TRIG_1 | SPI_RX_TRIG_1 ;
2019-03-27 08:56:29 +03:00
dma_burst = 1 ;
} else if ( ( ( len ) > > 4 ) & 0x1 ) {
2013-02-22 16:37:39 +04:00
val | = SPI_TX_TRIG_4 | SPI_RX_TRIG_4 ;
2019-03-27 08:56:29 +03:00
dma_burst = 4 ;
} else {
2013-02-22 16:37:39 +04:00
val | = SPI_TX_TRIG_8 | SPI_RX_TRIG_8 ;
2019-03-27 08:56:29 +03:00
dma_burst = 8 ;
}
2013-02-22 16:37:39 +04:00
2019-04-05 03:14:12 +03:00
if ( ! tspi - > soc_data - > has_intr_mask_reg ) {
if ( tspi - > cur_direction & DATA_DIR_TX )
val | = SPI_IE_TX ;
2013-02-22 16:37:39 +04:00
2019-04-05 03:14:12 +03:00
if ( tspi - > cur_direction & DATA_DIR_RX )
val | = SPI_IE_RX ;
}
2013-02-22 16:37:39 +04:00
tegra_spi_writel ( tspi , val , SPI_DMA_CTL ) ;
tspi - > dma_control_reg = val ;
2019-03-27 08:56:29 +03:00
dma_sconfig . device_fc = true ;
2013-02-22 16:37:39 +04:00
if ( tspi - > cur_direction & DATA_DIR_TX ) {
2019-03-27 08:56:29 +03:00
dma_sconfig . dst_addr = tspi - > phys + SPI_TX_FIFO ;
dma_sconfig . dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
dma_sconfig . dst_maxburst = dma_burst ;
ret = dmaengine_slave_config ( tspi - > tx_dma_chan , & dma_sconfig ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev ,
" DMA slave config failed: %d \n " , ret ) ;
return ret ;
}
2013-02-22 16:37:39 +04:00
tegra_spi_copy_client_txbuf_to_spi_txbuf ( tspi , t ) ;
ret = tegra_spi_start_tx_dma ( tspi , len ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev ,
" Starting tx dma failed, err %d \n " , ret ) ;
return ret ;
}
}
if ( tspi - > cur_direction & DATA_DIR_RX ) {
2019-03-27 08:56:29 +03:00
dma_sconfig . src_addr = tspi - > phys + SPI_RX_FIFO ;
dma_sconfig . src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
dma_sconfig . src_maxburst = dma_burst ;
ret = dmaengine_slave_config ( tspi - > rx_dma_chan , & dma_sconfig ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev ,
" DMA slave config failed: %d \n " , ret ) ;
return ret ;
}
2013-02-22 16:37:39 +04:00
/* Make the dma buffer to read by dma */
dma_sync_single_for_device ( tspi - > dev , tspi - > rx_dma_phys ,
tspi - > dma_buf_size , DMA_FROM_DEVICE ) ;
ret = tegra_spi_start_rx_dma ( tspi , len ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev ,
" Starting rx dma failed, err %d \n " , ret ) ;
if ( tspi - > cur_direction & DATA_DIR_TX )
dmaengine_terminate_all ( tspi - > tx_dma_chan ) ;
return ret ;
}
}
tspi - > is_curr_dma_xfer = true ;
tspi - > dma_control_reg = val ;
val | = SPI_DMA_EN ;
tegra_spi_writel ( tspi , val , SPI_DMA_CTL ) ;
return ret ;
}
static int tegra_spi_start_cpu_based_transfer (
struct tegra_spi_data * tspi , struct spi_transfer * t )
{
2013-12-08 19:35:09 +04:00
u32 val ;
2013-02-22 16:37:39 +04:00
unsigned cur_words ;
if ( tspi - > cur_direction & DATA_DIR_TX )
cur_words = tegra_spi_fill_tx_fifo_from_client_txbuf ( tspi , t ) ;
else
cur_words = tspi - > curr_dma_words ;
val = SPI_DMA_BLK_SET ( cur_words - 1 ) ;
tegra_spi_writel ( tspi , val , SPI_DMA_BLK ) ;
val = 0 ;
if ( tspi - > cur_direction & DATA_DIR_TX )
val | = SPI_IE_TX ;
if ( tspi - > cur_direction & DATA_DIR_RX )
val | = SPI_IE_RX ;
tegra_spi_writel ( tspi , val , SPI_DMA_CTL ) ;
tspi - > dma_control_reg = val ;
tspi - > is_curr_dma_xfer = false ;
2019-04-16 00:30:26 +03:00
val = tspi - > command1_reg ;
val | = SPI_PIO ;
tegra_spi_writel ( tspi , val , SPI_COMMAND1 ) ;
2013-02-22 16:37:39 +04:00
return 0 ;
}
static int tegra_spi_init_dma_param ( struct tegra_spi_data * tspi ,
bool dma_to_memory )
{
struct dma_chan * dma_chan ;
u32 * dma_buf ;
dma_addr_t dma_phys ;
int ret ;
2019-11-13 12:42:55 +03:00
dma_chan = dma_request_chan ( tspi - > dev , dma_to_memory ? " rx " : " tx " ) ;
2013-11-12 00:13:47 +04:00
if ( IS_ERR ( dma_chan ) ) {
ret = PTR_ERR ( dma_chan ) ;
if ( ret ! = - EPROBE_DEFER )
dev_err ( tspi - > dev ,
" Dma channel is not available: %d \n " , ret ) ;
return ret ;
2013-02-22 16:37:39 +04:00
}
dma_buf = dma_alloc_coherent ( tspi - > dev , tspi - > dma_buf_size ,
& dma_phys , GFP_KERNEL ) ;
if ( ! dma_buf ) {
dev_err ( tspi - > dev , " Not able to allocate the dma buffer \n " ) ;
dma_release_channel ( dma_chan ) ;
return - ENOMEM ;
}
if ( dma_to_memory ) {
tspi - > rx_dma_chan = dma_chan ;
tspi - > rx_dma_buf = dma_buf ;
tspi - > rx_dma_phys = dma_phys ;
} else {
tspi - > tx_dma_chan = dma_chan ;
tspi - > tx_dma_buf = dma_buf ;
tspi - > tx_dma_phys = dma_phys ;
}
return 0 ;
}
static void tegra_spi_deinit_dma_param ( struct tegra_spi_data * tspi ,
bool dma_to_memory )
{
u32 * dma_buf ;
dma_addr_t dma_phys ;
struct dma_chan * dma_chan ;
if ( dma_to_memory ) {
dma_buf = tspi - > rx_dma_buf ;
dma_chan = tspi - > rx_dma_chan ;
dma_phys = tspi - > rx_dma_phys ;
tspi - > rx_dma_chan = NULL ;
tspi - > rx_dma_buf = NULL ;
} else {
dma_buf = tspi - > tx_dma_buf ;
dma_chan = tspi - > tx_dma_chan ;
dma_phys = tspi - > tx_dma_phys ;
tspi - > tx_dma_buf = NULL ;
tspi - > tx_dma_chan = NULL ;
}
if ( ! dma_chan )
return ;
dma_free_coherent ( tspi - > dev , tspi - > dma_buf_size , dma_buf , dma_phys ) ;
dma_release_channel ( dma_chan ) ;
}
2019-09-26 13:51:42 +03:00
static int tegra_spi_set_hw_cs_timing ( struct spi_device * spi ,
struct spi_delay * setup ,
struct spi_delay * hold ,
struct spi_delay * inactive )
2019-05-14 08:03:54 +03:00
{
struct tegra_spi_data * tspi = spi_master_get_devdata ( spi - > master ) ;
2019-09-26 13:51:42 +03:00
u8 setup_dly , hold_dly , inactive_dly ;
2019-05-14 08:03:54 +03:00
u32 setup_hold ;
u32 spi_cs_timing ;
u32 inactive_cycles ;
u8 cs_state ;
2019-09-26 13:51:42 +03:00
if ( ( setup & & setup - > unit ! = SPI_DELAY_UNIT_SCK ) | |
( hold & & hold - > unit ! = SPI_DELAY_UNIT_SCK ) | |
( inactive & & inactive - > unit ! = SPI_DELAY_UNIT_SCK ) ) {
dev_err ( & spi - > dev ,
" Invalid delay unit %d, should be SPI_DELAY_UNIT_SCK \n " ,
SPI_DELAY_UNIT_SCK ) ;
return - EINVAL ;
}
setup_dly = setup ? setup - > value : 0 ;
hold_dly = hold ? hold - > value : 0 ;
inactive_dly = inactive ? inactive - > value : 0 ;
2019-05-14 08:03:54 +03:00
setup_dly = min_t ( u8 , setup_dly , MAX_SETUP_HOLD_CYCLES ) ;
hold_dly = min_t ( u8 , hold_dly , MAX_SETUP_HOLD_CYCLES ) ;
if ( setup_dly & & hold_dly ) {
setup_hold = SPI_SETUP_HOLD ( setup_dly - 1 , hold_dly - 1 ) ;
spi_cs_timing = SPI_CS_SETUP_HOLD ( tspi - > spi_cs_timing1 ,
spi - > chip_select ,
setup_hold ) ;
if ( tspi - > spi_cs_timing1 ! = spi_cs_timing ) {
tspi - > spi_cs_timing1 = spi_cs_timing ;
tegra_spi_writel ( tspi , spi_cs_timing , SPI_CS_TIMING1 ) ;
}
}
inactive_cycles = min_t ( u8 , inactive_dly , MAX_INACTIVE_CYCLES ) ;
if ( inactive_cycles )
inactive_cycles - - ;
cs_state = inactive_cycles ? 0 : 1 ;
spi_cs_timing = tspi - > spi_cs_timing2 ;
SPI_SET_CS_ACTIVE_BETWEEN_PACKETS ( spi_cs_timing , spi - > chip_select ,
cs_state ) ;
SPI_SET_CYCLES_BETWEEN_PACKETS ( spi_cs_timing , spi - > chip_select ,
inactive_cycles ) ;
if ( tspi - > spi_cs_timing2 ! = spi_cs_timing ) {
tspi - > spi_cs_timing2 = spi_cs_timing ;
tegra_spi_writel ( tspi , spi_cs_timing , SPI_CS_TIMING2 ) ;
}
2019-09-26 13:51:42 +03:00
return 0 ;
2019-05-14 08:03:54 +03:00
}
2013-12-08 19:35:09 +04:00
static u32 tegra_spi_setup_transfer_one ( struct spi_device * spi ,
2019-05-14 08:03:53 +03:00
struct spi_transfer * t ,
bool is_first_of_msg ,
bool is_single_xfer )
2013-02-22 16:37:39 +04:00
{
struct tegra_spi_data * tspi = spi_master_get_devdata ( spi - > master ) ;
2019-05-14 08:03:55 +03:00
struct tegra_spi_client_data * cdata = spi - > controller_data ;
2013-02-22 16:37:39 +04:00
u32 speed = t - > speed_hz ;
u8 bits_per_word = t - > bits_per_word ;
2019-05-14 08:03:55 +03:00
u32 command1 , command2 ;
2013-02-22 16:37:39 +04:00
int req_mode ;
2019-05-14 08:03:55 +03:00
u32 tx_tap = 0 , rx_tap = 0 ;
2013-02-22 16:37:39 +04:00
if ( speed ! = tspi - > cur_speed ) {
clk_set_rate ( tspi - > clk , speed ) ;
tspi - > cur_speed = speed ;
}
tspi - > cur_spi = spi ;
tspi - > cur_pos = 0 ;
tspi - > cur_rx_pos = 0 ;
tspi - > cur_tx_pos = 0 ;
tspi - > curr_xfer = t ;
if ( is_first_of_msg ) {
tegra_spi_clear_status ( tspi ) ;
command1 = tspi - > def_command1_reg ;
command1 | = SPI_BIT_LENGTH ( bits_per_word - 1 ) ;
command1 & = ~ SPI_CONTROL_MODE_MASK ;
req_mode = spi - > mode & 0x3 ;
if ( req_mode = = SPI_MODE_0 )
command1 | = SPI_CONTROL_MODE_0 ;
else if ( req_mode = = SPI_MODE_1 )
command1 | = SPI_CONTROL_MODE_1 ;
else if ( req_mode = = SPI_MODE_2 )
command1 | = SPI_CONTROL_MODE_2 ;
else if ( req_mode = = SPI_MODE_3 )
command1 | = SPI_CONTROL_MODE_3 ;
2019-03-27 08:56:33 +03:00
if ( spi - > mode & SPI_LSB_FIRST )
command1 | = SPI_LSBIT_FE ;
else
command1 & = ~ SPI_LSBIT_FE ;
2019-04-05 03:14:08 +03:00
if ( spi - > mode & SPI_3WIRE )
command1 | = SPI_BIDIROE ;
else
command1 & = ~ SPI_BIDIROE ;
2013-09-26 21:01:43 +04:00
if ( tspi - > cs_control ) {
if ( tspi - > cs_control ! = spi )
tegra_spi_writel ( tspi , command1 , SPI_COMMAND1 ) ;
tspi - > cs_control = NULL ;
} else
tegra_spi_writel ( tspi , command1 , SPI_COMMAND1 ) ;
2013-02-22 16:37:39 +04:00
2019-05-14 08:03:52 +03:00
/* GPIO based chip select control */
if ( spi - > cs_gpiod )
gpiod_set_value ( spi - > cs_gpiod , 1 ) ;
2019-05-14 08:03:53 +03:00
if ( is_single_xfer & & ! ( t - > cs_change ) ) {
tspi - > use_hw_based_cs = true ;
command1 & = ~ ( SPI_CS_SW_HW | SPI_CS_SW_VAL ) ;
} else {
tspi - > use_hw_based_cs = false ;
command1 | = SPI_CS_SW_HW ;
if ( spi - > mode & SPI_CS_HIGH )
command1 | = SPI_CS_SW_VAL ;
else
command1 & = ~ SPI_CS_SW_VAL ;
}
2013-02-22 16:37:39 +04:00
2019-05-14 08:03:55 +03:00
if ( tspi - > last_used_cs ! = spi - > chip_select ) {
if ( cdata & & cdata - > tx_clk_tap_delay )
tx_tap = cdata - > tx_clk_tap_delay ;
if ( cdata & & cdata - > rx_clk_tap_delay )
rx_tap = cdata - > rx_clk_tap_delay ;
command2 = SPI_TX_TAP_DELAY ( tx_tap ) |
SPI_RX_TAP_DELAY ( rx_tap ) ;
if ( command2 ! = tspi - > def_command2_reg )
tegra_spi_writel ( tspi , command2 , SPI_COMMAND2 ) ;
tspi - > last_used_cs = spi - > chip_select ;
}
2013-02-22 16:37:39 +04:00
} else {
command1 = tspi - > command1_reg ;
command1 & = ~ SPI_BIT_LENGTH ( ~ 0 ) ;
command1 | = SPI_BIT_LENGTH ( bits_per_word - 1 ) ;
}
2013-09-26 21:01:43 +04:00
return command1 ;
}
static int tegra_spi_start_transfer_one ( struct spi_device * spi ,
2013-12-08 19:35:09 +04:00
struct spi_transfer * t , u32 command1 )
2013-09-26 21:01:43 +04:00
{
struct tegra_spi_data * tspi = spi_master_get_devdata ( spi - > master ) ;
unsigned total_fifo_words ;
int ret ;
total_fifo_words = tegra_spi_calculate_curr_xfer_param ( spi , tspi , t ) ;
2019-04-05 03:14:07 +03:00
if ( t - > rx_nbits = = SPI_NBITS_DUAL | | t - > tx_nbits = = SPI_NBITS_DUAL )
command1 | = SPI_BOTH_EN_BIT ;
else
command1 & = ~ SPI_BOTH_EN_BIT ;
2013-02-22 16:37:39 +04:00
if ( tspi - > is_packed )
command1 | = SPI_PACKED ;
2019-03-27 08:56:23 +03:00
else
command1 & = ~ SPI_PACKED ;
2013-02-22 16:37:39 +04:00
command1 & = ~ ( SPI_CS_SEL_MASK | SPI_TX_EN | SPI_RX_EN ) ;
tspi - > cur_direction = 0 ;
if ( t - > rx_buf ) {
command1 | = SPI_RX_EN ;
tspi - > cur_direction | = DATA_DIR_RX ;
}
if ( t - > tx_buf ) {
command1 | = SPI_TX_EN ;
tspi - > cur_direction | = DATA_DIR_TX ;
}
command1 | = SPI_CS_SEL ( spi - > chip_select ) ;
tegra_spi_writel ( tspi , command1 , SPI_COMMAND1 ) ;
tspi - > command1_reg = command1 ;
2013-12-08 19:35:09 +04:00
dev_dbg ( tspi - > dev , " The def 0x%x and written 0x%x \n " ,
tspi - > def_command1_reg , ( unsigned ) command1 ) ;
2013-02-22 16:37:39 +04:00
2019-03-27 08:56:28 +03:00
ret = tegra_spi_flush_fifos ( tspi ) ;
if ( ret < 0 )
return ret ;
2013-02-22 16:37:39 +04:00
if ( total_fifo_words > SPI_FIFO_DEPTH )
ret = tegra_spi_start_dma_based_transfer ( tspi , t ) ;
else
ret = tegra_spi_start_cpu_based_transfer ( tspi , t ) ;
return ret ;
}
2019-05-14 08:03:55 +03:00
static struct tegra_spi_client_data
* tegra_spi_parse_cdata_dt ( struct spi_device * spi )
{
struct tegra_spi_client_data * cdata ;
struct device_node * slave_np ;
slave_np = spi - > dev . of_node ;
if ( ! slave_np ) {
dev_dbg ( & spi - > dev , " device node not found \n " ) ;
return NULL ;
}
cdata = kzalloc ( sizeof ( * cdata ) , GFP_KERNEL ) ;
if ( ! cdata )
return NULL ;
of_property_read_u32 ( slave_np , " nvidia,tx-clk-tap-delay " ,
& cdata - > tx_clk_tap_delay ) ;
of_property_read_u32 ( slave_np , " nvidia,rx-clk-tap-delay " ,
& cdata - > rx_clk_tap_delay ) ;
return cdata ;
}
static void tegra_spi_cleanup ( struct spi_device * spi )
{
struct tegra_spi_client_data * cdata = spi - > controller_data ;
spi - > controller_data = NULL ;
if ( spi - > dev . of_node )
kfree ( cdata ) ;
}
2013-02-22 16:37:39 +04:00
static int tegra_spi_setup ( struct spi_device * spi )
{
struct tegra_spi_data * tspi = spi_master_get_devdata ( spi - > master ) ;
2019-05-14 08:03:55 +03:00
struct tegra_spi_client_data * cdata = spi - > controller_data ;
2013-12-08 19:35:09 +04:00
u32 val ;
2013-02-22 16:37:39 +04:00
unsigned long flags ;
int ret ;
dev_dbg ( & spi - > dev , " setup %d bpw, %scpol, %scpha, %dHz \n " ,
spi - > bits_per_word ,
spi - > mode & SPI_CPOL ? " " : " ~ " ,
spi - > mode & SPI_CPHA ? " " : " ~ " ,
spi - > max_speed_hz ) ;
2019-05-14 08:03:55 +03:00
if ( ! cdata ) {
cdata = tegra_spi_parse_cdata_dt ( spi ) ;
spi - > controller_data = cdata ;
}
2013-02-22 16:37:39 +04:00
ret = pm_runtime_get_sync ( tspi - > dev ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev , " pm runtime failed, e = %d \n " , ret ) ;
2019-05-23 04:29:04 +03:00
if ( cdata )
tegra_spi_cleanup ( spi ) ;
2013-02-22 16:37:39 +04:00
return ret ;
}
2019-04-05 03:14:12 +03:00
if ( tspi - > soc_data - > has_intr_mask_reg ) {
val = tegra_spi_readl ( tspi , SPI_INTR_MASK ) ;
val & = ~ SPI_INTR_ALL_MASK ;
tegra_spi_writel ( tspi , val , SPI_INTR_MASK ) ;
}
2013-02-22 16:37:39 +04:00
spin_lock_irqsave ( & tspi - > lock , flags ) ;
2019-05-14 08:03:52 +03:00
/* GPIO based chip select control */
if ( spi - > cs_gpiod )
gpiod_set_value ( spi - > cs_gpiod , 0 ) ;
2013-02-22 16:37:39 +04:00
val = tspi - > def_command1_reg ;
if ( spi - > mode & SPI_CS_HIGH )
2013-12-08 19:35:09 +04:00
val & = ~ SPI_CS_POL_INACTIVE ( spi - > chip_select ) ;
2013-02-22 16:37:39 +04:00
else
2013-12-08 19:35:09 +04:00
val | = SPI_CS_POL_INACTIVE ( spi - > chip_select ) ;
2013-02-22 16:37:39 +04:00
tspi - > def_command1_reg = val ;
tegra_spi_writel ( tspi , tspi - > def_command1_reg , SPI_COMMAND1 ) ;
spin_unlock_irqrestore ( & tspi - > lock , flags ) ;
pm_runtime_put ( tspi - > dev ) ;
return 0 ;
}
2019-04-05 03:14:02 +03:00
static void tegra_spi_transfer_end ( struct spi_device * spi )
{
struct tegra_spi_data * tspi = spi_master_get_devdata ( spi - > master ) ;
int cs_val = ( spi - > mode & SPI_CS_HIGH ) ? 0 : 1 ;
2019-05-14 08:03:52 +03:00
/* GPIO based chip select control */
if ( spi - > cs_gpiod )
gpiod_set_value ( spi - > cs_gpiod , 0 ) ;
2019-05-14 08:03:53 +03:00
if ( ! tspi - > use_hw_based_cs ) {
if ( cs_val )
tspi - > command1_reg | = SPI_CS_SW_VAL ;
else
tspi - > command1_reg & = ~ SPI_CS_SW_VAL ;
tegra_spi_writel ( tspi , tspi - > command1_reg , SPI_COMMAND1 ) ;
}
2019-04-05 03:14:02 +03:00
tegra_spi_writel ( tspi , tspi - > def_command1_reg , SPI_COMMAND1 ) ;
}
2019-04-05 03:14:04 +03:00
static void tegra_spi_dump_regs ( struct tegra_spi_data * tspi )
{
dev_dbg ( tspi - > dev , " ============ SPI REGISTER DUMP ============ \n " ) ;
dev_dbg ( tspi - > dev , " Command1: 0x%08x | Command2: 0x%08x \n " ,
tegra_spi_readl ( tspi , SPI_COMMAND1 ) ,
tegra_spi_readl ( tspi , SPI_COMMAND2 ) ) ;
dev_dbg ( tspi - > dev , " DMA_CTL: 0x%08x | DMA_BLK: 0x%08x \n " ,
tegra_spi_readl ( tspi , SPI_DMA_CTL ) ,
tegra_spi_readl ( tspi , SPI_DMA_BLK ) ) ;
dev_dbg ( tspi - > dev , " TRANS_STAT: 0x%08x | FIFO_STATUS: 0x%08x \n " ,
tegra_spi_readl ( tspi , SPI_TRANS_STATUS ) ,
tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ) ;
}
2013-02-22 16:37:39 +04:00
static int tegra_spi_transfer_one_message ( struct spi_master * master ,
struct spi_message * msg )
{
bool is_first_msg = true ;
struct tegra_spi_data * tspi = spi_master_get_devdata ( master ) ;
struct spi_transfer * xfer ;
struct spi_device * spi = msg - > spi ;
int ret ;
2013-09-26 21:01:43 +04:00
bool skip = false ;
2019-05-14 08:03:53 +03:00
int single_xfer ;
2013-02-22 16:37:39 +04:00
msg - > status = 0 ;
msg - > actual_length = 0 ;
2019-05-14 08:03:53 +03:00
single_xfer = list_is_singular ( & msg - > transfers ) ;
2013-02-22 16:37:39 +04:00
list_for_each_entry ( xfer , & msg - > transfers , transfer_list ) {
2013-12-08 19:35:09 +04:00
u32 cmd1 ;
2013-09-26 21:01:43 +04:00
2013-11-15 02:32:02 +04:00
reinit_completion ( & tspi - > xfer_completion ) ;
2013-09-26 21:01:43 +04:00
2019-05-14 08:03:53 +03:00
cmd1 = tegra_spi_setup_transfer_one ( spi , xfer , is_first_msg ,
single_xfer ) ;
2013-09-26 21:01:43 +04:00
if ( ! xfer - > len ) {
ret = 0 ;
skip = true ;
goto complete_xfer ;
}
ret = tegra_spi_start_transfer_one ( spi , xfer , cmd1 ) ;
2013-02-22 16:37:39 +04:00
if ( ret < 0 ) {
dev_err ( tspi - > dev ,
" spi can not start transfer, err %d \n " , ret ) ;
2013-09-26 21:01:43 +04:00
goto complete_xfer ;
2013-02-22 16:37:39 +04:00
}
2013-09-26 21:01:43 +04:00
2013-02-22 16:37:39 +04:00
is_first_msg = false ;
ret = wait_for_completion_timeout ( & tspi - > xfer_completion ,
SPI_DMA_TIMEOUT ) ;
if ( WARN_ON ( ret = = 0 ) ) {
dev_err ( tspi - > dev ,
2017-04-23 20:14:36 +03:00
" spi transfer timeout, err %d \n " , ret ) ;
2019-03-27 08:56:27 +03:00
if ( tspi - > is_curr_dma_xfer & &
( tspi - > cur_direction & DATA_DIR_TX ) )
dmaengine_terminate_all ( tspi - > tx_dma_chan ) ;
if ( tspi - > is_curr_dma_xfer & &
( tspi - > cur_direction & DATA_DIR_RX ) )
dmaengine_terminate_all ( tspi - > rx_dma_chan ) ;
2013-02-22 16:37:39 +04:00
ret = - EIO ;
2019-04-05 03:14:04 +03:00
tegra_spi_dump_regs ( tspi ) ;
2019-03-27 08:56:28 +03:00
tegra_spi_flush_fifos ( tspi ) ;
2019-03-27 08:56:27 +03:00
reset_control_assert ( tspi - > rst ) ;
udelay ( 2 ) ;
reset_control_deassert ( tspi - > rst ) ;
2019-05-14 08:03:55 +03:00
tspi - > last_used_cs = master - > num_chipselect + 1 ;
2013-09-26 21:01:43 +04:00
goto complete_xfer ;
2013-02-22 16:37:39 +04:00
}
if ( tspi - > tx_status | | tspi - > rx_status ) {
dev_err ( tspi - > dev , " Error in Transfer \n " ) ;
ret = - EIO ;
2019-04-05 03:14:04 +03:00
tegra_spi_dump_regs ( tspi ) ;
2013-09-26 21:01:43 +04:00
goto complete_xfer ;
2013-02-22 16:37:39 +04:00
}
msg - > actual_length + = xfer - > len ;
2013-09-26 21:01:43 +04:00
complete_xfer :
if ( ret < 0 | | skip ) {
2019-04-05 03:14:02 +03:00
tegra_spi_transfer_end ( spi ) ;
2019-09-26 13:51:38 +03:00
spi_transfer_delay_exec ( xfer ) ;
2013-09-26 21:01:43 +04:00
goto exit ;
2014-01-15 10:07:04 +04:00
} else if ( list_is_last ( & xfer - > transfer_list ,
& msg - > transfers ) ) {
2013-09-26 21:01:43 +04:00
if ( xfer - > cs_change )
tspi - > cs_control = spi ;
else {
2019-04-05 03:14:02 +03:00
tegra_spi_transfer_end ( spi ) ;
2019-09-26 13:51:38 +03:00
spi_transfer_delay_exec ( xfer ) ;
2013-09-26 21:01:43 +04:00
}
} else if ( xfer - > cs_change ) {
2019-04-05 03:14:02 +03:00
tegra_spi_transfer_end ( spi ) ;
2019-09-26 13:51:38 +03:00
spi_transfer_delay_exec ( xfer ) ;
2013-02-22 16:37:39 +04:00
}
2013-09-26 21:01:43 +04:00
2013-02-22 16:37:39 +04:00
}
ret = 0 ;
exit :
msg - > status = ret ;
spi_finalize_current_message ( master ) ;
return ret ;
}
static irqreturn_t handle_cpu_based_xfer ( struct tegra_spi_data * tspi )
{
struct spi_transfer * t = tspi - > curr_xfer ;
unsigned long flags ;
spin_lock_irqsave ( & tspi - > lock , flags ) ;
if ( tspi - > tx_status | | tspi - > rx_status ) {
dev_err ( tspi - > dev , " CpuXfer ERROR bit set 0x%x \n " ,
tspi - > status_reg ) ;
dev_err ( tspi - > dev , " CpuXfer 0x%08x:0x%08x \n " ,
tspi - > command1_reg , tspi - > dma_control_reg ) ;
2019-04-05 03:14:04 +03:00
tegra_spi_dump_regs ( tspi ) ;
2019-03-27 08:56:28 +03:00
tegra_spi_flush_fifos ( tspi ) ;
2019-04-05 03:14:03 +03:00
complete ( & tspi - > xfer_completion ) ;
spin_unlock_irqrestore ( & tspi - > lock , flags ) ;
2013-11-07 03:31:24 +04:00
reset_control_assert ( tspi - > rst ) ;
2013-02-22 16:37:39 +04:00
udelay ( 2 ) ;
2013-11-07 03:31:24 +04:00
reset_control_deassert ( tspi - > rst ) ;
2019-04-05 03:14:03 +03:00
return IRQ_HANDLED ;
2013-02-22 16:37:39 +04:00
}
if ( tspi - > cur_direction & DATA_DIR_RX )
tegra_spi_read_rx_fifo_to_client_rxbuf ( tspi , t ) ;
if ( tspi - > cur_direction & DATA_DIR_TX )
tspi - > cur_pos = tspi - > cur_tx_pos ;
else
tspi - > cur_pos = tspi - > cur_rx_pos ;
if ( tspi - > cur_pos = = t - > len ) {
complete ( & tspi - > xfer_completion ) ;
goto exit ;
}
tegra_spi_calculate_curr_xfer_param ( tspi - > cur_spi , tspi , t ) ;
tegra_spi_start_cpu_based_transfer ( tspi , t ) ;
exit :
spin_unlock_irqrestore ( & tspi - > lock , flags ) ;
return IRQ_HANDLED ;
}
static irqreturn_t handle_dma_based_xfer ( struct tegra_spi_data * tspi )
{
struct spi_transfer * t = tspi - > curr_xfer ;
long wait_status ;
int err = 0 ;
unsigned total_fifo_words ;
unsigned long flags ;
/* Abort dmas if any error */
if ( tspi - > cur_direction & DATA_DIR_TX ) {
if ( tspi - > tx_status ) {
dmaengine_terminate_all ( tspi - > tx_dma_chan ) ;
err + = 1 ;
} else {
wait_status = wait_for_completion_interruptible_timeout (
& tspi - > tx_dma_complete , SPI_DMA_TIMEOUT ) ;
if ( wait_status < = 0 ) {
dmaengine_terminate_all ( tspi - > tx_dma_chan ) ;
dev_err ( tspi - > dev , " TxDma Xfer failed \n " ) ;
err + = 1 ;
}
}
}
if ( tspi - > cur_direction & DATA_DIR_RX ) {
if ( tspi - > rx_status ) {
dmaengine_terminate_all ( tspi - > rx_dma_chan ) ;
err + = 2 ;
} else {
wait_status = wait_for_completion_interruptible_timeout (
& tspi - > rx_dma_complete , SPI_DMA_TIMEOUT ) ;
if ( wait_status < = 0 ) {
dmaengine_terminate_all ( tspi - > rx_dma_chan ) ;
dev_err ( tspi - > dev , " RxDma Xfer failed \n " ) ;
err + = 2 ;
}
}
}
spin_lock_irqsave ( & tspi - > lock , flags ) ;
if ( err ) {
dev_err ( tspi - > dev , " DmaXfer: ERROR bit set 0x%x \n " ,
tspi - > status_reg ) ;
dev_err ( tspi - > dev , " DmaXfer 0x%08x:0x%08x \n " ,
tspi - > command1_reg , tspi - > dma_control_reg ) ;
2019-04-05 03:14:04 +03:00
tegra_spi_dump_regs ( tspi ) ;
2019-03-27 08:56:28 +03:00
tegra_spi_flush_fifos ( tspi ) ;
2019-04-05 03:14:03 +03:00
complete ( & tspi - > xfer_completion ) ;
spin_unlock_irqrestore ( & tspi - > lock , flags ) ;
2013-11-07 03:31:24 +04:00
reset_control_assert ( tspi - > rst ) ;
2013-02-22 16:37:39 +04:00
udelay ( 2 ) ;
2013-11-07 03:31:24 +04:00
reset_control_deassert ( tspi - > rst ) ;
2013-02-22 16:37:39 +04:00
return IRQ_HANDLED ;
}
if ( tspi - > cur_direction & DATA_DIR_RX )
tegra_spi_copy_spi_rxbuf_to_client_rxbuf ( tspi , t ) ;
if ( tspi - > cur_direction & DATA_DIR_TX )
tspi - > cur_pos = tspi - > cur_tx_pos ;
else
tspi - > cur_pos = tspi - > cur_rx_pos ;
if ( tspi - > cur_pos = = t - > len ) {
complete ( & tspi - > xfer_completion ) ;
goto exit ;
}
/* Continue transfer in current message */
total_fifo_words = tegra_spi_calculate_curr_xfer_param ( tspi - > cur_spi ,
tspi , t ) ;
if ( total_fifo_words > SPI_FIFO_DEPTH )
err = tegra_spi_start_dma_based_transfer ( tspi , t ) ;
else
err = tegra_spi_start_cpu_based_transfer ( tspi , t ) ;
exit :
spin_unlock_irqrestore ( & tspi - > lock , flags ) ;
return IRQ_HANDLED ;
}
static irqreturn_t tegra_spi_isr_thread ( int irq , void * context_data )
{
struct tegra_spi_data * tspi = context_data ;
if ( ! tspi - > is_curr_dma_xfer )
return handle_cpu_based_xfer ( tspi ) ;
return handle_dma_based_xfer ( tspi ) ;
}
static irqreturn_t tegra_spi_isr ( int irq , void * context_data )
{
struct tegra_spi_data * tspi = context_data ;
tspi - > status_reg = tegra_spi_readl ( tspi , SPI_FIFO_STATUS ) ;
if ( tspi - > cur_direction & DATA_DIR_TX )
tspi - > tx_status = tspi - > status_reg &
( SPI_TX_FIFO_UNF | SPI_TX_FIFO_OVF ) ;
if ( tspi - > cur_direction & DATA_DIR_RX )
tspi - > rx_status = tspi - > status_reg &
( SPI_RX_FIFO_OVF | SPI_RX_FIFO_UNF ) ;
tegra_spi_clear_status ( tspi ) ;
return IRQ_WAKE_THREAD ;
}
2019-04-05 03:14:12 +03:00
static struct tegra_spi_soc_data tegra114_spi_soc_data = {
. has_intr_mask_reg = false ,
} ;
static struct tegra_spi_soc_data tegra124_spi_soc_data = {
. has_intr_mask_reg = false ,
} ;
static struct tegra_spi_soc_data tegra210_spi_soc_data = {
. has_intr_mask_reg = true ,
} ;
2014-05-07 11:51:02 +04:00
static const struct of_device_id tegra_spi_of_match [ ] = {
2019-04-05 03:14:12 +03:00
{
. compatible = " nvidia,tegra114-spi " ,
. data = & tegra114_spi_soc_data ,
} , {
. compatible = " nvidia,tegra124-spi " ,
. data = & tegra124_spi_soc_data ,
} , {
. compatible = " nvidia,tegra210-spi " ,
. data = & tegra210_spi_soc_data ,
} ,
2013-02-22 16:37:39 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , tegra_spi_of_match ) ;
static int tegra_spi_probe ( struct platform_device * pdev )
{
struct spi_master * master ;
struct tegra_spi_data * tspi ;
struct resource * r ;
int ret , spi_irq ;
2019-04-05 03:14:06 +03:00
int bus_num ;
2013-02-22 16:37:39 +04:00
master = spi_alloc_master ( & pdev - > dev , sizeof ( * tspi ) ) ;
if ( ! master ) {
dev_err ( & pdev - > dev , " master allocation failed \n " ) ;
return - ENOMEM ;
}
2013-05-23 14:20:40 +04:00
platform_set_drvdata ( pdev , master ) ;
2013-02-22 16:37:39 +04:00
tspi = spi_master_get_devdata ( master ) ;
2014-02-10 17:48:16 +04:00
if ( of_property_read_u32 ( pdev - > dev . of_node , " spi-max-frequency " ,
& master - > max_speed_hz ) )
master - > max_speed_hz = 25000000 ; /* 25MHz */
2013-02-22 16:37:39 +04:00
/* the spi->mode bits understood by this driver: */
2019-05-14 08:03:52 +03:00
master - > use_gpio_descriptors = true ;
2019-04-05 03:14:07 +03:00
master - > mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH | SPI_LSB_FIRST |
2019-04-05 03:14:08 +03:00
SPI_TX_DUAL | SPI_RX_DUAL | SPI_3WIRE ;
2019-04-05 03:14:05 +03:00
master - > bits_per_word_mask = SPI_BPW_RANGE_MASK ( 4 , 32 ) ;
2013-02-22 16:37:39 +04:00
master - > setup = tegra_spi_setup ;
2019-05-23 04:29:04 +03:00
master - > cleanup = tegra_spi_cleanup ;
2013-02-22 16:37:39 +04:00
master - > transfer_one_message = tegra_spi_transfer_one_message ;
2019-05-14 08:03:54 +03:00
master - > set_cs_timing = tegra_spi_set_hw_cs_timing ;
2013-02-22 16:37:39 +04:00
master - > num_chipselect = MAX_CHIP_SELECT ;
2013-07-28 18:37:31 +04:00
master - > auto_runtime_pm = true ;
2019-04-05 03:14:06 +03:00
bus_num = of_alias_get_id ( pdev - > dev . of_node , " spi " ) ;
if ( bus_num > = 0 )
master - > bus_num = bus_num ;
2013-02-22 16:37:39 +04:00
tspi - > master = master ;
tspi - > dev = & pdev - > dev ;
spin_lock_init ( & tspi - > lock ) ;
2019-04-05 03:14:12 +03:00
tspi - > soc_data = of_device_get_match_data ( & pdev - > dev ) ;
if ( ! tspi - > soc_data ) {
dev_err ( & pdev - > dev , " unsupported tegra \n " ) ;
ret = - ENODEV ;
goto exit_free_master ;
}
2013-02-22 16:37:39 +04:00
r = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
tspi - > base = devm_ioremap_resource ( & pdev - > dev , r ) ;
if ( IS_ERR ( tspi - > base ) ) {
ret = PTR_ERR ( tspi - > base ) ;
goto exit_free_master ;
}
2013-05-14 14:07:12 +04:00
tspi - > phys = r - > start ;
2013-02-22 16:37:39 +04:00
spi_irq = platform_get_irq ( pdev , 0 ) ;
tspi - > irq = spi_irq ;
tspi - > clk = devm_clk_get ( & pdev - > dev , " spi " ) ;
if ( IS_ERR ( tspi - > clk ) ) {
dev_err ( & pdev - > dev , " can not get clock \n " ) ;
ret = PTR_ERR ( tspi - > clk ) ;
2019-03-27 08:56:32 +03:00
goto exit_free_master ;
2013-02-22 16:37:39 +04:00
}
2017-07-19 18:26:23 +03:00
tspi - > rst = devm_reset_control_get_exclusive ( & pdev - > dev , " spi " ) ;
2013-11-07 03:31:24 +04:00
if ( IS_ERR ( tspi - > rst ) ) {
dev_err ( & pdev - > dev , " can not get reset \n " ) ;
ret = PTR_ERR ( tspi - > rst ) ;
2019-03-27 08:56:32 +03:00
goto exit_free_master ;
2013-11-07 03:31:24 +04:00
}
2013-02-22 16:37:39 +04:00
tspi - > max_buf_size = SPI_FIFO_DEPTH < < 2 ;
tspi - > dma_buf_size = DEFAULT_SPI_DMA_BUF_LEN ;
2013-11-12 00:13:47 +04:00
ret = tegra_spi_init_dma_param ( tspi , true ) ;
if ( ret < 0 )
2019-03-27 08:56:32 +03:00
goto exit_free_master ;
2013-11-12 00:13:47 +04:00
ret = tegra_spi_init_dma_param ( tspi , false ) ;
if ( ret < 0 )
goto exit_rx_dma_free ;
tspi - > max_buf_size = tspi - > dma_buf_size ;
init_completion ( & tspi - > tx_dma_complete ) ;
init_completion ( & tspi - > rx_dma_complete ) ;
2013-02-22 16:37:39 +04:00
init_completion ( & tspi - > xfer_completion ) ;
pm_runtime_enable ( & pdev - > dev ) ;
if ( ! pm_runtime_enabled ( & pdev - > dev ) ) {
ret = tegra_spi_runtime_resume ( & pdev - > dev ) ;
if ( ret )
goto exit_pm_disable ;
}
ret = pm_runtime_get_sync ( & pdev - > dev ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " pm runtime get failed, e = %d \n " , ret ) ;
goto exit_pm_disable ;
}
2019-03-27 08:56:32 +03:00
reset_control_assert ( tspi - > rst ) ;
udelay ( 2 ) ;
reset_control_deassert ( tspi - > rst ) ;
2013-02-22 16:37:39 +04:00
tspi - > def_command1_reg = SPI_M_S ;
tegra_spi_writel ( tspi , tspi - > def_command1_reg , SPI_COMMAND1 ) ;
2019-05-14 08:03:54 +03:00
tspi - > spi_cs_timing1 = tegra_spi_readl ( tspi , SPI_CS_TIMING1 ) ;
tspi - > spi_cs_timing2 = tegra_spi_readl ( tspi , SPI_CS_TIMING2 ) ;
2019-05-14 08:03:55 +03:00
tspi - > def_command2_reg = tegra_spi_readl ( tspi , SPI_COMMAND2 ) ;
tspi - > last_used_cs = master - > num_chipselect + 1 ;
2013-02-22 16:37:39 +04:00
pm_runtime_put ( & pdev - > dev ) ;
2019-03-27 08:56:32 +03:00
ret = request_threaded_irq ( tspi - > irq , tegra_spi_isr ,
tegra_spi_isr_thread , IRQF_ONESHOT ,
dev_name ( & pdev - > dev ) , tspi ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to register ISR for IRQ %d \n " ,
tspi - > irq ) ;
goto exit_pm_disable ;
}
2013-02-22 16:37:39 +04:00
master - > dev . of_node = pdev - > dev . of_node ;
2013-09-24 08:49:24 +04:00
ret = devm_spi_register_master ( & pdev - > dev , master ) ;
2013-02-22 16:37:39 +04:00
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " can not register to master err %d \n " , ret ) ;
2019-03-27 08:56:32 +03:00
goto exit_free_irq ;
2013-02-22 16:37:39 +04:00
}
return ret ;
2019-03-27 08:56:32 +03:00
exit_free_irq :
free_irq ( spi_irq , tspi ) ;
2013-02-22 16:37:39 +04:00
exit_pm_disable :
pm_runtime_disable ( & pdev - > dev ) ;
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra_spi_runtime_suspend ( & pdev - > dev ) ;
tegra_spi_deinit_dma_param ( tspi , false ) ;
exit_rx_dma_free :
tegra_spi_deinit_dma_param ( tspi , true ) ;
exit_free_master :
spi_master_put ( master ) ;
return ret ;
}
static int tegra_spi_remove ( struct platform_device * pdev )
{
2013-05-23 14:20:40 +04:00
struct spi_master * master = platform_get_drvdata ( pdev ) ;
2013-02-22 16:37:39 +04:00
struct tegra_spi_data * tspi = spi_master_get_devdata ( master ) ;
free_irq ( tspi - > irq , tspi ) ;
if ( tspi - > tx_dma_chan )
tegra_spi_deinit_dma_param ( tspi , false ) ;
if ( tspi - > rx_dma_chan )
tegra_spi_deinit_dma_param ( tspi , true ) ;
pm_runtime_disable ( & pdev - > dev ) ;
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra_spi_runtime_suspend ( & pdev - > dev ) ;
return 0 ;
}
# ifdef CONFIG_PM_SLEEP
static int tegra_spi_suspend ( struct device * dev )
{
struct spi_master * master = dev_get_drvdata ( dev ) ;
return spi_master_suspend ( master ) ;
}
static int tegra_spi_resume ( struct device * dev )
{
struct spi_master * master = dev_get_drvdata ( dev ) ;
struct tegra_spi_data * tspi = spi_master_get_devdata ( master ) ;
int ret ;
ret = pm_runtime_get_sync ( dev ) ;
if ( ret < 0 ) {
dev_err ( dev , " pm runtime failed, e = %d \n " , ret ) ;
return ret ;
}
tegra_spi_writel ( tspi , tspi - > command1_reg , SPI_COMMAND1 ) ;
2019-05-14 08:03:55 +03:00
tegra_spi_writel ( tspi , tspi - > def_command2_reg , SPI_COMMAND2 ) ;
tspi - > last_used_cs = master - > num_chipselect + 1 ;
2013-02-22 16:37:39 +04:00
pm_runtime_put ( dev ) ;
return spi_master_resume ( master ) ;
}
# endif
static int tegra_spi_runtime_suspend ( struct device * dev )
{
struct spi_master * master = dev_get_drvdata ( dev ) ;
struct tegra_spi_data * tspi = spi_master_get_devdata ( master ) ;
/* Flush all write which are in PPSB queue by reading back */
tegra_spi_readl ( tspi , SPI_COMMAND1 ) ;
clk_disable_unprepare ( tspi - > clk ) ;
return 0 ;
}
static int tegra_spi_runtime_resume ( struct device * dev )
{
struct spi_master * master = dev_get_drvdata ( dev ) ;
struct tegra_spi_data * tspi = spi_master_get_devdata ( master ) ;
int ret ;
ret = clk_prepare_enable ( tspi - > clk ) ;
if ( ret < 0 ) {
dev_err ( tspi - > dev , " clk_prepare failed: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
static const struct dev_pm_ops tegra_spi_pm_ops = {
SET_RUNTIME_PM_OPS ( tegra_spi_runtime_suspend ,
tegra_spi_runtime_resume , NULL )
SET_SYSTEM_SLEEP_PM_OPS ( tegra_spi_suspend , tegra_spi_resume )
} ;
static struct platform_driver tegra_spi_driver = {
. driver = {
. name = " spi-tegra114 " ,
. pm = & tegra_spi_pm_ops ,
. of_match_table = tegra_spi_of_match ,
} ,
. probe = tegra_spi_probe ,
. remove = tegra_spi_remove ,
} ;
module_platform_driver ( tegra_spi_driver ) ;
MODULE_ALIAS ( " platform:spi-tegra114 " ) ;
MODULE_DESCRIPTION ( " NVIDIA Tegra114 SPI Controller Driver " ) ;
MODULE_AUTHOR ( " Laxman Dewangan <ldewangan@nvidia.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;