2012-02-01 14:14:09 +04:00
/*
* Broadcom BCM63xx SPI controller support
*
2012-04-20 17:37:33 +04:00
* Copyright ( C ) 2009 - 2012 Florian Fainelli < florian @ openwrt . org >
2012-02-01 14:14:09 +04:00
* Copyright ( C ) 2010 Tanguy Bouzeloc < tanguy . bouzeloc @ efixo . 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 .
*/
# include <linux/kernel.h>
# include <linux/clk.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/delay.h>
# include <linux/interrupt.h>
# include <linux/spi/spi.h>
# include <linux/completion.h>
# include <linux/err.h>
2012-04-20 17:37:33 +04:00
# include <linux/pm_runtime.h>
2012-02-01 14:14:09 +04:00
# include <bcm63xx_dev_spi.h>
2013-02-03 18:15:13 +04:00
# define BCM63XX_SPI_MAX_PREPEND 15
2012-02-01 14:14:09 +04:00
struct bcm63xx_spi {
struct completion done ;
void __iomem * regs ;
int irq ;
/* Platform data */
unsigned fifo_size ;
2012-06-18 14:07:51 +04:00
unsigned int msg_type_shift ;
unsigned int msg_ctl_width ;
2012-02-01 14:14:09 +04:00
/* data iomem */
u8 __iomem * tx_io ;
const u8 __iomem * rx_io ;
struct clk * clk ;
struct platform_device * pdev ;
} ;
static inline u8 bcm_spi_readb ( struct bcm63xx_spi * bs ,
unsigned int offset )
{
return bcm_readb ( bs - > regs + bcm63xx_spireg ( offset ) ) ;
}
static inline u16 bcm_spi_readw ( struct bcm63xx_spi * bs ,
unsigned int offset )
{
return bcm_readw ( bs - > regs + bcm63xx_spireg ( offset ) ) ;
}
static inline void bcm_spi_writeb ( struct bcm63xx_spi * bs ,
u8 value , unsigned int offset )
{
bcm_writeb ( value , bs - > regs + bcm63xx_spireg ( offset ) ) ;
}
static inline void bcm_spi_writew ( struct bcm63xx_spi * bs ,
u16 value , unsigned int offset )
{
bcm_writew ( value , bs - > regs + bcm63xx_spireg ( offset ) ) ;
}
static const unsigned bcm63xx_spi_freq_table [ SPI_CLK_MASK ] [ 2 ] = {
{ 20000000 , SPI_CLK_20MHZ } ,
{ 12500000 , SPI_CLK_12_50MHZ } ,
{ 6250000 , SPI_CLK_6_250MHZ } ,
{ 3125000 , SPI_CLK_3_125MHZ } ,
{ 1563000 , SPI_CLK_1_563MHZ } ,
{ 781000 , SPI_CLK_0_781MHZ } ,
{ 391000 , SPI_CLK_0_391MHZ }
} ;
2012-04-20 17:37:33 +04:00
static void bcm63xx_spi_setup_transfer ( struct spi_device * spi ,
struct spi_transfer * t )
{
struct bcm63xx_spi * bs = spi_master_get_devdata ( spi - > master ) ;
u8 clk_cfg , reg ;
int i ;
2012-02-01 14:14:09 +04:00
/* Find the closest clock configuration */
for ( i = 0 ; i < SPI_CLK_MASK ; i + + ) {
2013-03-12 03:13:46 +04:00
if ( t - > speed_hz > = bcm63xx_spi_freq_table [ i ] [ 0 ] ) {
2012-02-01 14:14:09 +04:00
clk_cfg = bcm63xx_spi_freq_table [ i ] [ 1 ] ;
break ;
}
}
/* No matching configuration found, default to lowest */
if ( i = = SPI_CLK_MASK )
clk_cfg = SPI_CLK_0_391MHZ ;
/* clear existing clock configuration bits of the register */
reg = bcm_spi_readb ( bs , SPI_CLK_CFG ) ;
reg & = ~ SPI_CLK_MASK ;
reg | = clk_cfg ;
bcm_spi_writeb ( bs , reg , SPI_CLK_CFG ) ;
dev_dbg ( & spi - > dev , " Setting clock register to %02x (hz %d) \n " ,
2013-03-12 03:13:46 +04:00
clk_cfg , t - > speed_hz ) ;
2012-02-01 14:14:09 +04:00
}
/* the spi->mode bits understood by this driver: */
# define MODEBITS (SPI_CPOL | SPI_CPHA)
2013-02-03 18:15:13 +04:00
static int bcm63xx_txrx_bufs ( struct spi_device * spi , struct spi_transfer * first ,
unsigned int num_transfers )
2012-02-01 14:14:09 +04:00
{
struct bcm63xx_spi * bs = spi_master_get_devdata ( spi - > master ) ;
u16 msg_ctl ;
u16 cmd ;
2013-02-03 18:15:12 +04:00
u8 rx_tail ;
2013-02-03 18:15:13 +04:00
unsigned int i , timeout = 0 , prepend_len = 0 , len = 0 ;
struct spi_transfer * t = first ;
bool do_rx = false ;
bool do_tx = false ;
2012-02-01 14:14:09 +04:00
2012-04-20 17:37:33 +04:00
/* Disable the CMD_DONE interrupt */
bcm_spi_writeb ( bs , 0 , SPI_INT_MASK ) ;
2012-02-01 14:14:09 +04:00
dev_dbg ( & spi - > dev , " txrx: tx %p, rx %p, len %d \n " ,
t - > tx_buf , t - > rx_buf , t - > len ) ;
2013-02-03 18:15:13 +04:00
if ( num_transfers > 1 & & t - > tx_buf & & t - > len < = BCM63XX_SPI_MAX_PREPEND )
prepend_len = t - > len ;
/* prepare the buffer */
for ( i = 0 ; i < num_transfers ; i + + ) {
if ( t - > tx_buf ) {
do_tx = true ;
memcpy_toio ( bs - > tx_io + len , t - > tx_buf , t - > len ) ;
/* don't prepend more than one tx */
if ( t ! = first )
prepend_len = 0 ;
}
if ( t - > rx_buf ) {
do_rx = true ;
/* prepend is half-duplex write only */
if ( t = = first )
prepend_len = 0 ;
}
len + = t - > len ;
t = list_entry ( t - > transfer_list . next , struct spi_transfer ,
transfer_list ) ;
}
2014-02-09 07:06:04 +04:00
reinit_completion ( & bs - > done ) ;
2012-02-01 14:14:09 +04:00
/* Fill in the Message control register */
2013-02-03 18:15:13 +04:00
msg_ctl = ( len < < SPI_BYTE_CNT_SHIFT ) ;
2012-02-01 14:14:09 +04:00
2013-02-03 18:15:13 +04:00
if ( do_rx & & do_tx & & prepend_len = = 0 )
2012-06-18 14:07:51 +04:00
msg_ctl | = ( SPI_FD_RW < < bs - > msg_type_shift ) ;
2013-02-03 18:15:13 +04:00
else if ( do_rx )
2012-06-18 14:07:51 +04:00
msg_ctl | = ( SPI_HD_R < < bs - > msg_type_shift ) ;
2013-02-03 18:15:13 +04:00
else if ( do_tx )
2012-06-18 14:07:51 +04:00
msg_ctl | = ( SPI_HD_W < < bs - > msg_type_shift ) ;
switch ( bs - > msg_ctl_width ) {
case 8 :
bcm_spi_writeb ( bs , msg_ctl , SPI_MSG_CTL ) ;
break ;
case 16 :
bcm_spi_writew ( bs , msg_ctl , SPI_MSG_CTL ) ;
break ;
}
2012-02-01 14:14:09 +04:00
/* Issue the transfer */
cmd = SPI_CMD_START_IMMEDIATE ;
2013-02-03 18:15:13 +04:00
cmd | = ( prepend_len < < SPI_CMD_PREPEND_BYTE_CNT_SHIFT ) ;
2012-02-01 14:14:09 +04:00
cmd | = ( spi - > chip_select < < SPI_CMD_DEVICE_ID_SHIFT ) ;
bcm_spi_writew ( bs , cmd , SPI_CMD ) ;
2012-04-20 17:37:33 +04:00
/* Enable the CMD_DONE interrupt */
bcm_spi_writeb ( bs , SPI_INTR_CMD_DONE , SPI_INT_MASK ) ;
2012-02-01 14:14:09 +04:00
2013-02-03 18:15:12 +04:00
timeout = wait_for_completion_timeout ( & bs - > done , HZ ) ;
if ( ! timeout )
return - ETIMEDOUT ;
2013-12-18 00:42:08 +04:00
if ( ! do_rx )
2013-02-03 18:15:13 +04:00
return 0 ;
len = 0 ;
t = first ;
2013-02-03 18:15:12 +04:00
/* Read out all the data */
2013-02-03 18:15:13 +04:00
for ( i = 0 ; i < num_transfers ; i + + ) {
if ( t - > rx_buf )
memcpy_fromio ( t - > rx_buf , bs - > rx_io + len , t - > len ) ;
if ( t ! = first | | prepend_len = = 0 )
len + = t - > len ;
t = list_entry ( t - > transfer_list . next , struct spi_transfer ,
transfer_list ) ;
}
2013-02-03 18:15:12 +04:00
return 0 ;
2012-02-01 14:14:09 +04:00
}
2012-04-20 17:37:33 +04:00
static int bcm63xx_spi_transfer_one ( struct spi_master * master ,
struct spi_message * m )
{
struct bcm63xx_spi * bs = spi_master_get_devdata ( master ) ;
2013-02-03 18:15:13 +04:00
struct spi_transfer * t , * first = NULL ;
2012-04-20 17:37:33 +04:00
struct spi_device * spi = m - > spi ;
int status = 0 ;
2013-02-03 18:15:13 +04:00
unsigned int n_transfers = 0 , total_len = 0 ;
bool can_use_prepend = false ;
/*
* This SPI controller does not support keeping CS active after a
* transfer .
* Work around this by merging as many transfers we can into one big
* full - duplex transfers .
*/
2012-02-01 14:14:09 +04:00
list_for_each_entry ( t , & m - > transfers , transfer_list ) {
2013-02-03 18:15:13 +04:00
if ( ! first )
first = t ;
n_transfers + + ;
total_len + = t - > len ;
if ( n_transfers = = 2 & & ! first - > rx_buf & & ! t - > tx_buf & &
first - > len < = BCM63XX_SPI_MAX_PREPEND )
can_use_prepend = true ;
else if ( can_use_prepend & & t - > tx_buf )
can_use_prepend = false ;
2013-02-03 18:15:12 +04:00
/* we can only transfer one fifo worth of data */
2013-02-03 18:15:13 +04:00
if ( ( can_use_prepend & &
total_len > ( bs - > fifo_size + BCM63XX_SPI_MAX_PREPEND ) ) | |
( ! can_use_prepend & & total_len > bs - > fifo_size ) ) {
2013-02-03 18:15:12 +04:00
dev_err ( & spi - > dev , " unable to do transfers larger than FIFO size (%i > %i) \n " ,
2013-02-03 18:15:13 +04:00
total_len , bs - > fifo_size ) ;
2013-02-03 18:15:12 +04:00
status = - EINVAL ;
goto exit ;
}
2012-04-20 17:37:33 +04:00
2013-02-03 18:15:13 +04:00
/* all combined transfers have to have the same speed */
if ( t - > speed_hz ! = first - > speed_hz ) {
dev_err ( & spi - > dev , " unable to change speed between transfers \n " ) ;
2013-02-03 18:15:12 +04:00
status = - EINVAL ;
goto exit ;
}
2012-04-20 17:37:33 +04:00
2013-02-03 18:15:13 +04:00
/* CS will be deasserted directly after transfer */
if ( t - > delay_usecs ) {
dev_err ( & spi - > dev , " unable to keep CS asserted after transfer \n " ) ;
2013-02-03 18:15:12 +04:00
status = - EINVAL ;
goto exit ;
}
2012-04-20 17:37:33 +04:00
2013-02-03 18:15:13 +04:00
if ( t - > cs_change | |
list_is_last ( & t - > transfer_list , & m - > transfers ) ) {
/* configure adapter for a new transfer */
bcm63xx_spi_setup_transfer ( spi , first ) ;
2012-04-20 17:37:33 +04:00
2013-02-03 18:15:13 +04:00
/* send the data */
status = bcm63xx_txrx_bufs ( spi , first , n_transfers ) ;
if ( status )
goto exit ;
m - > actual_length + = total_len ;
2012-02-01 14:14:09 +04:00
2013-02-03 18:15:13 +04:00
first = NULL ;
n_transfers = 0 ;
total_len = 0 ;
can_use_prepend = false ;
}
2012-04-20 17:37:33 +04:00
}
exit :
m - > status = status ;
spi_finalize_current_message ( master ) ;
2012-02-01 14:14:09 +04:00
2012-04-20 17:37:33 +04:00
return 0 ;
2012-02-01 14:14:09 +04:00
}
/* This driver supports single master mode only. Hence
* CMD_DONE is the only interrupt we care about
*/
static irqreturn_t bcm63xx_spi_interrupt ( int irq , void * dev_id )
{
struct spi_master * master = ( struct spi_master * ) dev_id ;
struct bcm63xx_spi * bs = spi_master_get_devdata ( master ) ;
u8 intr ;
/* Read interupts and clear them immediately */
intr = bcm_spi_readb ( bs , SPI_INT_STATUS ) ;
bcm_spi_writeb ( bs , SPI_INTR_CLEAR_ALL , SPI_INT_STATUS ) ;
bcm_spi_writeb ( bs , 0 , SPI_INT_MASK ) ;
2012-04-20 17:37:33 +04:00
/* A transfer completed */
if ( intr & SPI_INTR_CMD_DONE )
complete ( & bs - > done ) ;
2012-02-01 14:14:09 +04:00
return IRQ_HANDLED ;
}
2012-12-07 20:57:14 +04:00
static int bcm63xx_spi_probe ( struct platform_device * pdev )
2012-02-01 14:14:09 +04:00
{
struct resource * r ;
struct device * dev = & pdev - > dev ;
2013-07-30 11:58:59 +04:00
struct bcm63xx_spi_pdata * pdata = dev_get_platdata ( & pdev - > dev ) ;
2012-02-01 14:14:09 +04:00
int irq ;
struct spi_master * master ;
struct clk * clk ;
struct bcm63xx_spi * bs ;
int ret ;
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
dev_err ( dev , " no irq \n " ) ;
2013-12-09 14:20:15 +04:00
return - ENXIO ;
2012-02-01 14:14:09 +04:00
}
2013-12-09 14:20:15 +04:00
clk = devm_clk_get ( dev , " spi " ) ;
2012-02-01 14:14:09 +04:00
if ( IS_ERR ( clk ) ) {
dev_err ( dev , " no clock for device \n " ) ;
2013-12-09 14:20:15 +04:00
return PTR_ERR ( clk ) ;
2012-02-01 14:14:09 +04:00
}
master = spi_alloc_master ( dev , sizeof ( * bs ) ) ;
if ( ! master ) {
dev_err ( dev , " out of memory \n " ) ;
2013-12-09 14:20:15 +04:00
return - ENOMEM ;
2012-02-01 14:14:09 +04:00
}
bs = spi_master_get_devdata ( master ) ;
2014-02-09 07:06:04 +04:00
init_completion ( & bs - > done ) ;
2012-02-01 14:14:09 +04:00
platform_set_drvdata ( pdev , master ) ;
bs - > pdev = pdev ;
spi/spi-{bcm63xx.c,bfin-v3.c}: simplify use of devm_ioremap_resource
Remove unneeded error handling on the result of a call to
platform_get_resource when the value is passed to devm_ioremap_resource.
Move the call to platform_get_resource adjacent to the call to
devm_ioremap_resource to make the connection between them more clear.
A simplified version of the semantic patch that makes this change is as
follows: (http://coccinelle.lip6.fr/)
// <smpl>
@@
expression pdev,res,n,e,e1;
expression ret != 0;
identifier l;
@@
- res = platform_get_resource(pdev, IORESOURCE_MEM, n);
... when != res
- if (res == NULL) { ... \(goto l;\|return ret;\) }
... when != res
+ res = platform_get_resource(pdev, IORESOURCE_MEM, n);
e = devm_ioremap_resource(e1, res);
// </smpl>
Signed-off-by: Julia Lawall <Julia.Lawall@lip6.fr>
Signed-off-by: Mark Brown <broonie@linaro.org>
2013-08-14 13:11:09 +04:00
r = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-03-12 03:13:47 +04:00
bs - > regs = devm_ioremap_resource ( & pdev - > dev , r ) ;
if ( IS_ERR ( bs - > regs ) ) {
ret = PTR_ERR ( bs - > regs ) ;
2012-02-01 14:14:09 +04:00
goto out_err ;
}
bs - > irq = irq ;
bs - > clk = clk ;
bs - > fifo_size = pdata - > fifo_size ;
ret = devm_request_irq ( & pdev - > dev , irq , bcm63xx_spi_interrupt , 0 ,
pdev - > name , master ) ;
if ( ret ) {
dev_err ( dev , " unable to request irq \n " ) ;
goto out_err ;
}
master - > bus_num = pdata - > bus_num ;
master - > num_chipselect = pdata - > num_chipselect ;
2012-04-20 17:37:33 +04:00
master - > transfer_one_message = bcm63xx_spi_transfer_one ;
2012-04-20 17:37:35 +04:00
master - > mode_bits = MODEBITS ;
2013-05-22 06:36:35 +04:00
master - > bits_per_word_mask = SPI_BPW_MASK ( 8 ) ;
2013-07-28 18:34:06 +04:00
master - > auto_runtime_pm = true ;
2012-06-18 14:07:51 +04:00
bs - > msg_type_shift = pdata - > msg_type_shift ;
bs - > msg_ctl_width = pdata - > msg_ctl_width ;
2012-02-01 14:14:09 +04:00
bs - > tx_io = ( u8 * ) ( bs - > regs + bcm63xx_spireg ( SPI_MSG_DATA ) ) ;
bs - > rx_io = ( const u8 * ) ( bs - > regs + bcm63xx_spireg ( SPI_RX_DATA ) ) ;
2012-06-18 14:07:51 +04:00
switch ( bs - > msg_ctl_width ) {
case 8 :
case 16 :
break ;
default :
dev_err ( dev , " unsupported MSG_CTL width: %d \n " ,
bs - > msg_ctl_width ) ;
2013-03-12 03:13:37 +04:00
goto out_err ;
2012-06-18 14:07:51 +04:00
}
2012-02-01 14:14:09 +04:00
/* Initialize hardware */
2013-12-18 00:42:09 +04:00
ret = clk_prepare_enable ( bs - > clk ) ;
if ( ret )
goto out_err ;
2012-02-01 14:14:09 +04:00
bcm_spi_writeb ( bs , SPI_INTR_CLEAR_ALL , SPI_INT_STATUS ) ;
/* register and we are done */
2013-09-24 08:24:57 +04:00
ret = devm_spi_register_master ( dev , master ) ;
2012-02-01 14:14:09 +04:00
if ( ret ) {
dev_err ( dev , " spi register failed \n " ) ;
goto out_clk_disable ;
}
2012-10-03 13:56:53 +04:00
dev_info ( dev , " at 0x%08x (irq %d, FIFOs size %d) \n " ,
r - > start , irq , bs - > fifo_size ) ;
2012-02-01 14:14:09 +04:00
return 0 ;
out_clk_disable :
2013-03-12 03:13:38 +04:00
clk_disable_unprepare ( clk ) ;
2012-02-01 14:14:09 +04:00
out_err :
spi_master_put ( master ) ;
return ret ;
}
2012-12-07 20:57:14 +04:00
static int bcm63xx_spi_remove ( struct platform_device * pdev )
2012-02-01 14:14:09 +04:00
{
2013-11-15 11:50:59 +04:00
struct spi_master * master = platform_get_drvdata ( pdev ) ;
2012-02-01 14:14:09 +04:00
struct bcm63xx_spi * bs = spi_master_get_devdata ( master ) ;
/* reset spi block */
bcm_spi_writeb ( bs , 0 , SPI_INT_MASK ) ;
/* HW shutdown */
2013-03-12 03:13:38 +04:00
clk_disable_unprepare ( bs - > clk ) ;
2012-02-01 14:14:09 +04:00
return 0 ;
}
2013-12-18 00:42:10 +04:00
# ifdef CONFIG_PM_SLEEP
2012-02-01 14:14:09 +04:00
static int bcm63xx_spi_suspend ( struct device * dev )
{
2013-08-09 11:35:16 +04:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2012-02-01 14:14:09 +04:00
struct bcm63xx_spi * bs = spi_master_get_devdata ( master ) ;
2012-10-03 13:56:54 +04:00
spi_master_suspend ( master ) ;
2013-03-12 03:13:38 +04:00
clk_disable_unprepare ( bs - > clk ) ;
2012-02-01 14:14:09 +04:00
return 0 ;
}
static int bcm63xx_spi_resume ( struct device * dev )
{
2013-08-09 11:35:16 +04:00
struct spi_master * master = dev_get_drvdata ( dev ) ;
2012-02-01 14:14:09 +04:00
struct bcm63xx_spi * bs = spi_master_get_devdata ( master ) ;
2013-12-18 00:42:09 +04:00
int ret ;
2012-02-01 14:14:09 +04:00
2013-12-18 00:42:09 +04:00
ret = clk_prepare_enable ( bs - > clk ) ;
if ( ret )
return ret ;
2012-02-01 14:14:09 +04:00
2012-10-03 13:56:54 +04:00
spi_master_resume ( master ) ;
2012-02-01 14:14:09 +04:00
return 0 ;
}
2013-12-18 00:42:10 +04:00
# endif
2012-02-01 14:14:09 +04:00
static const struct dev_pm_ops bcm63xx_spi_pm_ops = {
2013-12-18 00:42:10 +04:00
SET_SYSTEM_SLEEP_PM_OPS ( bcm63xx_spi_suspend , bcm63xx_spi_resume )
2012-02-01 14:14:09 +04:00
} ;
static struct platform_driver bcm63xx_spi_driver = {
. driver = {
. name = " bcm63xx-spi " ,
2013-12-18 00:42:10 +04:00
. pm = & bcm63xx_spi_pm_ops ,
2012-02-01 14:14:09 +04:00
} ,
. probe = bcm63xx_spi_probe ,
2012-12-07 20:57:14 +04:00
. remove = bcm63xx_spi_remove ,
2012-02-01 14:14:09 +04:00
} ;
module_platform_driver ( bcm63xx_spi_driver ) ;
MODULE_ALIAS ( " platform:bcm63xx_spi " ) ;
MODULE_AUTHOR ( " Florian Fainelli <florian@openwrt.org> " ) ;
MODULE_AUTHOR ( " Tanguy Bouzeloc <tanguy.bouzeloc@efixo.com> " ) ;
MODULE_DESCRIPTION ( " Broadcom BCM63xx SPI Controller driver " ) ;
MODULE_LICENSE ( " GPL " ) ;