2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2011-01-04 21:28:22 +01:00
/*
* SPI controller driver for the Atheros AR71XX / AR724X / AR913X SoCs
*
* Copyright ( C ) 2009 - 2011 Gabor Juhos < juhosg @ openwrt . org >
*
* This driver has been based on the spi - gpio . c :
* Copyright ( C ) 2006 , 2008 David Brownell
*/
# include <linux/kernel.h>
2011-11-16 20:01:43 +01:00
# include <linux/module.h>
2011-01-04 21:28:22 +01:00
# include <linux/delay.h>
# include <linux/spinlock.h>
# include <linux/platform_device.h>
# include <linux/io.h>
# include <linux/spi/spi.h>
2022-01-29 01:04:53 -03:00
# include <linux/spi/spi-mem.h>
2011-01-04 21:28:22 +01:00
# include <linux/spi/spi_bitbang.h>
# include <linux/bitops.h>
2012-12-27 10:42:24 +01:00
# include <linux/clk.h>
# include <linux/err.h>
2011-01-04 21:28:22 +01:00
# define DRV_NAME "ath79-spi"
2012-12-27 10:42:24 +01:00
# define ATH79_SPI_RRW_DELAY_FACTOR 12000
# define MHZ (1000 * 1000)
2019-01-16 19:55:46 +01:00
# define AR71XX_SPI_REG_FS 0x00 /* Function Select */
# define AR71XX_SPI_REG_CTRL 0x04 /* SPI Control */
# define AR71XX_SPI_REG_IOC 0x08 /* SPI I/O Control */
# define AR71XX_SPI_REG_RDS 0x0c /* Read Data Shift */
# define AR71XX_SPI_FS_GPIO BIT(0) /* Enable GPIO mode */
# define AR71XX_SPI_IOC_DO BIT(0) /* Data Out pin */
# define AR71XX_SPI_IOC_CLK BIT(8) /* CLK pin */
# define AR71XX_SPI_IOC_CS(n) BIT(16 + (n))
2011-01-04 21:28:22 +01:00
struct ath79_spi {
struct spi_bitbang bitbang ;
u32 ioc_base ;
u32 reg_ctrl ;
void __iomem * base ;
2012-12-27 10:42:24 +01:00
struct clk * clk ;
2017-06-27 22:01:11 +05:30
unsigned int rrw_delay ;
2011-01-04 21:28:22 +01:00
} ;
2017-06-27 22:01:11 +05:30
static inline u32 ath79_spi_rr ( struct ath79_spi * sp , unsigned int reg )
2011-01-04 21:28:22 +01:00
{
return ioread32 ( sp - > base + reg ) ;
}
2017-06-27 22:01:11 +05:30
static inline void ath79_spi_wr ( struct ath79_spi * sp , unsigned int reg , u32 val )
2011-01-04 21:28:22 +01:00
{
iowrite32 ( val , sp - > base + reg ) ;
}
static inline struct ath79_spi * ath79_spidev_to_sp ( struct spi_device * spi )
{
2022-12-30 17:28:04 +08:00
return spi_controller_get_devdata ( spi - > controller ) ;
2011-01-04 21:28:22 +01:00
}
2017-06-27 22:01:11 +05:30
static inline void ath79_spi_delay ( struct ath79_spi * sp , unsigned int nsecs )
2012-12-27 10:42:24 +01:00
{
if ( nsecs > sp - > rrw_delay )
ndelay ( nsecs - sp - > rrw_delay ) ;
}
2011-01-04 21:28:22 +01:00
static void ath79_spi_chipselect ( struct spi_device * spi , int is_active )
{
struct ath79_spi * sp = ath79_spidev_to_sp ( spi ) ;
int cs_high = ( spi - > mode & SPI_CS_HIGH ) ? is_active : ! is_active ;
2023-03-10 23:02:03 +05:30
u32 cs_bit = AR71XX_SPI_IOC_CS ( spi_get_chipselect ( spi , 0 ) ) ;
2011-01-04 21:28:22 +01:00
2019-01-16 19:55:45 +01:00
if ( cs_high )
sp - > ioc_base | = cs_bit ;
else
sp - > ioc_base & = ~ cs_bit ;
2011-01-04 21:28:22 +01:00
2019-01-16 19:55:45 +01:00
ath79_spi_wr ( sp , AR71XX_SPI_REG_IOC , sp - > ioc_base ) ;
2011-01-04 21:28:22 +01:00
}
2012-12-27 10:42:28 +01:00
static void ath79_spi_enable ( struct ath79_spi * sp )
2011-01-04 21:28:22 +01:00
{
/* enable GPIO mode */
ath79_spi_wr ( sp , AR71XX_SPI_REG_FS , AR71XX_SPI_FS_GPIO ) ;
/* save CTRL register */
sp - > reg_ctrl = ath79_spi_rr ( sp , AR71XX_SPI_REG_CTRL ) ;
sp - > ioc_base = ath79_spi_rr ( sp , AR71XX_SPI_REG_IOC ) ;
2019-01-16 19:55:45 +01:00
/* clear clk and mosi in the base state */
sp - > ioc_base & = ~ ( AR71XX_SPI_IOC_DO | AR71XX_SPI_IOC_CLK ) ;
2011-01-04 21:28:22 +01:00
/* TODO: setup speed? */
ath79_spi_wr ( sp , AR71XX_SPI_REG_CTRL , 0x43 ) ;
2012-12-27 10:42:28 +01:00
}
static void ath79_spi_disable ( struct ath79_spi * sp )
{
/* restore CTRL register */
ath79_spi_wr ( sp , AR71XX_SPI_REG_CTRL , sp - > reg_ctrl ) ;
/* disable GPIO mode */
ath79_spi_wr ( sp , AR71XX_SPI_REG_FS , 0 ) ;
}
2017-06-27 22:01:11 +05:30
static u32 ath79_spi_txrx_mode0 ( struct spi_device * spi , unsigned int nsecs ,
2018-07-28 10:19:13 +02:00
u32 word , u8 bits , unsigned flags )
2011-01-04 21:28:22 +01:00
{
struct ath79_spi * sp = ath79_spidev_to_sp ( spi ) ;
u32 ioc = sp - > ioc_base ;
/* clock starts at inactive polarity */
for ( word < < = ( 32 - bits ) ; likely ( bits ) ; bits - - ) {
u32 out ;
if ( word & ( 1 < < 31 ) )
out = ioc | AR71XX_SPI_IOC_DO ;
else
out = ioc & ~ AR71XX_SPI_IOC_DO ;
2022-12-30 17:28:04 +08:00
/* setup MSB (to target) on trailing edge */
2011-01-04 21:28:22 +01:00
ath79_spi_wr ( sp , AR71XX_SPI_REG_IOC , out ) ;
2012-12-27 10:42:24 +01:00
ath79_spi_delay ( sp , nsecs ) ;
2011-01-04 21:28:22 +01:00
ath79_spi_wr ( sp , AR71XX_SPI_REG_IOC , out | AR71XX_SPI_IOC_CLK ) ;
2012-12-27 10:42:24 +01:00
ath79_spi_delay ( sp , nsecs ) ;
2012-12-27 10:42:25 +01:00
if ( bits = = 1 )
ath79_spi_wr ( sp , AR71XX_SPI_REG_IOC , out ) ;
2011-01-04 21:28:22 +01:00
word < < = 1 ;
}
return ath79_spi_rr ( sp , AR71XX_SPI_REG_RDS ) ;
}
2022-01-29 01:04:53 -03:00
static int ath79_exec_mem_op ( struct spi_mem * mem ,
const struct spi_mem_op * op )
{
struct ath79_spi * sp = ath79_spidev_to_sp ( mem - > spi ) ;
/* Ensures that reading is performed on device connected to hardware cs0 */
2023-03-10 23:02:03 +05:30
if ( spi_get_chipselect ( mem - > spi , 0 ) | | spi_get_csgpiod ( mem - > spi , 0 ) )
2022-01-29 01:04:53 -03:00
return - ENOTSUPP ;
/* Only use for fast-read op. */
if ( op - > cmd . opcode ! = 0x0b | | op - > data . dir ! = SPI_MEM_DATA_IN | |
op - > addr . nbytes ! = 3 | | op - > dummy . nbytes ! = 1 )
return - ENOTSUPP ;
/* disable GPIO mode */
ath79_spi_wr ( sp , AR71XX_SPI_REG_FS , 0 ) ;
memcpy_fromio ( op - > data . buf . in , sp - > base + op - > addr . val , op - > data . nbytes ) ;
/* enable GPIO mode */
ath79_spi_wr ( sp , AR71XX_SPI_REG_FS , AR71XX_SPI_FS_GPIO ) ;
/* restore IOC register */
ath79_spi_wr ( sp , AR71XX_SPI_REG_IOC , sp - > ioc_base ) ;
return 0 ;
}
static const struct spi_controller_mem_ops ath79_mem_ops = {
. exec_op = ath79_exec_mem_op ,
} ;
2012-12-07 16:57:14 +00:00
static int ath79_spi_probe ( struct platform_device * pdev )
2011-01-04 21:28:22 +01:00
{
2022-12-30 17:28:04 +08:00
struct spi_controller * host ;
2011-01-04 21:28:22 +01:00
struct ath79_spi * sp ;
2012-12-27 10:42:24 +01:00
unsigned long rate ;
2011-01-04 21:28:22 +01:00
int ret ;
2022-12-30 17:28:04 +08:00
host = spi_alloc_host ( & pdev - > dev , sizeof ( * sp ) ) ;
if ( host = = NULL ) {
dev_err ( & pdev - > dev , " failed to allocate spi host \n " ) ;
2011-01-04 21:28:22 +01:00
return - ENOMEM ;
}
2022-12-30 17:28:04 +08:00
sp = spi_controller_get_devdata ( host ) ;
host - > dev . of_node = pdev - > dev . of_node ;
2011-01-04 21:28:22 +01:00
platform_set_drvdata ( pdev , sp ) ;
2022-12-30 17:28:04 +08:00
host - > use_gpio_descriptors = true ;
host - > bits_per_word_mask = SPI_BPW_RANGE_MASK ( 1 , 32 ) ;
2023-07-10 18:49:30 +03:00
host - > flags = SPI_CONTROLLER_GPIO_SS ;
2022-12-30 17:28:04 +08:00
host - > num_chipselect = 3 ;
host - > mem_ops = & ath79_mem_ops ;
2011-01-04 21:28:22 +01:00
2022-12-30 17:28:04 +08:00
sp - > bitbang . master = host ;
2011-01-04 21:28:22 +01:00
sp - > bitbang . chipselect = ath79_spi_chipselect ;
sp - > bitbang . txrx_word [ SPI_MODE_0 ] = ath79_spi_txrx_mode0 ;
sp - > bitbang . flags = SPI_CS_HIGH ;
2019-09-04 21:58:45 +08:00
sp - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2015-09-27 18:47:35 +02:00
if ( IS_ERR ( sp - > base ) ) {
ret = PTR_ERR ( sp - > base ) ;
2022-12-30 17:28:04 +08:00
goto err_put_host ;
2011-01-04 21:28:22 +01:00
}
2013-12-09 19:14:58 +09:00
sp - > clk = devm_clk_get ( & pdev - > dev , " ahb " ) ;
2012-12-27 10:42:24 +01:00
if ( IS_ERR ( sp - > clk ) ) {
ret = PTR_ERR ( sp - > clk ) ;
2022-12-30 17:28:04 +08:00
goto err_put_host ;
2012-12-27 10:42:24 +01:00
}
2015-04-24 16:19:23 +02:00
ret = clk_prepare_enable ( sp - > clk ) ;
2012-12-27 10:42:24 +01:00
if ( ret )
2022-12-30 17:28:04 +08:00
goto err_put_host ;
2012-12-27 10:42:24 +01:00
rate = DIV_ROUND_UP ( clk_get_rate ( sp - > clk ) , MHZ ) ;
if ( ! rate ) {
ret = - EINVAL ;
goto err_clk_disable ;
}
sp - > rrw_delay = ATH79_SPI_RRW_DELAY_FACTOR / rate ;
dev_dbg ( & pdev - > dev , " register read/write delay is %u nsecs \n " ,
sp - > rrw_delay ) ;
2012-12-27 10:42:28 +01:00
ath79_spi_enable ( sp ) ;
2011-01-04 21:28:22 +01:00
ret = spi_bitbang_start ( & sp - > bitbang ) ;
if ( ret )
2012-12-27 10:42:28 +01:00
goto err_disable ;
2011-01-04 21:28:22 +01:00
return 0 ;
2012-12-27 10:42:28 +01:00
err_disable :
ath79_spi_disable ( sp ) ;
2012-12-27 10:42:24 +01:00
err_clk_disable :
2015-04-24 16:19:23 +02:00
clk_disable_unprepare ( sp - > clk ) ;
2022-12-30 17:28:04 +08:00
err_put_host :
spi_controller_put ( host ) ;
2011-01-04 21:28:22 +01:00
return ret ;
}
2023-03-03 18:19:19 +01:00
static void ath79_spi_remove ( struct platform_device * pdev )
2011-01-04 21:28:22 +01:00
{
struct ath79_spi * sp = platform_get_drvdata ( pdev ) ;
spi_bitbang_stop ( & sp - > bitbang ) ;
2012-12-27 10:42:28 +01:00
ath79_spi_disable ( sp ) ;
2015-04-24 16:19:23 +02:00
clk_disable_unprepare ( sp - > clk ) ;
2022-12-30 17:28:04 +08:00
spi_controller_put ( sp - > bitbang . master ) ;
2011-01-04 21:28:22 +01:00
}
2013-02-05 20:57:55 +01:00
static void ath79_spi_shutdown ( struct platform_device * pdev )
{
ath79_spi_remove ( pdev ) ;
}
2015-04-24 16:19:22 +02:00
static const struct of_device_id ath79_spi_of_match [ ] = {
{ . compatible = " qca,ar7100-spi " , } ,
{ } ,
} ;
2016-11-23 13:37:11 -03:00
MODULE_DEVICE_TABLE ( of , ath79_spi_of_match ) ;
2015-04-24 16:19:22 +02:00
2011-01-04 21:28:22 +01:00
static struct platform_driver ath79_spi_driver = {
. probe = ath79_spi_probe ,
2023-03-03 18:19:19 +01:00
. remove_new = ath79_spi_remove ,
2013-02-05 20:57:55 +01:00
. shutdown = ath79_spi_shutdown ,
2011-01-04 21:28:22 +01:00
. driver = {
. name = DRV_NAME ,
2015-04-24 16:19:22 +02:00
. of_match_table = ath79_spi_of_match ,
2011-01-04 21:28:22 +01:00
} ,
} ;
2011-10-05 11:29:49 -06:00
module_platform_driver ( ath79_spi_driver ) ;
2011-01-04 21:28:22 +01:00
MODULE_DESCRIPTION ( " SPI controller driver for Atheros AR71XX/AR724X/AR913X " ) ;
MODULE_AUTHOR ( " Gabor Juhos <juhosg@openwrt.org> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;