2010-11-24 10:17:14 +01:00
/*
* CE4100 ' s SPI device is more or less the same one as found on PXA
*
2016-07-04 12:44:27 +03:00
* Copyright ( C ) 2016 , Intel Corporation
2010-11-24 10:17:14 +01:00
*/
2016-07-04 12:44:27 +03:00
# include <linux/clk-provider.h>
# include <linux/module.h>
# include <linux/of_device.h>
2010-11-24 10:17:14 +01:00
# include <linux/pci.h>
# include <linux/platform_device.h>
# include <linux/spi/pxa2xx_spi.h>
2014-08-19 20:29:19 +03:00
# include <linux/dmaengine.h>
# include <linux/platform_data/dma-dw.h>
2014-04-18 00:26:06 +08:00
enum {
2016-07-04 12:44:27 +03:00
PORT_QUARK_X1000 ,
2014-04-18 00:26:06 +08:00
PORT_BYT ,
2016-07-04 12:44:25 +03:00
PORT_MRFLD ,
2014-08-19 20:29:21 +03:00
PORT_BSW0 ,
PORT_BSW1 ,
PORT_BSW2 ,
2016-07-04 12:44:27 +03:00
PORT_CE4100 ,
2016-02-20 20:20:22 +01:00
PORT_LPT ,
2014-04-18 00:26:06 +08:00
} ;
struct pxa_spi_info {
enum pxa_ssp_type type ;
int port_id ;
int num_chipselect ;
2014-07-25 01:10:54 +08:00
unsigned long max_clk_rate ;
2014-08-19 20:29:19 +03:00
/* DMA channel request parameters */
2016-07-04 12:44:24 +03:00
bool ( * dma_filter ) ( struct dma_chan * chan , void * param ) ;
2014-08-19 20:29:19 +03:00
void * tx_param ;
void * rx_param ;
2016-07-04 12:44:24 +03:00
int ( * setup ) ( struct pci_dev * pdev , struct pxa_spi_info * c ) ;
2014-04-18 00:26:06 +08:00
} ;
2014-08-19 20:29:19 +03:00
static struct dw_dma_slave byt_tx_param = { . dst_id = 0 } ;
static struct dw_dma_slave byt_rx_param = { . src_id = 1 } ;
2017-01-02 13:47:31 +02:00
static struct dw_dma_slave mrfld3_tx_param = { . dst_id = 15 } ;
static struct dw_dma_slave mrfld3_rx_param = { . src_id = 14 } ;
static struct dw_dma_slave mrfld5_tx_param = { . dst_id = 13 } ;
static struct dw_dma_slave mrfld5_rx_param = { . src_id = 12 } ;
static struct dw_dma_slave mrfld6_tx_param = { . dst_id = 11 } ;
static struct dw_dma_slave mrfld6_rx_param = { . src_id = 10 } ;
2014-08-19 20:29:21 +03:00
static struct dw_dma_slave bsw0_tx_param = { . dst_id = 0 } ;
static struct dw_dma_slave bsw0_rx_param = { . src_id = 1 } ;
static struct dw_dma_slave bsw1_tx_param = { . dst_id = 6 } ;
static struct dw_dma_slave bsw1_rx_param = { . src_id = 7 } ;
static struct dw_dma_slave bsw2_tx_param = { . dst_id = 8 } ;
static struct dw_dma_slave bsw2_rx_param = { . src_id = 9 } ;
2016-02-20 20:20:22 +01:00
static struct dw_dma_slave lpt_tx_param = { . dst_id = 0 } ;
static struct dw_dma_slave lpt_rx_param = { . src_id = 1 } ;
2014-08-19 20:29:19 +03:00
static bool lpss_dma_filter ( struct dma_chan * chan , void * param )
{
struct dw_dma_slave * dws = param ;
if ( dws - > dma_dev ! = chan - > device - > dev )
return false ;
chan - > private = dws ;
return true ;
}
2016-07-04 12:44:24 +03:00
static int lpss_spi_setup ( struct pci_dev * dev , struct pxa_spi_info * c )
{
struct pci_dev * dma_dev ;
c - > num_chipselect = 1 ;
c - > max_clk_rate = 50000000 ;
dma_dev = pci_get_slot ( dev - > bus , PCI_DEVFN ( PCI_SLOT ( dev - > devfn ) , 0 ) ) ;
if ( c - > tx_param ) {
struct dw_dma_slave * slave = c - > tx_param ;
slave - > dma_dev = & dma_dev - > dev ;
slave - > m_master = 0 ;
slave - > p_master = 1 ;
}
if ( c - > rx_param ) {
struct dw_dma_slave * slave = c - > rx_param ;
slave - > dma_dev = & dma_dev - > dev ;
slave - > m_master = 0 ;
slave - > p_master = 1 ;
}
c - > dma_filter = lpss_dma_filter ;
return 0 ;
}
2016-07-04 12:44:25 +03:00
static int mrfld_spi_setup ( struct pci_dev * dev , struct pxa_spi_info * c )
{
2017-01-02 13:47:31 +02:00
struct pci_dev * dma_dev = pci_get_slot ( dev - > bus , PCI_DEVFN ( 21 , 0 ) ) ;
struct dw_dma_slave * tx , * rx ;
2016-07-04 12:44:25 +03:00
switch ( PCI_FUNC ( dev - > devfn ) ) {
case 0 :
c - > port_id = 3 ;
c - > num_chipselect = 1 ;
2017-01-02 13:47:31 +02:00
c - > tx_param = & mrfld3_tx_param ;
c - > rx_param = & mrfld3_rx_param ;
2016-07-04 12:44:25 +03:00
break ;
case 1 :
c - > port_id = 5 ;
c - > num_chipselect = 4 ;
2017-01-02 13:47:31 +02:00
c - > tx_param = & mrfld5_tx_param ;
c - > rx_param = & mrfld5_rx_param ;
2016-07-04 12:44:25 +03:00
break ;
case 2 :
c - > port_id = 6 ;
c - > num_chipselect = 1 ;
2017-01-02 13:47:31 +02:00
c - > tx_param = & mrfld6_tx_param ;
c - > rx_param = & mrfld6_rx_param ;
2016-07-04 12:44:25 +03:00
break ;
default :
return - ENODEV ;
}
2017-01-02 13:47:31 +02:00
tx = c - > tx_param ;
tx - > dma_dev = & dma_dev - > dev ;
rx = c - > rx_param ;
rx - > dma_dev = & dma_dev - > dev ;
c - > dma_filter = lpss_dma_filter ;
2016-07-04 12:44:25 +03:00
return 0 ;
}
2014-04-18 00:26:06 +08:00
static struct pxa_spi_info spi_info_configs [ ] = {
[ PORT_CE4100 ] = {
. type = PXA25x_SSP ,
. port_id = - 1 ,
. num_chipselect = - 1 ,
2014-07-25 01:10:54 +08:00
. max_clk_rate = 3686400 ,
2014-04-18 00:26:06 +08:00
} ,
[ PORT_BYT ] = {
2015-06-04 16:55:10 +03:00
. type = LPSS_BYT_SSP ,
2014-04-18 00:26:06 +08:00
. port_id = 0 ,
2016-07-04 12:44:24 +03:00
. setup = lpss_spi_setup ,
2014-08-19 20:29:19 +03:00
. tx_param = & byt_tx_param ,
. rx_param = & byt_rx_param ,
2014-04-18 00:26:06 +08:00
} ,
2014-08-19 20:29:21 +03:00
[ PORT_BSW0 ] = {
2016-07-05 23:12:05 +03:00
. type = LPSS_BSW_SSP ,
2014-08-19 20:29:21 +03:00
. port_id = 0 ,
2016-07-04 12:44:24 +03:00
. setup = lpss_spi_setup ,
2014-08-19 20:29:21 +03:00
. tx_param = & bsw0_tx_param ,
. rx_param = & bsw0_rx_param ,
} ,
[ PORT_BSW1 ] = {
2016-07-05 23:12:05 +03:00
. type = LPSS_BSW_SSP ,
2014-08-19 20:29:21 +03:00
. port_id = 1 ,
2016-07-04 12:44:24 +03:00
. setup = lpss_spi_setup ,
2014-08-19 20:29:21 +03:00
. tx_param = & bsw1_tx_param ,
. rx_param = & bsw1_rx_param ,
} ,
[ PORT_BSW2 ] = {
2016-07-05 23:12:05 +03:00
. type = LPSS_BSW_SSP ,
2014-08-19 20:29:21 +03:00
. port_id = 2 ,
2016-07-04 12:44:24 +03:00
. setup = lpss_spi_setup ,
2014-08-19 20:29:21 +03:00
. tx_param = & bsw2_tx_param ,
. rx_param = & bsw2_rx_param ,
2014-04-18 00:26:06 +08:00
} ,
2016-07-04 12:44:25 +03:00
[ PORT_MRFLD ] = {
. type = PXA27x_SSP ,
. max_clk_rate = 25000000 ,
. setup = mrfld_spi_setup ,
} ,
2014-11-26 02:35:10 -08:00
[ PORT_QUARK_X1000 ] = {
. type = QUARK_X1000_SSP ,
. port_id = - 1 ,
. num_chipselect = 1 ,
. max_clk_rate = 50000000 ,
} ,
2016-02-20 20:20:22 +01:00
[ PORT_LPT ] = {
. type = LPSS_LPT_SSP ,
. port_id = 0 ,
2016-07-04 12:44:24 +03:00
. setup = lpss_spi_setup ,
2016-02-20 20:20:22 +01:00
. tx_param = & lpt_tx_param ,
. rx_param = & lpt_rx_param ,
} ,
2014-04-18 00:26:06 +08:00
} ;
static int pxa2xx_spi_pci_probe ( struct pci_dev * dev ,
2010-11-24 10:17:14 +01:00
const struct pci_device_id * ent )
{
2013-01-07 12:44:32 +02:00
struct platform_device_info pi ;
2010-11-24 10:17:14 +01:00
int ret ;
struct platform_device * pdev ;
2011-02-03 00:31:21 +05:30
struct pxa2xx_spi_master spi_pdata ;
2010-11-24 10:17:14 +01:00
struct ssp_device * ssp ;
2014-04-18 00:26:06 +08:00
struct pxa_spi_info * c ;
2014-07-25 01:10:54 +08:00
char buf [ 40 ] ;
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:32 +02:00
ret = pcim_enable_device ( dev ) ;
2010-11-24 10:17:14 +01:00
if ( ret )
return ret ;
2013-01-07 12:44:32 +02:00
ret = pcim_iomap_regions ( dev , 1 < < 0 , " PXA2xx SPI " ) ;
2013-03-05 12:05:16 +02:00
if ( ret )
2010-11-24 10:17:14 +01:00
return ret ;
2014-04-18 00:26:06 +08:00
c = & spi_info_configs [ ent - > driver_data ] ;
2016-07-04 12:44:24 +03:00
if ( c - > setup ) {
ret = c - > setup ( dev , c ) ;
if ( ret )
return ret ;
2014-08-19 20:29:19 +03:00
}
2016-07-04 12:44:24 +03:00
memset ( & spi_pdata , 0 , sizeof ( spi_pdata ) ) ;
spi_pdata . num_chipselect = ( c - > num_chipselect > 0 ) ? c - > num_chipselect : dev - > devfn ;
spi_pdata . dma_filter = c - > dma_filter ;
2014-08-19 20:29:19 +03:00
spi_pdata . tx_param = c - > tx_param ;
spi_pdata . rx_param = c - > rx_param ;
spi_pdata . enable_dma = c - > rx_param & & c - > tx_param ;
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:33 +02:00
ssp = & spi_pdata . ssp ;
2010-11-24 10:17:14 +01:00
ssp - > phys_base = pci_resource_start ( dev , 0 ) ;
2013-01-07 12:44:32 +02:00
ssp - > mmio_base = pcim_iomap_table ( dev ) [ 0 ] ;
2014-04-18 00:26:06 +08:00
ssp - > port_id = ( c - > port_id > = 0 ) ? c - > port_id : dev - > devfn ;
ssp - > type = c - > type ;
2010-11-24 10:17:14 +01:00
2017-01-21 10:06:39 +01:00
pci_set_master ( dev ) ;
ret = pci_alloc_irq_vectors ( dev , 1 , 1 , PCI_IRQ_ALL_TYPES ) ;
if ( ret < 0 )
return ret ;
ssp - > irq = pci_irq_vector ( dev , 0 ) ;
2014-07-25 01:10:54 +08:00
snprintf ( buf , sizeof ( buf ) , " pxa2xx-spi.%d " , ssp - > port_id ) ;
2016-04-19 18:10:07 -07:00
ssp - > clk = clk_register_fixed_rate ( & dev - > dev , buf , NULL , 0 ,
c - > max_clk_rate ) ;
2014-07-25 01:10:54 +08:00
if ( IS_ERR ( ssp - > clk ) )
return PTR_ERR ( ssp - > clk ) ;
2013-01-07 12:44:32 +02:00
memset ( & pi , 0 , sizeof ( pi ) ) ;
2016-08-24 14:11:30 +03:00
pi . fwnode = dev - > dev . fwnode ;
2013-01-07 12:44:32 +02:00
pi . parent = & dev - > dev ;
pi . name = " pxa2xx-spi " ;
pi . id = ssp - > port_id ;
pi . data = & spi_pdata ;
pi . size_data = sizeof ( spi_pdata ) ;
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:32 +02:00
pdev = platform_device_register_full ( & pi ) ;
2014-07-25 01:10:54 +08:00
if ( IS_ERR ( pdev ) ) {
clk_unregister ( ssp - > clk ) ;
2013-02-22 10:52:35 +08:00
return PTR_ERR ( pdev ) ;
2014-07-25 01:10:54 +08:00
}
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:33 +02:00
pci_set_drvdata ( dev , pdev ) ;
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:32 +02:00
return 0 ;
2010-11-24 10:17:14 +01:00
}
2014-04-18 00:26:06 +08:00
static void pxa2xx_spi_pci_remove ( struct pci_dev * dev )
2010-11-24 10:17:14 +01:00
{
2013-01-07 12:44:33 +02:00
struct platform_device * pdev = pci_get_drvdata ( dev ) ;
2014-07-25 01:10:54 +08:00
struct pxa2xx_spi_master * spi_pdata ;
spi_pdata = dev_get_platdata ( & pdev - > dev ) ;
2010-11-24 10:17:14 +01:00
2013-01-07 12:44:33 +02:00
platform_device_unregister ( pdev ) ;
2014-07-25 01:10:54 +08:00
clk_unregister ( spi_pdata - > ssp . clk ) ;
2010-11-24 10:17:14 +01:00
}
2014-04-18 00:26:06 +08:00
static const struct pci_device_id pxa2xx_spi_pci_devices [ ] = {
2014-11-26 02:35:10 -08:00
{ PCI_VDEVICE ( INTEL , 0x0935 ) , PORT_QUARK_X1000 } ,
2014-04-18 00:26:06 +08:00
{ PCI_VDEVICE ( INTEL , 0x0f0e ) , PORT_BYT } ,
2016-07-04 12:44:25 +03:00
{ PCI_VDEVICE ( INTEL , 0x1194 ) , PORT_MRFLD } ,
2014-08-19 20:29:21 +03:00
{ PCI_VDEVICE ( INTEL , 0x228e ) , PORT_BSW0 } ,
{ PCI_VDEVICE ( INTEL , 0x2290 ) , PORT_BSW1 } ,
{ PCI_VDEVICE ( INTEL , 0x22ac ) , PORT_BSW2 } ,
2016-07-04 12:44:27 +03:00
{ PCI_VDEVICE ( INTEL , 0x2e6a ) , PORT_CE4100 } ,
2016-02-20 20:20:22 +01:00
{ PCI_VDEVICE ( INTEL , 0x9ce6 ) , PORT_LPT } ,
2010-11-24 10:17:14 +01:00
{ } ,
} ;
2014-04-18 00:26:06 +08:00
MODULE_DEVICE_TABLE ( pci , pxa2xx_spi_pci_devices ) ;
2010-11-24 10:17:14 +01:00
2014-04-18 00:26:06 +08:00
static struct pci_driver pxa2xx_spi_pci_driver = {
. name = " pxa2xx_spi_pci " ,
. id_table = pxa2xx_spi_pci_devices ,
. probe = pxa2xx_spi_pci_probe ,
. remove = pxa2xx_spi_pci_remove ,
2010-11-24 10:17:14 +01:00
} ;
2014-04-18 00:26:06 +08:00
module_pci_driver ( pxa2xx_spi_pci_driver ) ;
2010-11-24 10:17:14 +01:00
2014-04-18 00:26:06 +08:00
MODULE_DESCRIPTION ( " CE4100/LPSS PCI-SPI glue code for PXA's driver " ) ;
2010-11-24 10:17:14 +01:00
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_AUTHOR ( " Sebastian Andrzej Siewior <bigeasy@linutronix.de> " ) ;