2021-04-16 09:57:19 -07:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* 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>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/spi/altera.h>
# include <linux/spi/spi.h>
# include <linux/io.h>
# include <linux/of.h>
# define DRV_NAME "spi_altera"
enum altera_spi_type {
ALTERA_SPI_TYPE_UNKNOWN ,
ALTERA_SPI_TYPE_SUBDEV ,
} ;
static const struct regmap_config spi_altera_config = {
. reg_bits = 32 ,
. reg_stride = 4 ,
. val_bits = 32 ,
. fast_io = true ,
} ;
static int altera_spi_probe ( struct platform_device * pdev )
{
const struct platform_device_id * platid = platform_get_device_id ( pdev ) ;
struct altera_spi_platform_data * pdata = dev_get_platdata ( & pdev - > dev ) ;
enum altera_spi_type type = ALTERA_SPI_TYPE_UNKNOWN ;
struct altera_spi * hw ;
2022-12-29 18:38:37 +08:00
struct spi_controller * host ;
2021-04-16 09:57:19 -07:00
int err = - ENODEV ;
u16 i ;
2022-12-29 18:38:37 +08:00
host = spi_alloc_host ( & pdev - > dev , sizeof ( struct altera_spi ) ) ;
if ( ! host )
2021-04-16 09:57:19 -07:00
return err ;
2022-12-29 18:38:37 +08:00
/* setup the host state. */
host - > bus_num = - 1 ;
2021-04-16 09:57:19 -07:00
if ( pdata ) {
if ( pdata - > num_chipselect > ALTERA_SPI_MAX_CS ) {
dev_err ( & pdev - > dev ,
" Invalid number of chipselect: %u \n " ,
pdata - > num_chipselect ) ;
err = - EINVAL ;
goto exit ;
}
2022-12-29 18:38:37 +08:00
host - > num_chipselect = pdata - > num_chipselect ;
host - > mode_bits = pdata - > mode_bits ;
host - > bits_per_word_mask = pdata - > bits_per_word_mask ;
2021-04-16 09:57:19 -07:00
} else {
2022-12-29 18:38:37 +08:00
host - > num_chipselect = 16 ;
host - > mode_bits = SPI_CS_HIGH ;
host - > bits_per_word_mask = SPI_BPW_RANGE_MASK ( 1 , 16 ) ;
2021-04-16 09:57:19 -07:00
}
2022-12-29 18:38:37 +08:00
host - > dev . of_node = pdev - > dev . of_node ;
2021-04-16 09:57:19 -07:00
2022-12-29 18:38:37 +08:00
hw = spi_controller_get_devdata ( host ) ;
2021-04-16 09:57:19 -07:00
hw - > dev = & pdev - > dev ;
if ( platid )
type = platid - > driver_data ;
/* find and map our resources */
if ( type = = ALTERA_SPI_TYPE_SUBDEV ) {
struct resource * regoff ;
hw - > regmap = dev_get_regmap ( pdev - > dev . parent , NULL ) ;
if ( ! hw - > regmap ) {
dev_err ( & pdev - > dev , " get regmap failed \n " ) ;
goto exit ;
}
regoff = platform_get_resource ( pdev , IORESOURCE_REG , 0 ) ;
if ( regoff )
hw - > regoff = regoff - > start ;
} else {
void __iomem * res ;
res = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( res ) ) {
err = PTR_ERR ( res ) ;
goto exit ;
}
hw - > regmap = devm_regmap_init_mmio ( & pdev - > dev , res ,
& spi_altera_config ) ;
if ( IS_ERR ( hw - > regmap ) ) {
dev_err ( & pdev - > dev , " regmap mmio init failed \n " ) ;
err = PTR_ERR ( hw - > regmap ) ;
goto exit ;
}
}
2022-12-29 18:38:37 +08:00
altera_spi_init_host ( host ) ;
2021-04-16 09:57:19 -07:00
/* 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 ,
2022-12-29 18:38:37 +08:00
pdev - > name , host ) ;
2021-04-16 09:57:19 -07:00
if ( err )
goto exit ;
}
2022-12-29 18:38:37 +08:00
err = devm_spi_register_controller ( & pdev - > dev , host ) ;
2021-04-16 09:57:19 -07:00
if ( err )
goto exit ;
if ( pdata ) {
for ( i = 0 ; i < pdata - > num_devices ; i + + ) {
2022-12-29 18:38:37 +08:00
if ( ! spi_new_device ( host , pdata - > devices + i ) )
2021-04-16 09:57:19 -07:00
dev_warn ( & pdev - > dev ,
" unable to create SPI device: %s \n " ,
pdata - > devices [ i ] . modalias ) ;
}
}
dev_info ( & pdev - > dev , " regoff %u, irq %d \n " , hw - > regoff , hw - > irq ) ;
return 0 ;
exit :
2022-12-29 18:38:37 +08:00
spi_controller_put ( host ) ;
2021-04-16 09:57:19 -07:00
return err ;
}
# ifdef CONFIG_OF
static const struct of_device_id altera_spi_match [ ] = {
{ . compatible = " ALTR,spi-1.0 " , } ,
{ . compatible = " altr,spi-1.0 " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , altera_spi_match ) ;
# endif /* CONFIG_OF */
static const struct platform_device_id altera_spi_ids [ ] = {
{ DRV_NAME , ALTERA_SPI_TYPE_UNKNOWN } ,
{ " subdev_spi_altera " , ALTERA_SPI_TYPE_SUBDEV } ,
{ }
} ;
MODULE_DEVICE_TABLE ( platform , altera_spi_ids ) ;
static struct platform_driver altera_spi_driver = {
. probe = altera_spi_probe ,
. driver = {
. name = DRV_NAME ,
. pm = NULL ,
. of_match_table = of_match_ptr ( altera_spi_match ) ,
} ,
. id_table = altera_spi_ids ,
} ;
module_platform_driver ( altera_spi_driver ) ;
MODULE_DESCRIPTION ( " Altera SPI driver " ) ;
MODULE_AUTHOR ( " Thomas Chou <thomas@wytron.com.tw> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;