2011-06-06 01:16:30 -06:00
/*
2009-11-30 07:39:42 +00:00
* Copyright ( C ) 2009 Samsung Electronics Ltd .
* Jaswinder Singh < jassi . brar @ samsung . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/workqueue.h>
2011-11-10 10:57:32 +00:00
# include <linux/interrupt.h>
2009-11-30 07:39:42 +00:00
# include <linux/delay.h>
# include <linux/clk.h>
# include <linux/dma-mapping.h>
2013-04-11 22:42:03 +02:00
# include <linux/dmaengine.h>
2009-11-30 07:39:42 +00:00
# include <linux/platform_device.h>
2011-12-04 00:58:06 +00:00
# include <linux/pm_runtime.h>
2009-11-30 07:39:42 +00:00
# include <linux/spi/spi.h>
2012-07-13 07:15:14 +09:00
# include <linux/gpio.h>
2012-07-13 07:15:15 +09:00
# include <linux/of.h>
# include <linux/of_gpio.h>
2009-11-30 07:39:42 +00:00
2012-08-24 15:22:12 +02:00
# include <linux/platform_data/spi-s3c64xx.h>
2009-11-30 07:39:42 +00:00
2013-04-18 18:06:05 +01:00
# ifdef CONFIG_S3C_DMA
2013-04-11 22:42:03 +02:00
# include <mach/dma.h>
# endif
2012-07-13 07:15:14 +09:00
# define MAX_SPI_PORTS 3
2013-05-20 12:21:32 +05:30
# define S3C64XX_SPI_QUIRK_POLL (1 << 0)
2012-07-13 07:15:14 +09:00
2009-11-30 07:39:42 +00:00
/* Registers and bit-fields */
# define S3C64XX_SPI_CH_CFG 0x00
# define S3C64XX_SPI_CLK_CFG 0x04
# define S3C64XX_SPI_MODE_CFG 0x08
# define S3C64XX_SPI_SLAVE_SEL 0x0C
# define S3C64XX_SPI_INT_EN 0x10
# define S3C64XX_SPI_STATUS 0x14
# define S3C64XX_SPI_TX_DATA 0x18
# define S3C64XX_SPI_RX_DATA 0x1C
# define S3C64XX_SPI_PACKET_CNT 0x20
# define S3C64XX_SPI_PENDING_CLR 0x24
# define S3C64XX_SPI_SWAP_CFG 0x28
# define S3C64XX_SPI_FB_CLK 0x2C
# define S3C64XX_SPI_CH_HS_EN (1<<6) /* High Speed Enable */
# define S3C64XX_SPI_CH_SW_RST (1<<5)
# define S3C64XX_SPI_CH_SLAVE (1<<4)
# define S3C64XX_SPI_CPOL_L (1<<3)
# define S3C64XX_SPI_CPHA_B (1<<2)
# define S3C64XX_SPI_CH_RXCH_ON (1<<1)
# define S3C64XX_SPI_CH_TXCH_ON (1<<0)
# define S3C64XX_SPI_CLKSEL_SRCMSK (3<<9)
# define S3C64XX_SPI_CLKSEL_SRCSHFT 9
# define S3C64XX_SPI_ENCLK_ENABLE (1<<8)
2013-01-31 15:25:01 +09:00
# define S3C64XX_SPI_PSR_MASK 0xff
2009-11-30 07:39:42 +00:00
# define S3C64XX_SPI_MODE_CH_TSZ_BYTE (0<<29)
# define S3C64XX_SPI_MODE_CH_TSZ_HALFWORD (1<<29)
# define S3C64XX_SPI_MODE_CH_TSZ_WORD (2<<29)
# define S3C64XX_SPI_MODE_CH_TSZ_MASK (3<<29)
# define S3C64XX_SPI_MODE_BUS_TSZ_BYTE (0<<17)
# define S3C64XX_SPI_MODE_BUS_TSZ_HALFWORD (1<<17)
# define S3C64XX_SPI_MODE_BUS_TSZ_WORD (2<<17)
# define S3C64XX_SPI_MODE_BUS_TSZ_MASK (3<<17)
# define S3C64XX_SPI_MODE_RXDMA_ON (1<<2)
# define S3C64XX_SPI_MODE_TXDMA_ON (1<<1)
# define S3C64XX_SPI_MODE_4BURST (1<<0)
# define S3C64XX_SPI_SLAVE_AUTO (1<<1)
# define S3C64XX_SPI_SLAVE_SIG_INACT (1<<0)
# define S3C64XX_SPI_INT_TRAILING_EN (1<<6)
# define S3C64XX_SPI_INT_RX_OVERRUN_EN (1<<5)
# define S3C64XX_SPI_INT_RX_UNDERRUN_EN (1<<4)
# define S3C64XX_SPI_INT_TX_OVERRUN_EN (1<<3)
# define S3C64XX_SPI_INT_TX_UNDERRUN_EN (1<<2)
# define S3C64XX_SPI_INT_RX_FIFORDY_EN (1<<1)
# define S3C64XX_SPI_INT_TX_FIFORDY_EN (1<<0)
# define S3C64XX_SPI_ST_RX_OVERRUN_ERR (1<<5)
# define S3C64XX_SPI_ST_RX_UNDERRUN_ERR (1<<4)
# define S3C64XX_SPI_ST_TX_OVERRUN_ERR (1<<3)
# define S3C64XX_SPI_ST_TX_UNDERRUN_ERR (1<<2)
# define S3C64XX_SPI_ST_RX_FIFORDY (1<<1)
# define S3C64XX_SPI_ST_TX_FIFORDY (1<<0)
# define S3C64XX_SPI_PACKET_CNT_EN (1<<16)
# define S3C64XX_SPI_PND_TX_UNDERRUN_CLR (1<<4)
# define S3C64XX_SPI_PND_TX_OVERRUN_CLR (1<<3)
# define S3C64XX_SPI_PND_RX_UNDERRUN_CLR (1<<2)
# define S3C64XX_SPI_PND_RX_OVERRUN_CLR (1<<1)
# define S3C64XX_SPI_PND_TRAILING_CLR (1<<0)
# define S3C64XX_SPI_SWAP_RX_HALF_WORD (1<<7)
# define S3C64XX_SPI_SWAP_RX_BYTE (1<<6)
# define S3C64XX_SPI_SWAP_RX_BIT (1<<5)
# define S3C64XX_SPI_SWAP_RX_EN (1<<4)
# define S3C64XX_SPI_SWAP_TX_HALF_WORD (1<<3)
# define S3C64XX_SPI_SWAP_TX_BYTE (1<<2)
# define S3C64XX_SPI_SWAP_TX_BIT (1<<1)
# define S3C64XX_SPI_SWAP_TX_EN (1<<0)
# define S3C64XX_SPI_FBCLK_MSK (3<<0)
2012-07-13 07:15:14 +09:00
# define FIFO_LVL_MASK(i) ((i)->port_conf->fifo_lvl_mask[i->port_id])
# define S3C64XX_SPI_ST_TX_DONE(v, i) (((v) & \
( 1 < < ( i ) - > port_conf - > tx_st_done ) ) ? 1 : 0 )
# define TX_FIFO_LVL(v, i) (((v) >> 6) & FIFO_LVL_MASK(i))
# define RX_FIFO_LVL(v, i) (((v) >> (i)->port_conf->rx_lvl_offset) & \
FIFO_LVL_MASK ( i ) )
2009-11-30 07:39:42 +00:00
# define S3C64XX_SPI_MAX_TRAILCNT 0x3ff
# define S3C64XX_SPI_TRAILCNT_OFF 19
# define S3C64XX_SPI_TRAILCNT S3C64XX_SPI_MAX_TRAILCNT
# define msecs_to_loops(t) (loops_per_jiffy / 1000 * HZ * t)
2013-05-20 12:21:32 +05:30
# define is_polling(x) (x->port_conf->quirks & S3C64XX_SPI_QUIRK_POLL)
2009-11-30 07:39:42 +00:00
# define RXBUSY (1<<2)
# define TXBUSY (1<<3)
2011-09-02 09:44:42 +09:00
struct s3c64xx_spi_dma_data {
2013-04-11 22:42:03 +02:00
struct dma_chan * ch ;
2012-04-30 16:31:27 +00:00
enum dma_transfer_direction direction ;
2013-04-11 22:42:03 +02:00
unsigned int dmach ;
2011-09-02 09:44:42 +09:00
} ;
2012-07-13 07:15:14 +09:00
/**
* struct s3c64xx_spi_info - SPI Controller hardware info
* @ fifo_lvl_mask : Bit - mask for { TX | RX } _FIFO_LVL bits in SPI_STATUS register .
* @ rx_lvl_offset : Bit offset of RX_FIFO_LVL bits in SPI_STATUS regiter .
* @ tx_st_done : Bit offset of TX_DONE bit in SPI_STATUS regiter .
* @ high_speed : True , if the controller supports HIGH_SPEED_EN bit .
* @ clk_from_cmu : True , if the controller does not include a clock mux and
* prescaler unit .
*
* The Samsung s3c64xx SPI controller are used on various Samsung SoC ' s but
* differ in some aspects such as the size of the fifo and spi bus clock
* setup . Such differences are specified to the driver using this structure
* which is provided as driver data to the driver .
*/
struct s3c64xx_spi_port_config {
int fifo_lvl_mask [ MAX_SPI_PORTS ] ;
int rx_lvl_offset ;
int tx_st_done ;
2013-05-20 12:21:32 +05:30
int quirks ;
2012-07-13 07:15:14 +09:00
bool high_speed ;
bool clk_from_cmu ;
} ;
2009-11-30 07:39:42 +00:00
/**
* struct s3c64xx_spi_driver_data - Runtime info holder for SPI driver .
* @ clk : Pointer to the spi clock .
2010-01-20 13:49:44 -07:00
* @ src_clk : Pointer to the clock used to generate SPI signals .
2009-11-30 07:39:42 +00:00
* @ master : Pointer to the SPI Protocol master .
* @ cntrlr_info : Platform specific data for the controller this driver manages .
* @ tgl_spi : Pointer to the last CS left untoggled by the cs_change hint .
* @ queue : To log SPI xfer requests .
* @ lock : Controller specific lock .
* @ state : Set of FLAGS to indicate status .
* @ rx_dmach : Controller ' s DMA channel for Rx .
* @ tx_dmach : Controller ' s DMA channel for Tx .
* @ sfr_start : BUS address of SPI controller regs .
* @ regs : Pointer to ioremap ' ed controller registers .
2011-11-10 10:57:32 +00:00
* @ irq : interrupt
2009-11-30 07:39:42 +00:00
* @ xfer_completion : To indicate completion of xfer task .
* @ cur_mode : Stores the active configuration of the controller .
* @ cur_bpw : Stores the active bits per word settings .
* @ cur_speed : Stores the active xfer clock speed .
*/
struct s3c64xx_spi_driver_data {
void __iomem * regs ;
struct clk * clk ;
2010-01-20 13:49:44 -07:00
struct clk * src_clk ;
2009-11-30 07:39:42 +00:00
struct platform_device * pdev ;
struct spi_master * master ;
2010-01-20 13:49:44 -07:00
struct s3c64xx_spi_info * cntrlr_info ;
2009-11-30 07:39:42 +00:00
struct spi_device * tgl_spi ;
struct list_head queue ;
spinlock_t lock ;
unsigned long sfr_start ;
struct completion xfer_completion ;
unsigned state ;
unsigned cur_mode , cur_bpw ;
unsigned cur_speed ;
2011-09-02 09:44:42 +09:00
struct s3c64xx_spi_dma_data rx_dma ;
struct s3c64xx_spi_dma_data tx_dma ;
2013-04-18 18:06:05 +01:00
# ifdef CONFIG_S3C_DMA
2011-09-02 09:44:41 +09:00
struct samsung_dma_ops * ops ;
2013-04-11 22:42:03 +02:00
# endif
2012-07-13 07:15:14 +09:00
struct s3c64xx_spi_port_config * port_conf ;
unsigned int port_id ;
2012-07-13 07:15:15 +09:00
unsigned long gpios [ 4 ] ;
2013-06-21 11:26:12 +05:30
bool cs_gpio ;
2009-11-30 07:39:42 +00:00
} ;
static void flush_fifo ( struct s3c64xx_spi_driver_data * sdd )
{
void __iomem * regs = sdd - > regs ;
unsigned long loops ;
u32 val ;
writel ( 0 , regs + S3C64XX_SPI_PACKET_CNT ) ;
2012-05-23 21:29:51 +09:00
val = readl ( regs + S3C64XX_SPI_CH_CFG ) ;
val & = ~ ( S3C64XX_SPI_CH_RXCH_ON | S3C64XX_SPI_CH_TXCH_ON ) ;
writel ( val , regs + S3C64XX_SPI_CH_CFG ) ;
2009-11-30 07:39:42 +00:00
val = readl ( regs + S3C64XX_SPI_CH_CFG ) ;
val | = S3C64XX_SPI_CH_SW_RST ;
val & = ~ S3C64XX_SPI_CH_HS_EN ;
writel ( val , regs + S3C64XX_SPI_CH_CFG ) ;
/* Flush TxFIFO*/
loops = msecs_to_loops ( 1 ) ;
do {
val = readl ( regs + S3C64XX_SPI_STATUS ) ;
2012-07-13 07:15:14 +09:00
} while ( TX_FIFO_LVL ( val , sdd ) & & loops - - ) ;
2009-11-30 07:39:42 +00:00
2010-08-23 17:40:56 +01:00
if ( loops = = 0 )
dev_warn ( & sdd - > pdev - > dev , " Timed out flushing TX FIFO \n " ) ;
2009-11-30 07:39:42 +00:00
/* Flush RxFIFO*/
loops = msecs_to_loops ( 1 ) ;
do {
val = readl ( regs + S3C64XX_SPI_STATUS ) ;
2012-07-13 07:15:14 +09:00
if ( RX_FIFO_LVL ( val , sdd ) )
2009-11-30 07:39:42 +00:00
readl ( regs + S3C64XX_SPI_RX_DATA ) ;
else
break ;
} while ( loops - - ) ;
2010-08-23 17:40:56 +01:00
if ( loops = = 0 )
dev_warn ( & sdd - > pdev - > dev , " Timed out flushing RX FIFO \n " ) ;
2009-11-30 07:39:42 +00:00
val = readl ( regs + S3C64XX_SPI_CH_CFG ) ;
val & = ~ S3C64XX_SPI_CH_SW_RST ;
writel ( val , regs + S3C64XX_SPI_CH_CFG ) ;
val = readl ( regs + S3C64XX_SPI_MODE_CFG ) ;
val & = ~ ( S3C64XX_SPI_MODE_TXDMA_ON | S3C64XX_SPI_MODE_RXDMA_ON ) ;
writel ( val , regs + S3C64XX_SPI_MODE_CFG ) ;
}
2011-09-02 09:44:42 +09:00
static void s3c64xx_spi_dmacb ( void * data )
2011-09-02 09:44:41 +09:00
{
2011-09-02 09:44:42 +09:00
struct s3c64xx_spi_driver_data * sdd ;
struct s3c64xx_spi_dma_data * dma = data ;
2011-09-02 09:44:41 +09:00
unsigned long flags ;
2012-03-10 09:48:46 +09:00
if ( dma - > direction = = DMA_DEV_TO_MEM )
2011-09-02 09:44:42 +09:00
sdd = container_of ( data ,
struct s3c64xx_spi_driver_data , rx_dma ) ;
else
sdd = container_of ( data ,
struct s3c64xx_spi_driver_data , tx_dma ) ;
2011-09-02 09:44:41 +09:00
spin_lock_irqsave ( & sdd - > lock , flags ) ;
2012-03-10 09:48:46 +09:00
if ( dma - > direction = = DMA_DEV_TO_MEM ) {
2011-09-02 09:44:42 +09:00
sdd - > state & = ~ RXBUSY ;
if ( ! ( sdd - > state & TXBUSY ) )
complete ( & sdd - > xfer_completion ) ;
} else {
sdd - > state & = ~ TXBUSY ;
if ( ! ( sdd - > state & RXBUSY ) )
complete ( & sdd - > xfer_completion ) ;
}
2011-09-02 09:44:41 +09:00
spin_unlock_irqrestore ( & sdd - > lock , flags ) ;
}
2013-04-18 18:06:05 +01:00
# ifdef CONFIG_S3C_DMA
2013-04-11 22:42:03 +02:00
/* FIXME: remove this section once arch/arm/mach-s3c64xx uses dmaengine */
static struct s3c2410_dma_client s3c64xx_spi_dma_client = {
. name = " samsung-spi-dma " ,
} ;
2011-09-02 09:44:42 +09:00
static void prepare_dma ( struct s3c64xx_spi_dma_data * dma ,
unsigned len , dma_addr_t buf )
2011-09-02 09:44:41 +09:00
{
2011-09-02 09:44:42 +09:00
struct s3c64xx_spi_driver_data * sdd ;
2012-06-19 13:27:03 +09:00
struct samsung_dma_prep info ;
struct samsung_dma_config config ;
2011-09-02 09:44:41 +09:00
2012-06-19 13:27:03 +09:00
if ( dma - > direction = = DMA_DEV_TO_MEM ) {
2011-09-02 09:44:42 +09:00
sdd = container_of ( ( void * ) dma ,
struct s3c64xx_spi_driver_data , rx_dma ) ;
2012-06-19 13:27:03 +09:00
config . direction = sdd - > rx_dma . direction ;
config . fifo = sdd - > sfr_start + S3C64XX_SPI_RX_DATA ;
config . width = sdd - > cur_bpw / 8 ;
2013-04-11 22:42:03 +02:00
sdd - > ops - > config ( ( enum dma_ch ) sdd - > rx_dma . ch , & config ) ;
2012-06-19 13:27:03 +09:00
} else {
2011-09-02 09:44:42 +09:00
sdd = container_of ( ( void * ) dma ,
struct s3c64xx_spi_driver_data , tx_dma ) ;
2012-06-19 13:27:03 +09:00
config . direction = sdd - > tx_dma . direction ;
config . fifo = sdd - > sfr_start + S3C64XX_SPI_TX_DATA ;
config . width = sdd - > cur_bpw / 8 ;
2013-04-11 22:42:03 +02:00
sdd - > ops - > config ( ( enum dma_ch ) sdd - > tx_dma . ch , & config ) ;
2012-06-19 13:27:03 +09:00
}
2011-09-02 09:44:41 +09:00
2011-09-02 09:44:42 +09:00
info . cap = DMA_SLAVE ;
info . len = len ;
info . fp = s3c64xx_spi_dmacb ;
info . fp_param = dma ;
info . direction = dma - > direction ;
info . buf = buf ;
2013-04-11 22:42:03 +02:00
sdd - > ops - > prepare ( ( enum dma_ch ) dma - > ch , & info ) ;
sdd - > ops - > trigger ( ( enum dma_ch ) dma - > ch ) ;
2011-09-02 09:44:42 +09:00
}
2011-09-02 09:44:41 +09:00
2011-09-02 09:44:42 +09:00
static int acquire_dma ( struct s3c64xx_spi_driver_data * sdd )
{
2012-06-19 13:27:03 +09:00
struct samsung_dma_req req ;
2013-01-18 17:17:03 +05:30
struct device * dev = & sdd - > pdev - > dev ;
2011-09-02 09:44:42 +09:00
sdd - > ops = samsung_dma_get_ops ( ) ;
2012-06-19 13:27:03 +09:00
req . cap = DMA_SLAVE ;
req . client = & s3c64xx_spi_dma_client ;
2013-04-11 22:42:03 +02:00
sdd - > rx_dma . ch = ( void * ) sdd - > ops - > request ( sdd - > rx_dma . dmach , & req , dev , " rx " ) ;
sdd - > tx_dma . ch = ( void * ) sdd - > ops - > request ( sdd - > tx_dma . dmach , & req , dev , " tx " ) ;
2011-09-02 09:44:42 +09:00
return 1 ;
2011-09-02 09:44:41 +09:00
}
2013-04-11 22:42:03 +02:00
static int s3c64xx_spi_prepare_transfer ( struct spi_master * spi )
{
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( spi ) ;
2013-05-20 12:21:32 +05:30
/*
* If DMA resource was not available during
* probe , no need to continue with dma requests
* else Acquire DMA channels
*/
while ( ! is_polling ( sdd ) & & ! acquire_dma ( sdd ) )
2013-04-11 22:42:03 +02:00
usleep_range ( 10000 , 11000 ) ;
pm_runtime_get_sync ( & sdd - > pdev - > dev ) ;
return 0 ;
}
static int s3c64xx_spi_unprepare_transfer ( struct spi_master * spi )
{
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( spi ) ;
/* Free DMA channels */
2013-05-20 12:21:32 +05:30
if ( ! is_polling ( sdd ) ) {
sdd - > ops - > release ( ( enum dma_ch ) sdd - > rx_dma . ch ,
& s3c64xx_spi_dma_client ) ;
sdd - > ops - > release ( ( enum dma_ch ) sdd - > tx_dma . ch ,
& s3c64xx_spi_dma_client ) ;
}
2013-04-11 22:42:03 +02:00
pm_runtime_put ( & sdd - > pdev - > dev ) ;
return 0 ;
}
static void s3c64xx_spi_dma_stop ( struct s3c64xx_spi_driver_data * sdd ,
struct s3c64xx_spi_dma_data * dma )
{
sdd - > ops - > stop ( ( enum dma_ch ) dma - > ch ) ;
}
# else
static void prepare_dma ( struct s3c64xx_spi_dma_data * dma ,
unsigned len , dma_addr_t buf )
{
struct s3c64xx_spi_driver_data * sdd ;
struct dma_slave_config config ;
struct scatterlist sg ;
struct dma_async_tx_descriptor * desc ;
if ( dma - > direction = = DMA_DEV_TO_MEM ) {
sdd = container_of ( ( void * ) dma ,
struct s3c64xx_spi_driver_data , rx_dma ) ;
config . direction = dma - > direction ;
config . src_addr = sdd - > sfr_start + S3C64XX_SPI_RX_DATA ;
config . src_addr_width = sdd - > cur_bpw / 8 ;
config . src_maxburst = 1 ;
dmaengine_slave_config ( dma - > ch , & config ) ;
} else {
sdd = container_of ( ( void * ) dma ,
struct s3c64xx_spi_driver_data , tx_dma ) ;
config . direction = dma - > direction ;
config . dst_addr = sdd - > sfr_start + S3C64XX_SPI_TX_DATA ;
config . dst_addr_width = sdd - > cur_bpw / 8 ;
config . dst_maxburst = 1 ;
dmaengine_slave_config ( dma - > ch , & config ) ;
}
sg_init_table ( & sg , 1 ) ;
sg_dma_len ( & sg ) = len ;
sg_set_page ( & sg , pfn_to_page ( PFN_DOWN ( buf ) ) ,
len , offset_in_page ( buf ) ) ;
sg_dma_address ( & sg ) = buf ;
desc = dmaengine_prep_slave_sg ( dma - > ch ,
& sg , 1 , dma - > direction , DMA_PREP_INTERRUPT ) ;
desc - > callback = s3c64xx_spi_dmacb ;
desc - > callback_param = dma ;
dmaengine_submit ( desc ) ;
dma_async_issue_pending ( dma - > ch ) ;
}
static int s3c64xx_spi_prepare_transfer ( struct spi_master * spi )
{
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( spi ) ;
dma_filter_fn filter = sdd - > cntrlr_info - > filter ;
struct device * dev = & sdd - > pdev - > dev ;
dma_cap_mask_t mask ;
2013-04-18 18:12:00 +01:00
int ret ;
2013-04-11 22:42:03 +02:00
2013-06-27 12:26:53 +05:30
if ( is_polling ( sdd ) )
return 0 ;
2013-04-11 22:42:03 +02:00
dma_cap_zero ( mask ) ;
dma_cap_set ( DMA_SLAVE , mask ) ;
/* Acquire DMA channels */
sdd - > rx_dma . ch = dma_request_slave_channel_compat ( mask , filter ,
( void * ) sdd - > rx_dma . dmach , dev , " rx " ) ;
2013-04-18 18:12:00 +01:00
if ( ! sdd - > rx_dma . ch ) {
dev_err ( dev , " Failed to get RX DMA channel \n " ) ;
ret = - EBUSY ;
goto out ;
}
2013-04-11 22:42:03 +02:00
sdd - > tx_dma . ch = dma_request_slave_channel_compat ( mask , filter ,
( void * ) sdd - > tx_dma . dmach , dev , " tx " ) ;
2013-04-18 18:12:00 +01:00
if ( ! sdd - > tx_dma . ch ) {
dev_err ( dev , " Failed to get TX DMA channel \n " ) ;
ret = - EBUSY ;
goto out_rx ;
}
ret = pm_runtime_get_sync ( & sdd - > pdev - > dev ) ;
2013-06-10 18:22:26 +02:00
if ( ret < 0 ) {
2013-04-18 18:12:00 +01:00
dev_err ( dev , " Failed to enable device: %d \n " , ret ) ;
goto out_tx ;
}
2013-04-11 22:42:03 +02:00
return 0 ;
2013-04-18 18:12:00 +01:00
out_tx :
dma_release_channel ( sdd - > tx_dma . ch ) ;
out_rx :
dma_release_channel ( sdd - > rx_dma . ch ) ;
out :
return ret ;
2013-04-11 22:42:03 +02:00
}
static int s3c64xx_spi_unprepare_transfer ( struct spi_master * spi )
{
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( spi ) ;
/* Free DMA channels */
2013-05-20 12:21:32 +05:30
if ( ! is_polling ( sdd ) ) {
dma_release_channel ( sdd - > rx_dma . ch ) ;
dma_release_channel ( sdd - > tx_dma . ch ) ;
}
2013-04-11 22:42:03 +02:00
pm_runtime_put ( & sdd - > pdev - > dev ) ;
return 0 ;
}
static void s3c64xx_spi_dma_stop ( struct s3c64xx_spi_driver_data * sdd ,
struct s3c64xx_spi_dma_data * dma )
{
dmaengine_terminate_all ( dma - > ch ) ;
}
# endif
2009-11-30 07:39:42 +00:00
static void enable_datapath ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_device * spi ,
struct spi_transfer * xfer , int dma_mode )
{
void __iomem * regs = sdd - > regs ;
u32 modecfg , chcfg ;
modecfg = readl ( regs + S3C64XX_SPI_MODE_CFG ) ;
modecfg & = ~ ( S3C64XX_SPI_MODE_TXDMA_ON | S3C64XX_SPI_MODE_RXDMA_ON ) ;
chcfg = readl ( regs + S3C64XX_SPI_CH_CFG ) ;
chcfg & = ~ S3C64XX_SPI_CH_TXCH_ON ;
if ( dma_mode ) {
chcfg & = ~ S3C64XX_SPI_CH_RXCH_ON ;
} else {
/* Always shift in data in FIFO, even if xfer is Tx only,
* this helps setting PCKT_CNT value for generating clocks
* as exactly needed .
*/
chcfg | = S3C64XX_SPI_CH_RXCH_ON ;
writel ( ( ( xfer - > len * 8 / sdd - > cur_bpw ) & 0xffff )
| S3C64XX_SPI_PACKET_CNT_EN ,
regs + S3C64XX_SPI_PACKET_CNT ) ;
}
if ( xfer - > tx_buf ! = NULL ) {
sdd - > state | = TXBUSY ;
chcfg | = S3C64XX_SPI_CH_TXCH_ON ;
if ( dma_mode ) {
modecfg | = S3C64XX_SPI_MODE_TXDMA_ON ;
2011-09-02 09:44:42 +09:00
prepare_dma ( & sdd - > tx_dma , xfer - > len , xfer - > tx_dma ) ;
2009-11-30 07:39:42 +00:00
} else {
2010-09-29 17:31:33 +09:00
switch ( sdd - > cur_bpw ) {
case 32 :
iowrite32_rep ( regs + S3C64XX_SPI_TX_DATA ,
xfer - > tx_buf , xfer - > len / 4 ) ;
break ;
case 16 :
iowrite16_rep ( regs + S3C64XX_SPI_TX_DATA ,
xfer - > tx_buf , xfer - > len / 2 ) ;
break ;
default :
iowrite8_rep ( regs + S3C64XX_SPI_TX_DATA ,
xfer - > tx_buf , xfer - > len ) ;
break ;
}
2009-11-30 07:39:42 +00:00
}
}
if ( xfer - > rx_buf ! = NULL ) {
sdd - > state | = RXBUSY ;
2012-07-13 07:15:14 +09:00
if ( sdd - > port_conf - > high_speed & & sdd - > cur_speed > = 30000000UL
2009-11-30 07:39:42 +00:00
& & ! ( sdd - > cur_mode & SPI_CPHA ) )
chcfg | = S3C64XX_SPI_CH_HS_EN ;
if ( dma_mode ) {
modecfg | = S3C64XX_SPI_MODE_RXDMA_ON ;
chcfg | = S3C64XX_SPI_CH_RXCH_ON ;
writel ( ( ( xfer - > len * 8 / sdd - > cur_bpw ) & 0xffff )
| S3C64XX_SPI_PACKET_CNT_EN ,
regs + S3C64XX_SPI_PACKET_CNT ) ;
2011-09-02 09:44:42 +09:00
prepare_dma ( & sdd - > rx_dma , xfer - > len , xfer - > rx_dma ) ;
2009-11-30 07:39:42 +00:00
}
}
writel ( modecfg , regs + S3C64XX_SPI_MODE_CFG ) ;
writel ( chcfg , regs + S3C64XX_SPI_CH_CFG ) ;
}
static inline void enable_cs ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_device * spi )
{
struct s3c64xx_spi_csinfo * cs ;
if ( sdd - > tgl_spi ! = NULL ) { /* If last device toggled after mssg */
if ( sdd - > tgl_spi ! = spi ) { /* if last mssg on diff device */
/* Deselect the last toggled device */
cs = sdd - > tgl_spi - > controller_data ;
2013-06-21 11:26:12 +05:30
if ( sdd - > cs_gpio )
gpio_set_value ( cs - > line ,
spi - > mode & SPI_CS_HIGH ? 0 : 1 ) ;
2009-11-30 07:39:42 +00:00
}
sdd - > tgl_spi = NULL ;
}
cs = spi - > controller_data ;
2013-06-21 11:26:12 +05:30
if ( sdd - > cs_gpio )
gpio_set_value ( cs - > line , spi - > mode & SPI_CS_HIGH ? 1 : 0 ) ;
2013-05-20 12:21:32 +05:30
/* Start the signals */
writel ( 0 , sdd - > regs + S3C64XX_SPI_SLAVE_SEL ) ;
}
2013-06-19 19:12:39 +01:00
static u32 s3c64xx_spi_wait_for_timeout ( struct s3c64xx_spi_driver_data * sdd ,
2013-05-20 12:21:32 +05:30
int timeout_ms )
{
void __iomem * regs = sdd - > regs ;
unsigned long val = 1 ;
u32 status ;
/* max fifo depth available */
u32 max_fifo = ( FIFO_LVL_MASK ( sdd ) > > 1 ) + 1 ;
if ( timeout_ms )
val = msecs_to_loops ( timeout_ms ) ;
do {
status = readl ( regs + S3C64XX_SPI_STATUS ) ;
} while ( RX_FIFO_LVL ( status , sdd ) < max_fifo & & - - val ) ;
/* return the actual received data length */
return RX_FIFO_LVL ( status , sdd ) ;
2009-11-30 07:39:42 +00:00
}
static int wait_for_xfer ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_transfer * xfer , int dma_mode )
{
void __iomem * regs = sdd - > regs ;
unsigned long val ;
int ms ;
/* millisecs to xfer 'len' bytes @ 'cur_speed' */
ms = xfer - > len * 8 * 1000 / sdd - > cur_speed ;
2010-09-07 16:37:52 +01:00
ms + = 10 ; /* some tolerance */
2009-11-30 07:39:42 +00:00
if ( dma_mode ) {
val = msecs_to_jiffies ( ms ) + 10 ;
val = wait_for_completion_timeout ( & sdd - > xfer_completion , val ) ;
} else {
2010-09-03 10:36:46 +09:00
u32 status ;
2009-11-30 07:39:42 +00:00
val = msecs_to_loops ( ms ) ;
do {
2010-09-03 10:36:46 +09:00
status = readl ( regs + S3C64XX_SPI_STATUS ) ;
2012-07-13 07:15:14 +09:00
} while ( RX_FIFO_LVL ( status , sdd ) < xfer - > len & & - - val ) ;
2009-11-30 07:39:42 +00:00
}
if ( dma_mode ) {
u32 status ;
/*
2013-05-20 12:21:32 +05:30
* If the previous xfer was completed within timeout , then
* proceed further else return - EIO .
2009-11-30 07:39:42 +00:00
* DmaTx returns after simply writing data in the FIFO ,
* w / o waiting for real transmission on the bus to finish .
* DmaRx returns only after Dma read data from FIFO which
* needs bus transmission to finish , so we don ' t worry if
* Xfer involved Rx ( with or without Tx ) .
*/
2013-05-20 12:21:32 +05:30
if ( val & & ! xfer - > rx_buf ) {
2009-11-30 07:39:42 +00:00
val = msecs_to_loops ( 10 ) ;
status = readl ( regs + S3C64XX_SPI_STATUS ) ;
2012-07-13 07:15:14 +09:00
while ( ( TX_FIFO_LVL ( status , sdd )
| | ! S3C64XX_SPI_ST_TX_DONE ( status , sdd ) )
2009-11-30 07:39:42 +00:00
& & - - val ) {
cpu_relax ( ) ;
status = readl ( regs + S3C64XX_SPI_STATUS ) ;
}
}
2013-05-20 12:21:32 +05:30
/* If timed out while checking rx/tx status return error */
if ( ! val )
return - EIO ;
2009-11-30 07:39:42 +00:00
} else {
2013-05-20 12:21:32 +05:30
int loops ;
u32 cpy_len ;
u8 * buf ;
2009-11-30 07:39:42 +00:00
/* If it was only Tx */
2013-05-20 12:21:32 +05:30
if ( ! xfer - > rx_buf ) {
2009-11-30 07:39:42 +00:00
sdd - > state & = ~ TXBUSY ;
return 0 ;
}
2013-05-20 12:21:32 +05:30
/*
* If the receive length is bigger than the controller fifo
* size , calculate the loops and read the fifo as many times .
* loops = length / max fifo size ( calculated by using the
* fifo mask ) .
* For any size less than the fifo size the below code is
* executed atleast once .
*/
loops = xfer - > len / ( ( FIFO_LVL_MASK ( sdd ) > > 1 ) + 1 ) ;
buf = xfer - > rx_buf ;
do {
/* wait for data to be received in the fifo */
2013-06-19 19:12:39 +01:00
cpy_len = s3c64xx_spi_wait_for_timeout ( sdd ,
( loops ? ms : 0 ) ) ;
2013-05-20 12:21:32 +05:30
switch ( sdd - > cur_bpw ) {
case 32 :
ioread32_rep ( regs + S3C64XX_SPI_RX_DATA ,
buf , cpy_len / 4 ) ;
break ;
case 16 :
ioread16_rep ( regs + S3C64XX_SPI_RX_DATA ,
buf , cpy_len / 2 ) ;
break ;
default :
ioread8_rep ( regs + S3C64XX_SPI_RX_DATA ,
buf , cpy_len ) ;
break ;
}
buf = buf + cpy_len ;
} while ( loops - - ) ;
2009-11-30 07:39:42 +00:00
sdd - > state & = ~ RXBUSY ;
}
return 0 ;
}
static inline void disable_cs ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_device * spi )
{
struct s3c64xx_spi_csinfo * cs = spi - > controller_data ;
if ( sdd - > tgl_spi = = spi )
sdd - > tgl_spi = NULL ;
2013-06-21 11:26:12 +05:30
if ( sdd - > cs_gpio )
gpio_set_value ( cs - > line , spi - > mode & SPI_CS_HIGH ? 0 : 1 ) ;
2013-05-20 12:21:32 +05:30
/* Quiese the signals */
writel ( S3C64XX_SPI_SLAVE_SIG_INACT , sdd - > regs + S3C64XX_SPI_SLAVE_SEL ) ;
2009-11-30 07:39:42 +00:00
}
static void s3c64xx_spi_config ( struct s3c64xx_spi_driver_data * sdd )
{
void __iomem * regs = sdd - > regs ;
u32 val ;
/* Disable Clock */
2012-07-13 07:15:14 +09:00
if ( sdd - > port_conf - > clk_from_cmu ) {
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > src_clk ) ;
2010-09-29 17:31:33 +09:00
} else {
val = readl ( regs + S3C64XX_SPI_CLK_CFG ) ;
val & = ~ S3C64XX_SPI_ENCLK_ENABLE ;
writel ( val , regs + S3C64XX_SPI_CLK_CFG ) ;
}
2009-11-30 07:39:42 +00:00
/* Set Polarity and Phase */
val = readl ( regs + S3C64XX_SPI_CH_CFG ) ;
val & = ~ ( S3C64XX_SPI_CH_SLAVE |
S3C64XX_SPI_CPOL_L |
S3C64XX_SPI_CPHA_B ) ;
if ( sdd - > cur_mode & SPI_CPOL )
val | = S3C64XX_SPI_CPOL_L ;
if ( sdd - > cur_mode & SPI_CPHA )
val | = S3C64XX_SPI_CPHA_B ;
writel ( val , regs + S3C64XX_SPI_CH_CFG ) ;
/* Set Channel & DMA Mode */
val = readl ( regs + S3C64XX_SPI_MODE_CFG ) ;
val & = ~ ( S3C64XX_SPI_MODE_BUS_TSZ_MASK
| S3C64XX_SPI_MODE_CH_TSZ_MASK ) ;
switch ( sdd - > cur_bpw ) {
case 32 :
val | = S3C64XX_SPI_MODE_BUS_TSZ_WORD ;
2010-09-29 17:31:33 +09:00
val | = S3C64XX_SPI_MODE_CH_TSZ_WORD ;
2009-11-30 07:39:42 +00:00
break ;
case 16 :
val | = S3C64XX_SPI_MODE_BUS_TSZ_HALFWORD ;
2010-09-29 17:31:33 +09:00
val | = S3C64XX_SPI_MODE_CH_TSZ_HALFWORD ;
2009-11-30 07:39:42 +00:00
break ;
default :
val | = S3C64XX_SPI_MODE_BUS_TSZ_BYTE ;
2010-09-29 17:31:33 +09:00
val | = S3C64XX_SPI_MODE_CH_TSZ_BYTE ;
2009-11-30 07:39:42 +00:00
break ;
}
writel ( val , regs + S3C64XX_SPI_MODE_CFG ) ;
2012-07-13 07:15:14 +09:00
if ( sdd - > port_conf - > clk_from_cmu ) {
2010-09-29 17:31:33 +09:00
/* Configure Clock */
/* There is half-multiplier before the SPI */
clk_set_rate ( sdd - > src_clk , sdd - > cur_speed * 2 ) ;
/* Enable Clock */
2012-10-03 08:30:12 +09:00
clk_prepare_enable ( sdd - > src_clk ) ;
2010-09-29 17:31:33 +09:00
} else {
/* Configure Clock */
val = readl ( regs + S3C64XX_SPI_CLK_CFG ) ;
val & = ~ S3C64XX_SPI_PSR_MASK ;
val | = ( ( clk_get_rate ( sdd - > src_clk ) / sdd - > cur_speed / 2 - 1 )
& S3C64XX_SPI_PSR_MASK ) ;
writel ( val , regs + S3C64XX_SPI_CLK_CFG ) ;
/* Enable Clock */
val = readl ( regs + S3C64XX_SPI_CLK_CFG ) ;
val | = S3C64XX_SPI_ENCLK_ENABLE ;
writel ( val , regs + S3C64XX_SPI_CLK_CFG ) ;
}
2009-11-30 07:39:42 +00:00
}
# define XFER_DMAADDR_INVALID DMA_BIT_MASK(32)
static int s3c64xx_spi_map_mssg ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_message * msg )
{
struct device * dev = & sdd - > pdev - > dev ;
struct spi_transfer * xfer ;
2013-05-20 12:21:32 +05:30
if ( is_polling ( sdd ) | | msg - > is_dma_mapped )
2009-11-30 07:39:42 +00:00
return 0 ;
/* First mark all xfer unmapped */
list_for_each_entry ( xfer , & msg - > transfers , transfer_list ) {
xfer - > rx_dma = XFER_DMAADDR_INVALID ;
xfer - > tx_dma = XFER_DMAADDR_INVALID ;
}
/* Map until end or first fail */
list_for_each_entry ( xfer , & msg - > transfers , transfer_list ) {
2012-07-13 07:15:14 +09:00
if ( xfer - > len < = ( ( FIFO_LVL_MASK ( sdd ) > > 1 ) + 1 ) )
2010-09-29 17:31:31 +09:00
continue ;
2009-11-30 07:39:42 +00:00
if ( xfer - > tx_buf ! = NULL ) {
2010-09-03 10:36:26 +09:00
xfer - > tx_dma = dma_map_single ( dev ,
( void * ) xfer - > tx_buf , xfer - > len ,
DMA_TO_DEVICE ) ;
2009-11-30 07:39:42 +00:00
if ( dma_mapping_error ( dev , xfer - > tx_dma ) ) {
dev_err ( dev , " dma_map_single Tx failed \n " ) ;
xfer - > tx_dma = XFER_DMAADDR_INVALID ;
return - ENOMEM ;
}
}
if ( xfer - > rx_buf ! = NULL ) {
xfer - > rx_dma = dma_map_single ( dev , xfer - > rx_buf ,
xfer - > len , DMA_FROM_DEVICE ) ;
if ( dma_mapping_error ( dev , xfer - > rx_dma ) ) {
dev_err ( dev , " dma_map_single Rx failed \n " ) ;
dma_unmap_single ( dev , xfer - > tx_dma ,
xfer - > len , DMA_TO_DEVICE ) ;
xfer - > tx_dma = XFER_DMAADDR_INVALID ;
xfer - > rx_dma = XFER_DMAADDR_INVALID ;
return - ENOMEM ;
}
}
}
return 0 ;
}
static void s3c64xx_spi_unmap_mssg ( struct s3c64xx_spi_driver_data * sdd ,
struct spi_message * msg )
{
struct device * dev = & sdd - > pdev - > dev ;
struct spi_transfer * xfer ;
2013-05-20 12:21:32 +05:30
if ( is_polling ( sdd ) | | msg - > is_dma_mapped )
2009-11-30 07:39:42 +00:00
return ;
list_for_each_entry ( xfer , & msg - > transfers , transfer_list ) {
2012-07-13 07:15:14 +09:00
if ( xfer - > len < = ( ( FIFO_LVL_MASK ( sdd ) > > 1 ) + 1 ) )
2010-09-29 17:31:31 +09:00
continue ;
2009-11-30 07:39:42 +00:00
if ( xfer - > rx_buf ! = NULL
& & xfer - > rx_dma ! = XFER_DMAADDR_INVALID )
dma_unmap_single ( dev , xfer - > rx_dma ,
xfer - > len , DMA_FROM_DEVICE ) ;
if ( xfer - > tx_buf ! = NULL
& & xfer - > tx_dma ! = XFER_DMAADDR_INVALID )
dma_unmap_single ( dev , xfer - > tx_dma ,
xfer - > len , DMA_TO_DEVICE ) ;
}
}
2012-02-15 14:48:32 -08:00
static int s3c64xx_spi_transfer_one_message ( struct spi_master * master ,
struct spi_message * msg )
2009-11-30 07:39:42 +00:00
{
2012-02-15 14:48:32 -08:00
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2009-11-30 07:39:42 +00:00
struct spi_device * spi = msg - > spi ;
struct s3c64xx_spi_csinfo * cs = spi - > controller_data ;
struct spi_transfer * xfer ;
int status = 0 , cs_toggle = 0 ;
u32 speed ;
u8 bpw ;
/* If Master's(controller) state differs from that needed by Slave */
if ( sdd - > cur_speed ! = spi - > max_speed_hz
| | sdd - > cur_mode ! = spi - > mode
| | sdd - > cur_bpw ! = spi - > bits_per_word ) {
sdd - > cur_bpw = spi - > bits_per_word ;
sdd - > cur_speed = spi - > max_speed_hz ;
sdd - > cur_mode = spi - > mode ;
s3c64xx_spi_config ( sdd ) ;
}
/* Map all the transfers if needed */
if ( s3c64xx_spi_map_mssg ( sdd , msg ) ) {
dev_err ( & spi - > dev ,
" Xfer: Unable to map message buffers! \n " ) ;
status = - ENOMEM ;
goto out ;
}
/* Configure feedback delay */
writel ( cs - > fb_delay & 0x3 , sdd - > regs + S3C64XX_SPI_FB_CLK ) ;
list_for_each_entry ( xfer , & msg - > transfers , transfer_list ) {
unsigned long flags ;
int use_dma ;
INIT_COMPLETION ( sdd - > xfer_completion ) ;
/* Only BPW and Speed may change across transfers */
2012-12-18 14:25:43 +05:30
bpw = xfer - > bits_per_word ;
2009-11-30 07:39:42 +00:00
speed = xfer - > speed_hz ? : spi - > max_speed_hz ;
2010-09-29 17:31:33 +09:00
if ( xfer - > len % ( bpw / 8 ) ) {
dev_err ( & spi - > dev ,
" Xfer length(%u) not a multiple of word size(%u) \n " ,
xfer - > len , bpw / 8 ) ;
status = - EIO ;
goto out ;
}
2009-11-30 07:39:42 +00:00
if ( bpw ! = sdd - > cur_bpw | | speed ! = sdd - > cur_speed ) {
sdd - > cur_bpw = bpw ;
sdd - > cur_speed = speed ;
s3c64xx_spi_config ( sdd ) ;
}
/* Polling method for xfers not bigger than FIFO capacity */
2013-04-11 22:42:03 +02:00
use_dma = 0 ;
2013-05-20 12:21:32 +05:30
if ( ! is_polling ( sdd ) & &
( sdd - > rx_dma . ch & & sdd - > tx_dma . ch & &
( xfer - > len > ( ( FIFO_LVL_MASK ( sdd ) > > 1 ) + 1 ) ) ) )
2009-11-30 07:39:42 +00:00
use_dma = 1 ;
spin_lock_irqsave ( & sdd - > lock , flags ) ;
/* Pending only which is to be done */
sdd - > state & = ~ RXBUSY ;
sdd - > state & = ~ TXBUSY ;
enable_datapath ( sdd , spi , xfer , use_dma ) ;
/* Slave Select */
enable_cs ( sdd , spi ) ;
spin_unlock_irqrestore ( & sdd - > lock , flags ) ;
status = wait_for_xfer ( sdd , xfer , use_dma ) ;
if ( status ) {
2013-01-31 15:25:01 +09:00
dev_err ( & spi - > dev , " I/O Error: rx-%d tx-%d res:rx-%c tx-%c len-%d \n " ,
2009-11-30 07:39:42 +00:00
xfer - > rx_buf ? 1 : 0 , xfer - > tx_buf ? 1 : 0 ,
( sdd - > state & RXBUSY ) ? ' f ' : ' p ' ,
( sdd - > state & TXBUSY ) ? ' f ' : ' p ' ,
xfer - > len ) ;
if ( use_dma ) {
if ( xfer - > tx_buf ! = NULL
& & ( sdd - > state & TXBUSY ) )
2013-04-11 22:42:03 +02:00
s3c64xx_spi_dma_stop ( sdd , & sdd - > tx_dma ) ;
2009-11-30 07:39:42 +00:00
if ( xfer - > rx_buf ! = NULL
& & ( sdd - > state & RXBUSY ) )
2013-04-11 22:42:03 +02:00
s3c64xx_spi_dma_stop ( sdd , & sdd - > rx_dma ) ;
2009-11-30 07:39:42 +00:00
}
goto out ;
}
if ( xfer - > delay_usecs )
udelay ( xfer - > delay_usecs ) ;
if ( xfer - > cs_change ) {
/* Hint that the next mssg is gonna be
for the same device */
if ( list_is_last ( & xfer - > transfer_list ,
& msg - > transfers ) )
cs_toggle = 1 ;
}
msg - > actual_length + = xfer - > len ;
flush_fifo ( sdd ) ;
}
out :
if ( ! cs_toggle | | status )
disable_cs ( sdd , spi ) ;
else
sdd - > tgl_spi = spi ;
s3c64xx_spi_unmap_mssg ( sdd , msg ) ;
msg - > status = status ;
2012-02-15 14:48:32 -08:00
spi_finalize_current_message ( master ) ;
return 0 ;
2009-11-30 07:39:42 +00:00
}
2012-07-13 07:15:15 +09:00
static struct s3c64xx_spi_csinfo * s3c64xx_get_slave_ctrldata (
struct spi_device * spi )
{
struct s3c64xx_spi_csinfo * cs ;
2012-08-04 11:18:20 +00:00
struct device_node * slave_np , * data_np = NULL ;
2013-06-21 11:26:12 +05:30
struct s3c64xx_spi_driver_data * sdd ;
2012-07-13 07:15:15 +09:00
u32 fb_delay = 0 ;
2013-06-21 11:26:12 +05:30
sdd = spi_master_get_devdata ( spi - > master ) ;
2012-07-13 07:15:15 +09:00
slave_np = spi - > dev . of_node ;
if ( ! slave_np ) {
dev_err ( & spi - > dev , " device node not found \n " ) ;
return ERR_PTR ( - EINVAL ) ;
}
2012-09-18 08:10:49 +01:00
data_np = of_get_child_by_name ( slave_np , " controller-data " ) ;
2012-07-13 07:15:15 +09:00
if ( ! data_np ) {
dev_err ( & spi - > dev , " child node 'controller-data' not found \n " ) ;
return ERR_PTR ( - EINVAL ) ;
}
cs = kzalloc ( sizeof ( * cs ) , GFP_KERNEL ) ;
if ( ! cs ) {
2013-01-31 15:25:01 +09:00
dev_err ( & spi - > dev , " could not allocate memory for controller data \n " ) ;
2012-09-18 08:10:49 +01:00
of_node_put ( data_np ) ;
2012-07-13 07:15:15 +09:00
return ERR_PTR ( - ENOMEM ) ;
}
2013-06-21 11:26:12 +05:30
/* The CS line is asserted/deasserted by the gpio pin */
if ( sdd - > cs_gpio )
cs - > line = of_get_named_gpio ( data_np , " cs-gpio " , 0 ) ;
2012-07-13 07:15:15 +09:00
if ( ! gpio_is_valid ( cs - > line ) ) {
2013-01-31 15:25:01 +09:00
dev_err ( & spi - > dev , " chip select gpio is not specified or invalid \n " ) ;
2012-07-13 07:15:15 +09:00
kfree ( cs ) ;
2012-09-18 08:10:49 +01:00
of_node_put ( data_np ) ;
2012-07-13 07:15:15 +09:00
return ERR_PTR ( - EINVAL ) ;
}
of_property_read_u32 ( data_np , " samsung,spi-feedback-delay " , & fb_delay ) ;
cs - > fb_delay = fb_delay ;
2012-09-18 08:10:49 +01:00
of_node_put ( data_np ) ;
2012-07-13 07:15:15 +09:00
return cs ;
}
2009-11-30 07:39:42 +00:00
/*
* Here we only check the validity of requested configuration
* and save the configuration in a local data - structure .
* The controller is actually configured only just before we
* get a message to transfer .
*/
static int s3c64xx_spi_setup ( struct spi_device * spi )
{
struct s3c64xx_spi_csinfo * cs = spi - > controller_data ;
struct s3c64xx_spi_driver_data * sdd ;
2010-01-20 13:49:44 -07:00
struct s3c64xx_spi_info * sci ;
2009-11-30 07:39:42 +00:00
struct spi_message * msg ;
unsigned long flags ;
2012-07-13 07:15:15 +09:00
int err ;
2009-11-30 07:39:42 +00:00
2012-07-13 07:15:15 +09:00
sdd = spi_master_get_devdata ( spi - > master ) ;
if ( ! cs & & spi - > dev . of_node ) {
2013-03-26 10:27:35 +01:00
cs = s3c64xx_get_slave_ctrldata ( spi ) ;
2012-07-13 07:15:15 +09:00
spi - > controller_data = cs ;
}
if ( IS_ERR_OR_NULL ( cs ) ) {
2009-11-30 07:39:42 +00:00
dev_err ( & spi - > dev , " No CS for SPI(%d) \n " , spi - > chip_select ) ;
return - ENODEV ;
}
2013-06-21 11:26:12 +05:30
/* Request gpio only if cs line is asserted by gpio pins */
if ( sdd - > cs_gpio ) {
2012-07-19 14:36:16 +09:00
err = gpio_request_one ( cs - > line , GPIOF_OUT_INIT_HIGH ,
dev_name ( & spi - > dev ) ) ;
2012-07-13 07:15:14 +09:00
if ( err ) {
2012-07-19 14:36:13 +09:00
dev_err ( & spi - > dev ,
" Failed to get /CS gpio [%d]: %d \n " ,
cs - > line , err ) ;
2012-07-13 07:15:15 +09:00
goto err_gpio_req ;
2012-07-13 07:15:14 +09:00
}
}
2013-06-21 11:26:12 +05:30
if ( ! spi_get_ctldata ( spi ) )
spi_set_ctldata ( spi , cs ) ;
2009-11-30 07:39:42 +00:00
sci = sdd - > cntrlr_info ;
spin_lock_irqsave ( & sdd - > lock , flags ) ;
list_for_each_entry ( msg , & sdd - > queue , queue ) {
/* Is some mssg is already queued for this device */
if ( msg - > spi = = spi ) {
dev_err ( & spi - > dev ,
" setup: attempt while mssg in queue! \n " ) ;
spin_unlock_irqrestore ( & sdd - > lock , flags ) ;
2012-07-13 07:15:15 +09:00
err = - EBUSY ;
goto err_msgq ;
2009-11-30 07:39:42 +00:00
}
}
spin_unlock_irqrestore ( & sdd - > lock , flags ) ;
2011-12-04 00:58:06 +00:00
pm_runtime_get_sync ( & sdd - > pdev - > dev ) ;
2009-11-30 07:39:42 +00:00
/* Check if we can provide the requested rate */
2012-07-13 07:15:14 +09:00
if ( ! sdd - > port_conf - > clk_from_cmu ) {
2010-09-29 17:31:33 +09:00
u32 psr , speed ;
/* Max possible */
speed = clk_get_rate ( sdd - > src_clk ) / 2 / ( 0 + 1 ) ;
if ( spi - > max_speed_hz > speed )
spi - > max_speed_hz = speed ;
psr = clk_get_rate ( sdd - > src_clk ) / 2 / spi - > max_speed_hz - 1 ;
psr & = S3C64XX_SPI_PSR_MASK ;
if ( psr = = S3C64XX_SPI_PSR_MASK )
psr - - ;
speed = clk_get_rate ( sdd - > src_clk ) / 2 / ( psr + 1 ) ;
if ( spi - > max_speed_hz < speed ) {
if ( psr + 1 < S3C64XX_SPI_PSR_MASK ) {
psr + + ;
} else {
err = - EINVAL ;
goto setup_exit ;
}
}
2009-11-30 07:39:42 +00:00
2010-09-29 17:31:33 +09:00
speed = clk_get_rate ( sdd - > src_clk ) / 2 / ( psr + 1 ) ;
2012-07-13 07:15:15 +09:00
if ( spi - > max_speed_hz > = speed ) {
2010-09-29 17:31:33 +09:00
spi - > max_speed_hz = speed ;
2012-07-13 07:15:15 +09:00
} else {
2012-12-20 18:27:31 +00:00
dev_err ( & spi - > dev , " Can't set %dHz transfer speed \n " ,
spi - > max_speed_hz ) ;
2009-11-30 07:39:42 +00:00
err = - EINVAL ;
2012-07-13 07:15:15 +09:00
goto setup_exit ;
}
2009-11-30 07:39:42 +00:00
}
2011-12-04 00:58:06 +00:00
pm_runtime_put ( & sdd - > pdev - > dev ) ;
2012-07-13 07:15:15 +09:00
disable_cs ( sdd , spi ) ;
return 0 ;
2011-12-04 00:58:06 +00:00
2009-11-30 07:39:42 +00:00
setup_exit :
/* setup() returns with device de-selected */
disable_cs ( sdd , spi ) ;
2012-07-13 07:15:15 +09:00
err_msgq :
gpio_free ( cs - > line ) ;
spi_set_ctldata ( spi , NULL ) ;
err_gpio_req :
2012-09-13 16:31:30 +02:00
if ( spi - > dev . of_node )
kfree ( cs ) ;
2012-07-13 07:15:15 +09:00
2009-11-30 07:39:42 +00:00
return err ;
}
2012-07-13 07:15:14 +09:00
static void s3c64xx_spi_cleanup ( struct spi_device * spi )
{
struct s3c64xx_spi_csinfo * cs = spi_get_ctldata ( spi ) ;
2013-06-21 11:26:12 +05:30
struct s3c64xx_spi_driver_data * sdd ;
2012-07-13 07:15:14 +09:00
2013-06-21 11:26:12 +05:30
sdd = spi_master_get_devdata ( spi - > master ) ;
if ( cs & & sdd - > cs_gpio ) {
2012-07-13 07:15:14 +09:00
gpio_free ( cs - > line ) ;
2012-07-13 07:15:15 +09:00
if ( spi - > dev . of_node )
kfree ( cs ) ;
}
2012-07-13 07:15:14 +09:00
spi_set_ctldata ( spi , NULL ) ;
}
2011-11-10 10:57:32 +00:00
static irqreturn_t s3c64xx_spi_irq ( int irq , void * data )
{
struct s3c64xx_spi_driver_data * sdd = data ;
struct spi_master * spi = sdd - > master ;
2013-03-13 12:13:30 +05:30
unsigned int val , clr = 0 ;
2011-11-10 10:57:32 +00:00
2013-03-13 12:13:30 +05:30
val = readl ( sdd - > regs + S3C64XX_SPI_STATUS ) ;
2011-11-10 10:57:32 +00:00
2013-03-13 12:13:30 +05:30
if ( val & S3C64XX_SPI_ST_RX_OVERRUN_ERR ) {
clr = S3C64XX_SPI_PND_RX_OVERRUN_CLR ;
2011-11-10 10:57:32 +00:00
dev_err ( & spi - > dev , " RX overrun \n " ) ;
2013-03-13 12:13:30 +05:30
}
if ( val & S3C64XX_SPI_ST_RX_UNDERRUN_ERR ) {
clr | = S3C64XX_SPI_PND_RX_UNDERRUN_CLR ;
2011-11-10 10:57:32 +00:00
dev_err ( & spi - > dev , " RX underrun \n " ) ;
2013-03-13 12:13:30 +05:30
}
if ( val & S3C64XX_SPI_ST_TX_OVERRUN_ERR ) {
clr | = S3C64XX_SPI_PND_TX_OVERRUN_CLR ;
2011-11-10 10:57:32 +00:00
dev_err ( & spi - > dev , " TX overrun \n " ) ;
2013-03-13 12:13:30 +05:30
}
if ( val & S3C64XX_SPI_ST_TX_UNDERRUN_ERR ) {
clr | = S3C64XX_SPI_PND_TX_UNDERRUN_CLR ;
2011-11-10 10:57:32 +00:00
dev_err ( & spi - > dev , " TX underrun \n " ) ;
2013-03-13 12:13:30 +05:30
}
/* Clear the pending irq by setting and then clearing it */
writel ( clr , sdd - > regs + S3C64XX_SPI_PENDING_CLR ) ;
writel ( 0 , sdd - > regs + S3C64XX_SPI_PENDING_CLR ) ;
2011-11-10 10:57:32 +00:00
return IRQ_HANDLED ;
}
2009-11-30 07:39:42 +00:00
static void s3c64xx_spi_hwinit ( struct s3c64xx_spi_driver_data * sdd , int channel )
{
2010-01-20 13:49:44 -07:00
struct s3c64xx_spi_info * sci = sdd - > cntrlr_info ;
2009-11-30 07:39:42 +00:00
void __iomem * regs = sdd - > regs ;
unsigned int val ;
sdd - > cur_speed = 0 ;
2012-07-19 14:36:23 +09:00
writel ( S3C64XX_SPI_SLAVE_SIG_INACT , sdd - > regs + S3C64XX_SPI_SLAVE_SEL ) ;
2009-11-30 07:39:42 +00:00
/* Disable Interrupts - we use Polling if not DMA mode */
writel ( 0 , regs + S3C64XX_SPI_INT_EN ) ;
2012-07-13 07:15:14 +09:00
if ( ! sdd - > port_conf - > clk_from_cmu )
2010-09-29 17:31:33 +09:00
writel ( sci - > src_clk_nr < < S3C64XX_SPI_CLKSEL_SRCSHFT ,
2009-11-30 07:39:42 +00:00
regs + S3C64XX_SPI_CLK_CFG ) ;
writel ( 0 , regs + S3C64XX_SPI_MODE_CFG ) ;
writel ( 0 , regs + S3C64XX_SPI_PACKET_CNT ) ;
2013-03-13 12:13:30 +05:30
/* Clear any irq pending bits, should set and clear the bits */
val = S3C64XX_SPI_PND_RX_OVERRUN_CLR |
S3C64XX_SPI_PND_RX_UNDERRUN_CLR |
S3C64XX_SPI_PND_TX_OVERRUN_CLR |
S3C64XX_SPI_PND_TX_UNDERRUN_CLR ;
writel ( val , regs + S3C64XX_SPI_PENDING_CLR ) ;
writel ( 0 , regs + S3C64XX_SPI_PENDING_CLR ) ;
2009-11-30 07:39:42 +00:00
writel ( 0 , regs + S3C64XX_SPI_SWAP_CFG ) ;
val = readl ( regs + S3C64XX_SPI_MODE_CFG ) ;
val & = ~ S3C64XX_SPI_MODE_4BURST ;
val & = ~ ( S3C64XX_SPI_MAX_TRAILCNT < < S3C64XX_SPI_TRAILCNT_OFF ) ;
val | = ( S3C64XX_SPI_TRAILCNT < < S3C64XX_SPI_TRAILCNT_OFF ) ;
writel ( val , regs + S3C64XX_SPI_MODE_CFG ) ;
flush_fifo ( sdd ) ;
}
2012-07-13 07:15:15 +09:00
# ifdef CONFIG_OF
2013-01-31 15:25:01 +09:00
static struct s3c64xx_spi_info * s3c64xx_spi_parse_dt ( struct device * dev )
2012-07-13 07:15:15 +09:00
{
struct s3c64xx_spi_info * sci ;
u32 temp ;
sci = devm_kzalloc ( dev , sizeof ( * sci ) , GFP_KERNEL ) ;
if ( ! sci ) {
dev_err ( dev , " memory allocation for spi_info failed \n " ) ;
return ERR_PTR ( - ENOMEM ) ;
}
if ( of_property_read_u32 ( dev - > of_node , " samsung,spi-src-clk " , & temp ) ) {
2013-01-31 15:25:01 +09:00
dev_warn ( dev , " spi bus clock parent not specified, using clock at index 0 as parent \n " ) ;
2012-07-13 07:15:15 +09:00
sci - > src_clk_nr = 0 ;
} else {
sci - > src_clk_nr = temp ;
}
if ( of_property_read_u32 ( dev - > of_node , " num-cs " , & temp ) ) {
2013-01-31 15:25:01 +09:00
dev_warn ( dev , " number of chip select lines not specified, assuming 1 chip select line \n " ) ;
2012-07-13 07:15:15 +09:00
sci - > num_cs = 1 ;
} else {
sci - > num_cs = temp ;
}
return sci ;
}
# else
static struct s3c64xx_spi_info * s3c64xx_spi_parse_dt ( struct device * dev )
{
return dev - > platform_data ;
}
# endif
static const struct of_device_id s3c64xx_spi_dt_match [ ] ;
2012-07-13 07:15:14 +09:00
static inline struct s3c64xx_spi_port_config * s3c64xx_spi_get_port_config (
struct platform_device * pdev )
{
2012-07-13 07:15:15 +09:00
# ifdef CONFIG_OF
if ( pdev - > dev . of_node ) {
const struct of_device_id * match ;
match = of_match_node ( s3c64xx_spi_dt_match , pdev - > dev . of_node ) ;
return ( struct s3c64xx_spi_port_config * ) match - > data ;
}
# endif
2012-07-13 07:15:14 +09:00
return ( struct s3c64xx_spi_port_config * )
platform_get_device_id ( pdev ) - > driver_data ;
}
2013-02-05 13:27:35 +00:00
static int s3c64xx_spi_probe ( struct platform_device * pdev )
2009-11-30 07:39:42 +00:00
{
2012-07-13 07:15:15 +09:00
struct resource * mem_res ;
2013-01-18 17:17:03 +05:30
struct resource * res ;
2009-11-30 07:39:42 +00:00
struct s3c64xx_spi_driver_data * sdd ;
2012-07-13 07:15:15 +09:00
struct s3c64xx_spi_info * sci = pdev - > dev . platform_data ;
2009-11-30 07:39:42 +00:00
struct spi_master * master ;
2011-11-10 10:57:32 +00:00
int ret , irq ;
2011-11-02 20:04:19 +09:00
char clk_name [ 16 ] ;
2009-11-30 07:39:42 +00:00
2012-07-13 07:15:15 +09:00
if ( ! sci & & pdev - > dev . of_node ) {
sci = s3c64xx_spi_parse_dt ( & pdev - > dev ) ;
if ( IS_ERR ( sci ) )
return PTR_ERR ( sci ) ;
2009-11-30 07:39:42 +00:00
}
2012-07-13 07:15:15 +09:00
if ( ! sci ) {
2009-11-30 07:39:42 +00:00
dev_err ( & pdev - > dev , " platform_data missing! \n " ) ;
return - ENODEV ;
}
mem_res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( mem_res = = NULL ) {
dev_err ( & pdev - > dev , " Unable to get SPI MEM resource \n " ) ;
return - ENXIO ;
}
2011-11-10 10:57:32 +00:00
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
dev_warn ( & pdev - > dev , " Failed to get IRQ: %d \n " , irq ) ;
return irq ;
}
2009-11-30 07:39:42 +00:00
master = spi_alloc_master ( & pdev - > dev ,
sizeof ( struct s3c64xx_spi_driver_data ) ) ;
if ( master = = NULL ) {
dev_err ( & pdev - > dev , " Unable to allocate SPI Master \n " ) ;
return - ENOMEM ;
}
platform_set_drvdata ( pdev , master ) ;
sdd = spi_master_get_devdata ( master ) ;
2012-07-13 07:15:14 +09:00
sdd - > port_conf = s3c64xx_spi_get_port_config ( pdev ) ;
2009-11-30 07:39:42 +00:00
sdd - > master = master ;
sdd - > cntrlr_info = sci ;
sdd - > pdev = pdev ;
sdd - > sfr_start = mem_res - > start ;
2013-06-21 11:26:12 +05:30
sdd - > cs_gpio = true ;
2012-07-13 07:15:15 +09:00
if ( pdev - > dev . of_node ) {
2013-06-21 11:26:12 +05:30
if ( ! of_find_property ( pdev - > dev . of_node , " cs-gpio " , NULL ) )
sdd - > cs_gpio = false ;
2012-07-13 07:15:15 +09:00
ret = of_alias_get_id ( pdev - > dev . of_node , " spi " ) ;
if ( ret < 0 ) {
2013-01-31 15:25:01 +09:00
dev_err ( & pdev - > dev , " failed to get alias id, errno %d \n " ,
ret ) ;
2012-07-13 07:15:15 +09:00
goto err0 ;
}
sdd - > port_id = ret ;
} else {
sdd - > port_id = pdev - > id ;
}
2009-11-30 07:39:42 +00:00
sdd - > cur_bpw = 8 ;
2013-01-18 17:17:03 +05:30
if ( ! sdd - > pdev - > dev . of_node ) {
res = platform_get_resource ( pdev , IORESOURCE_DMA , 0 ) ;
if ( ! res ) {
2013-05-20 12:21:32 +05:30
dev_warn ( & pdev - > dev , " Unable to get SPI tx dma "
" resource. Switching to poll mode \n " ) ;
sdd - > port_conf - > quirks = S3C64XX_SPI_QUIRK_POLL ;
} else
sdd - > tx_dma . dmach = res - > start ;
2013-01-18 17:17:03 +05:30
res = platform_get_resource ( pdev , IORESOURCE_DMA , 1 ) ;
if ( ! res ) {
2013-05-20 12:21:32 +05:30
dev_warn ( & pdev - > dev , " Unable to get SPI rx dma "
" resource. Switching to poll mode \n " ) ;
sdd - > port_conf - > quirks = S3C64XX_SPI_QUIRK_POLL ;
} else
sdd - > rx_dma . dmach = res - > start ;
2013-01-18 17:17:03 +05:30
}
2012-07-13 07:15:15 +09:00
2013-01-18 17:17:03 +05:30
sdd - > tx_dma . direction = DMA_MEM_TO_DEV ;
sdd - > rx_dma . direction = DMA_DEV_TO_MEM ;
2012-07-13 07:15:15 +09:00
master - > dev . of_node = pdev - > dev . of_node ;
2012-07-13 07:15:14 +09:00
master - > bus_num = sdd - > port_id ;
2009-11-30 07:39:42 +00:00
master - > setup = s3c64xx_spi_setup ;
2012-07-13 07:15:14 +09:00
master - > cleanup = s3c64xx_spi_cleanup ;
2012-02-15 14:48:32 -08:00
master - > prepare_transfer_hardware = s3c64xx_spi_prepare_transfer ;
master - > transfer_one_message = s3c64xx_spi_transfer_one_message ;
master - > unprepare_transfer_hardware = s3c64xx_spi_unprepare_transfer ;
2009-11-30 07:39:42 +00:00
master - > num_chipselect = sci - > num_cs ;
master - > dma_alignment = 8 ;
2013-05-21 20:36:35 -06:00
master - > bits_per_word_mask = SPI_BPW_MASK ( 32 ) | SPI_BPW_MASK ( 16 ) |
SPI_BPW_MASK ( 8 ) ;
2009-11-30 07:39:42 +00:00
/* the spi->mode bits understood by this driver: */
master - > mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH ;
2013-01-21 11:09:18 +01:00
sdd - > regs = devm_ioremap_resource ( & pdev - > dev , mem_res ) ;
if ( IS_ERR ( sdd - > regs ) ) {
ret = PTR_ERR ( sdd - > regs ) ;
2013-01-10 11:04:21 +09:00
goto err0 ;
2009-11-30 07:39:42 +00:00
}
2013-04-15 20:42:57 -07:00
if ( sci - > cfg_gpio & & sci - > cfg_gpio ( ) ) {
2009-11-30 07:39:42 +00:00
dev_err ( & pdev - > dev , " Unable to config gpio \n " ) ;
ret = - EBUSY ;
2013-01-10 11:04:21 +09:00
goto err0 ;
2009-11-30 07:39:42 +00:00
}
/* Setup clocks */
2013-01-10 11:04:21 +09:00
sdd - > clk = devm_clk_get ( & pdev - > dev , " spi " ) ;
2009-11-30 07:39:42 +00:00
if ( IS_ERR ( sdd - > clk ) ) {
dev_err ( & pdev - > dev , " Unable to acquire clock 'spi' \n " ) ;
ret = PTR_ERR ( sdd - > clk ) ;
2013-04-15 20:42:57 -07:00
goto err0 ;
2009-11-30 07:39:42 +00:00
}
2012-10-03 08:30:12 +09:00
if ( clk_prepare_enable ( sdd - > clk ) ) {
2009-11-30 07:39:42 +00:00
dev_err ( & pdev - > dev , " Couldn't enable clock 'spi' \n " ) ;
ret = - EBUSY ;
2013-04-15 20:42:57 -07:00
goto err0 ;
2009-11-30 07:39:42 +00:00
}
2011-11-02 20:04:19 +09:00
sprintf ( clk_name , " spi_busclk%d " , sci - > src_clk_nr ) ;
2013-01-10 11:04:21 +09:00
sdd - > src_clk = devm_clk_get ( & pdev - > dev , clk_name ) ;
2010-01-20 13:49:44 -07:00
if ( IS_ERR ( sdd - > src_clk ) ) {
2009-11-30 07:39:42 +00:00
dev_err ( & pdev - > dev ,
2011-11-02 20:04:19 +09:00
" Unable to acquire clock '%s' \n " , clk_name ) ;
2010-01-20 13:49:44 -07:00
ret = PTR_ERR ( sdd - > src_clk ) ;
2013-01-10 11:04:21 +09:00
goto err2 ;
2009-11-30 07:39:42 +00:00
}
2012-10-03 08:30:12 +09:00
if ( clk_prepare_enable ( sdd - > src_clk ) ) {
2011-11-02 20:04:19 +09:00
dev_err ( & pdev - > dev , " Couldn't enable clock '%s' \n " , clk_name ) ;
2009-11-30 07:39:42 +00:00
ret = - EBUSY ;
2013-01-10 11:04:21 +09:00
goto err2 ;
2009-11-30 07:39:42 +00:00
}
/* Setup Deufult Mode */
2012-07-13 07:15:14 +09:00
s3c64xx_spi_hwinit ( sdd , sdd - > port_id ) ;
2009-11-30 07:39:42 +00:00
spin_lock_init ( & sdd - > lock ) ;
init_completion ( & sdd - > xfer_completion ) ;
INIT_LIST_HEAD ( & sdd - > queue ) ;
2013-01-10 11:04:21 +09:00
ret = devm_request_irq ( & pdev - > dev , irq , s3c64xx_spi_irq , 0 ,
" spi-s3c64xx " , sdd ) ;
2011-11-10 10:57:32 +00:00
if ( ret ! = 0 ) {
dev_err ( & pdev - > dev , " Failed to request IRQ %d: %d \n " ,
irq , ret ) ;
2013-01-10 11:04:21 +09:00
goto err3 ;
2011-11-10 10:57:32 +00:00
}
writel ( S3C64XX_SPI_INT_RX_OVERRUN_EN | S3C64XX_SPI_INT_RX_UNDERRUN_EN |
S3C64XX_SPI_INT_TX_OVERRUN_EN | S3C64XX_SPI_INT_TX_UNDERRUN_EN ,
sdd - > regs + S3C64XX_SPI_INT_EN ) ;
2009-11-30 07:39:42 +00:00
if ( spi_register_master ( master ) ) {
dev_err ( & pdev - > dev , " cannot register SPI master \n " ) ;
ret = - EBUSY ;
2013-01-10 11:04:21 +09:00
goto err3 ;
2009-11-30 07:39:42 +00:00
}
2013-01-31 15:25:01 +09:00
dev_dbg ( & pdev - > dev , " Samsung SoC SPI Driver loaded for Bus SPI-%d with %d Slaves attached \n " ,
2012-07-13 07:15:14 +09:00
sdd - > port_id , master - > num_chipselect ) ;
2010-02-02 07:22:13 +00:00
dev_dbg ( & pdev - > dev , " \t IOmem=[0x%x-0x%x] \t DMA=[Rx-%d, Tx-%d] \n " ,
2009-11-30 07:39:42 +00:00
mem_res - > end , mem_res - > start ,
2011-09-02 09:44:42 +09:00
sdd - > rx_dma . dmach , sdd - > tx_dma . dmach ) ;
2009-11-30 07:39:42 +00:00
2011-12-04 00:58:06 +00:00
pm_runtime_enable ( & pdev - > dev ) ;
2009-11-30 07:39:42 +00:00
return 0 ;
2013-01-10 11:04:21 +09:00
err3 :
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > src_clk ) ;
2013-01-10 11:04:21 +09:00
err2 :
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > clk ) ;
2009-11-30 07:39:42 +00:00
err0 :
spi_master_put ( master ) ;
return ret ;
}
static int s3c64xx_spi_remove ( struct platform_device * pdev )
{
struct spi_master * master = spi_master_get ( platform_get_drvdata ( pdev ) ) ;
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2011-12-04 00:58:06 +00:00
pm_runtime_disable ( & pdev - > dev ) ;
2009-11-30 07:39:42 +00:00
spi_unregister_master ( master ) ;
2011-11-10 10:57:32 +00:00
writel ( 0 , sdd - > regs + S3C64XX_SPI_INT_EN ) ;
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > src_clk ) ;
2009-11-30 07:39:42 +00:00
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > clk ) ;
2009-11-30 07:39:42 +00:00
spi_master_put ( master ) ;
return 0 ;
}
2013-03-22 02:09:08 +00:00
# ifdef CONFIG_PM_SLEEP
2011-12-04 00:36:18 +00:00
static int s3c64xx_spi_suspend ( struct device * dev )
2009-11-30 07:39:42 +00:00
{
2012-08-16 20:14:25 -07:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2009-11-30 07:39:42 +00:00
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2012-02-15 14:48:32 -08:00
spi_master_suspend ( master ) ;
2009-11-30 07:39:42 +00:00
/* Disable the clock */
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > src_clk ) ;
clk_disable_unprepare ( sdd - > clk ) ;
2009-11-30 07:39:42 +00:00
sdd - > cur_speed = 0 ; /* Output Clock is stopped */
return 0 ;
}
2011-12-04 00:36:18 +00:00
static int s3c64xx_spi_resume ( struct device * dev )
2009-11-30 07:39:42 +00:00
{
2012-08-16 20:14:25 -07:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2009-11-30 07:39:42 +00:00
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2010-01-20 13:49:44 -07:00
struct s3c64xx_spi_info * sci = sdd - > cntrlr_info ;
2009-11-30 07:39:42 +00:00
2013-04-15 20:42:57 -07:00
if ( sci - > cfg_gpio )
2012-07-13 07:15:15 +09:00
sci - > cfg_gpio ( ) ;
2009-11-30 07:39:42 +00:00
/* Enable the clock */
2012-10-03 08:30:12 +09:00
clk_prepare_enable ( sdd - > src_clk ) ;
clk_prepare_enable ( sdd - > clk ) ;
2009-11-30 07:39:42 +00:00
2012-07-13 07:15:14 +09:00
s3c64xx_spi_hwinit ( sdd , sdd - > port_id ) ;
2009-11-30 07:39:42 +00:00
2012-02-15 14:48:32 -08:00
spi_master_resume ( master ) ;
2009-11-30 07:39:42 +00:00
return 0 ;
}
2013-03-22 02:09:08 +00:00
# endif /* CONFIG_PM_SLEEP */
2009-11-30 07:39:42 +00:00
2011-12-04 00:58:06 +00:00
# ifdef CONFIG_PM_RUNTIME
static int s3c64xx_spi_runtime_suspend ( struct device * dev )
{
2012-08-16 20:14:25 -07:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2011-12-04 00:58:06 +00:00
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2012-10-03 08:30:12 +09:00
clk_disable_unprepare ( sdd - > clk ) ;
clk_disable_unprepare ( sdd - > src_clk ) ;
2011-12-04 00:58:06 +00:00
return 0 ;
}
static int s3c64xx_spi_runtime_resume ( struct device * dev )
{
2012-08-16 20:14:25 -07:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2011-12-04 00:58:06 +00:00
struct s3c64xx_spi_driver_data * sdd = spi_master_get_devdata ( master ) ;
2012-10-03 08:30:12 +09:00
clk_prepare_enable ( sdd - > src_clk ) ;
clk_prepare_enable ( sdd - > clk ) ;
2011-12-04 00:58:06 +00:00
return 0 ;
}
# endif /* CONFIG_PM_RUNTIME */
2011-12-04 00:36:18 +00:00
static const struct dev_pm_ops s3c64xx_spi_pm = {
SET_SYSTEM_SLEEP_PM_OPS ( s3c64xx_spi_suspend , s3c64xx_spi_resume )
2011-12-04 00:58:06 +00:00
SET_RUNTIME_PM_OPS ( s3c64xx_spi_runtime_suspend ,
s3c64xx_spi_runtime_resume , NULL )
2011-12-04 00:36:18 +00:00
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config s3c2443_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x7f } ,
. rx_lvl_offset = 13 ,
. tx_st_done = 21 ,
. high_speed = true ,
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config s3c6410_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x7f , 0x7F } ,
. rx_lvl_offset = 13 ,
. tx_st_done = 21 ,
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config s5p64x0_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x1ff , 0x7F } ,
. rx_lvl_offset = 15 ,
. tx_st_done = 25 ,
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config s5pc100_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x7f , 0x7F } ,
. rx_lvl_offset = 13 ,
. tx_st_done = 21 ,
. high_speed = true ,
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config s5pv210_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x1ff , 0x7F } ,
. rx_lvl_offset = 15 ,
. tx_st_done = 25 ,
. high_speed = true ,
} ;
2012-08-03 10:08:12 +05:30
static struct s3c64xx_spi_port_config exynos4_spi_port_config = {
2012-07-13 07:15:14 +09:00
. fifo_lvl_mask = { 0x1ff , 0x7F , 0x7F } ,
. rx_lvl_offset = 15 ,
. tx_st_done = 25 ,
. high_speed = true ,
. clk_from_cmu = true ,
} ;
2013-06-21 11:26:13 +05:30
static struct s3c64xx_spi_port_config exynos5440_spi_port_config = {
. fifo_lvl_mask = { 0x1ff } ,
. rx_lvl_offset = 15 ,
. tx_st_done = 25 ,
. high_speed = true ,
. clk_from_cmu = true ,
. quirks = S3C64XX_SPI_QUIRK_POLL ,
} ;
2012-07-13 07:15:14 +09:00
static struct platform_device_id s3c64xx_spi_driver_ids [ ] = {
{
. name = " s3c2443-spi " ,
. driver_data = ( kernel_ulong_t ) & s3c2443_spi_port_config ,
} , {
. name = " s3c6410-spi " ,
. driver_data = ( kernel_ulong_t ) & s3c6410_spi_port_config ,
} , {
. name = " s5p64x0-spi " ,
. driver_data = ( kernel_ulong_t ) & s5p64x0_spi_port_config ,
} , {
. name = " s5pc100-spi " ,
. driver_data = ( kernel_ulong_t ) & s5pc100_spi_port_config ,
} , {
. name = " s5pv210-spi " ,
. driver_data = ( kernel_ulong_t ) & s5pv210_spi_port_config ,
} , {
. name = " exynos4210-spi " ,
. driver_data = ( kernel_ulong_t ) & exynos4_spi_port_config ,
} ,
{ } ,
} ;
2012-07-13 07:15:15 +09:00
static const struct of_device_id s3c64xx_spi_dt_match [ ] = {
{ . compatible = " samsung,exynos4210-spi " ,
. data = ( void * ) & exynos4_spi_port_config ,
} ,
2013-06-21 11:26:13 +05:30
{ . compatible = " samsung,exynos5440-spi " ,
. data = ( void * ) & exynos5440_spi_port_config ,
} ,
2012-07-13 07:15:15 +09:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , s3c64xx_spi_dt_match ) ;
2009-11-30 07:39:42 +00:00
static struct platform_driver s3c64xx_spi_driver = {
. driver = {
. name = " s3c64xx-spi " ,
. owner = THIS_MODULE ,
2011-12-04 00:36:18 +00:00
. pm = & s3c64xx_spi_pm ,
2012-07-13 07:15:15 +09:00
. of_match_table = of_match_ptr ( s3c64xx_spi_dt_match ) ,
2009-11-30 07:39:42 +00:00
} ,
. remove = s3c64xx_spi_remove ,
2012-07-13 07:15:14 +09:00
. id_table = s3c64xx_spi_driver_ids ,
2009-11-30 07:39:42 +00:00
} ;
MODULE_ALIAS ( " platform:s3c64xx-spi " ) ;
static int __init s3c64xx_spi_init ( void )
{
return platform_driver_probe ( & s3c64xx_spi_driver , s3c64xx_spi_probe ) ;
}
2010-09-07 11:29:17 +01:00
subsys_initcall ( s3c64xx_spi_init ) ;
2009-11-30 07:39:42 +00:00
static void __exit s3c64xx_spi_exit ( void )
{
platform_driver_unregister ( & s3c64xx_spi_driver ) ;
}
module_exit ( s3c64xx_spi_exit ) ;
MODULE_AUTHOR ( " Jaswinder Singh <jassi.brar@samsung.com> " ) ;
MODULE_DESCRIPTION ( " S3C64XX SPI Controller Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;