2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2011-02-14 05:10:43 +03:00
/*
* Altera SPI driver
*
* Copyright ( C ) 2008 Thomas Chou < thomas @ wytron . com . tw >
*
* Based on spi_s3c24xx . c , which is :
* Copyright ( c ) 2006 Ben Dooks
* Copyright ( c ) 2006 Simtec Electronics
* Ben Dooks < ben @ simtec . co . uk >
*/
# include <linux/interrupt.h>
# include <linux/errno.h>
2011-07-03 23:44:29 +04:00
# include <linux/module.h>
2011-02-14 05:10:43 +03:00
# include <linux/platform_device.h>
# include <linux/spi/spi.h>
# include <linux/io.h>
# include <linux/of.h>
# define DRV_NAME "spi_altera"
# define ALTERA_SPI_RXDATA 0
# define ALTERA_SPI_TXDATA 4
# define ALTERA_SPI_STATUS 8
# define ALTERA_SPI_CONTROL 12
# define ALTERA_SPI_SLAVE_SEL 20
# define ALTERA_SPI_STATUS_ROE_MSK 0x8
# define ALTERA_SPI_STATUS_TOE_MSK 0x10
# define ALTERA_SPI_STATUS_TMT_MSK 0x20
# define ALTERA_SPI_STATUS_TRDY_MSK 0x40
# define ALTERA_SPI_STATUS_RRDY_MSK 0x80
# define ALTERA_SPI_STATUS_E_MSK 0x100
# define ALTERA_SPI_CONTROL_IROE_MSK 0x8
# define ALTERA_SPI_CONTROL_ITOE_MSK 0x10
# define ALTERA_SPI_CONTROL_ITRDY_MSK 0x40
# define ALTERA_SPI_CONTROL_IRRDY_MSK 0x80
# define ALTERA_SPI_CONTROL_IE_MSK 0x100
# define ALTERA_SPI_CONTROL_SSO_MSK 0x400
struct altera_spi {
void __iomem * base ;
int irq ;
int len ;
int count ;
int bytes_per_word ;
unsigned long imr ;
/* data buffers */
const unsigned char * tx ;
unsigned char * rx ;
} ;
static inline struct altera_spi * altera_spi_to_hw ( struct spi_device * sdev )
{
return spi_master_get_devdata ( sdev - > master ) ;
}
2017-08-16 12:33:11 +03:00
static void altera_spi_set_cs ( struct spi_device * spi , bool is_high )
2011-02-14 05:10:43 +03:00
{
struct altera_spi * hw = altera_spi_to_hw ( spi ) ;
2017-08-16 12:33:11 +03:00
if ( is_high ) {
hw - > imr & = ~ ALTERA_SPI_CONTROL_SSO_MSK ;
writel ( hw - > imr , hw - > base + ALTERA_SPI_CONTROL ) ;
writel ( 0 , hw - > base + ALTERA_SPI_SLAVE_SEL ) ;
2011-02-14 05:10:43 +03:00
} else {
2017-08-16 12:33:11 +03:00
writel ( BIT ( spi - > chip_select ) , hw - > base + ALTERA_SPI_SLAVE_SEL ) ;
hw - > imr | = ALTERA_SPI_CONTROL_SSO_MSK ;
writel ( hw - > imr , hw - > base + ALTERA_SPI_CONTROL ) ;
2011-02-14 05:10:43 +03:00
}
}
2017-08-16 12:33:12 +03:00
static void altera_spi_tx_word ( struct altera_spi * hw )
2011-02-14 05:10:43 +03:00
{
2017-08-16 12:33:12 +03:00
unsigned int txd = 0 ;
2011-02-14 05:10:43 +03:00
if ( hw - > tx ) {
switch ( hw - > bytes_per_word ) {
case 1 :
2017-08-16 12:33:12 +03:00
txd = hw - > tx [ hw - > count ] ;
break ;
2011-02-14 05:10:43 +03:00
case 2 :
2017-08-16 12:33:12 +03:00
txd = ( hw - > tx [ hw - > count * 2 ]
| ( hw - > tx [ hw - > count * 2 + 1 ] < < 8 ) ) ;
break ;
2011-02-14 05:10:43 +03:00
}
}
2017-08-16 12:33:12 +03:00
writel ( txd , hw - > base + ALTERA_SPI_TXDATA ) ;
}
static void altera_spi_rx_word ( struct altera_spi * hw )
{
unsigned int rxd ;
rxd = readl ( hw - > base + ALTERA_SPI_RXDATA ) ;
if ( hw - > rx ) {
switch ( hw - > bytes_per_word ) {
case 1 :
hw - > rx [ hw - > count ] = rxd ;
break ;
case 2 :
hw - > rx [ hw - > count * 2 ] = rxd ;
hw - > rx [ hw - > count * 2 + 1 ] = rxd > > 8 ;
break ;
}
}
hw - > count + + ;
2011-02-14 05:10:43 +03:00
}
2017-08-16 12:33:11 +03:00
static int altera_spi_txrx ( struct spi_master * master ,
struct spi_device * spi , struct spi_transfer * t )
2011-02-14 05:10:43 +03:00
{
2017-08-16 12:33:11 +03:00
struct altera_spi * hw = spi_master_get_devdata ( master ) ;
2011-02-14 05:10:43 +03:00
hw - > tx = t - > tx_buf ;
hw - > rx = t - > rx_buf ;
hw - > count = 0 ;
2013-08-29 19:41:20 +04:00
hw - > bytes_per_word = DIV_ROUND_UP ( t - > bits_per_word , 8 ) ;
2011-02-14 05:10:43 +03:00
hw - > len = t - > len / hw - > bytes_per_word ;
if ( hw - > irq > = 0 ) {
/* enable receive interrupt */
hw - > imr | = ALTERA_SPI_CONTROL_IRRDY_MSK ;
writel ( hw - > imr , hw - > base + ALTERA_SPI_CONTROL ) ;
/* send the first byte */
2017-08-16 12:33:12 +03:00
altera_spi_tx_word ( hw ) ;
2011-02-14 05:10:43 +03:00
} else {
2013-08-15 10:18:46 +04:00
while ( hw - > count < hw - > len ) {
2017-08-16 12:33:12 +03:00
altera_spi_tx_word ( hw ) ;
2013-08-15 10:18:46 +04:00
2011-02-14 05:10:43 +03:00
while ( ! ( readl ( hw - > base + ALTERA_SPI_STATUS ) &
ALTERA_SPI_STATUS_RRDY_MSK ) )
cpu_relax ( ) ;
2017-08-16 12:33:12 +03:00
altera_spi_rx_word ( hw ) ;
2011-02-14 05:10:43 +03:00
}
2017-08-16 12:33:11 +03:00
spi_finalize_current_transfer ( master ) ;
2011-02-14 05:10:43 +03:00
}
2017-08-16 12:33:11 +03:00
return t - > len ;
2011-02-14 05:10:43 +03:00
}
static irqreturn_t altera_spi_irq ( int irq , void * dev )
{
2017-08-16 12:33:11 +03:00
struct spi_master * master = dev ;
struct altera_spi * hw = spi_master_get_devdata ( master ) ;
2011-02-14 05:10:43 +03:00
2017-08-16 12:33:12 +03:00
altera_spi_rx_word ( hw ) ;
2011-02-14 05:10:43 +03:00
2017-08-16 12:33:11 +03:00
if ( hw - > count < hw - > len ) {
2017-08-16 12:33:12 +03:00
altera_spi_tx_word ( hw ) ;
2017-08-16 12:33:11 +03:00
} else {
/* disable receive interrupt */
hw - > imr & = ~ ALTERA_SPI_CONTROL_IRRDY_MSK ;
writel ( hw - > imr , hw - > base + ALTERA_SPI_CONTROL ) ;
spi_finalize_current_transfer ( master ) ;
}
2011-02-14 05:10:43 +03:00
return IRQ_HANDLED ;
}
2012-12-07 20:57:14 +04:00
static int altera_spi_probe ( struct platform_device * pdev )
2011-02-14 05:10:43 +03:00
{
struct altera_spi * hw ;
struct spi_master * master ;
struct resource * res ;
int err = - ENODEV ;
master = spi_alloc_master ( & pdev - > dev , sizeof ( struct altera_spi ) ) ;
if ( ! master )
return err ;
/* setup the master state. */
master - > bus_num = pdev - > id ;
master - > num_chipselect = 16 ;
master - > mode_bits = SPI_CS_HIGH ;
2014-03-05 09:37:00 +04:00
master - > bits_per_word_mask = SPI_BPW_RANGE_MASK ( 1 , 16 ) ;
2014-03-21 07:21:58 +04:00
master - > dev . of_node = pdev - > dev . of_node ;
2017-08-16 12:33:11 +03:00
master - > transfer_one = altera_spi_txrx ;
master - > set_cs = altera_spi_set_cs ;
2011-02-14 05:10:43 +03:00
hw = spi_master_get_devdata ( master ) ;
/* find and map our resources */
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-08-24 21:13:15 +04:00
hw - > base = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( hw - > base ) ) {
err = PTR_ERR ( hw - > base ) ;
goto exit ;
}
2011-02-14 05:10:43 +03:00
/* program defaults into the registers */
hw - > imr = 0 ; /* disable spi interrupts */
writel ( hw - > imr , hw - > base + ALTERA_SPI_CONTROL ) ;
writel ( 0 , hw - > base + ALTERA_SPI_STATUS ) ; /* clear status reg */
if ( readl ( hw - > base + ALTERA_SPI_STATUS ) & ALTERA_SPI_STATUS_RRDY_MSK )
readl ( hw - > base + ALTERA_SPI_RXDATA ) ; /* flush rxdata */
/* irq is optional */
hw - > irq = platform_get_irq ( pdev , 0 ) ;
if ( hw - > irq > = 0 ) {
err = devm_request_irq ( & pdev - > dev , hw - > irq , altera_spi_irq , 0 ,
2017-08-16 12:33:11 +03:00
pdev - > name , master ) ;
2011-02-14 05:10:43 +03:00
if ( err )
goto exit ;
}
2017-08-16 12:33:11 +03:00
err = devm_spi_register_master ( & pdev - > dev , master ) ;
2011-02-14 05:10:43 +03:00
if ( err )
goto exit ;
dev_info ( & pdev - > dev , " base %p, irq %d \n " , hw - > base , hw - > irq ) ;
return 0 ;
exit :
spi_master_put ( master ) ;
return err ;
}
# ifdef CONFIG_OF
static const struct of_device_id altera_spi_match [ ] = {
{ . compatible = " ALTR,spi-1.0 " , } ,
2013-08-15 00:25:19 +04:00
{ . compatible = " altr,spi-1.0 " , } ,
2011-02-14 05:10:43 +03:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , altera_spi_match ) ;
# endif /* CONFIG_OF */
static struct platform_driver altera_spi_driver = {
. probe = altera_spi_probe ,
. driver = {
. name = DRV_NAME ,
. pm = NULL ,
2012-08-15 11:30:28 +04:00
. of_match_table = of_match_ptr ( altera_spi_match ) ,
2011-02-14 05:10:43 +03:00
} ,
} ;
2011-10-05 21:29:49 +04:00
module_platform_driver ( altera_spi_driver ) ;
2011-02-14 05:10:43 +03:00
MODULE_DESCRIPTION ( " Altera SPI driver " ) ;
MODULE_AUTHOR ( " Thomas Chou <thomas@wytron.com.tw> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;