2019-05-29 17:17:59 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2017-02-28 01:14:22 +03:00
/*
* Technologic Systems TS - 73 xx SBC FPGA loader
*
* Copyright ( C ) 2016 Florian Fainelli < f . fainelli @ gmail . com >
*
* FPGA Manager Driver for the on - board Altera Cyclone II FPGA found on
* TS - 7300 , heavily based on load_fpga . c in their vendor tree .
*/
# include <linux/delay.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/string.h>
# include <linux/iopoll.h>
# include <linux/fpga/fpga-mgr.h>
# define TS73XX_FPGA_DATA_REG 0
# define TS73XX_FPGA_CONFIG_REG 1
# define TS73XX_FPGA_WRITE_DONE 0x1
# define TS73XX_FPGA_WRITE_DONE_TIMEOUT 1000 /* us */
# define TS73XX_FPGA_RESET 0x2
# define TS73XX_FPGA_RESET_LOW_DELAY 30 /* us */
# define TS73XX_FPGA_RESET_HIGH_DELAY 80 /* us */
# define TS73XX_FPGA_LOAD_OK 0x4
# define TS73XX_FPGA_CONFIG_LOAD 0x8
struct ts73xx_fpga_priv {
void __iomem * io_base ;
struct device * dev ;
} ;
static int ts73xx_fpga_write_init ( struct fpga_manager * mgr ,
struct fpga_image_info * info ,
const char * buf , size_t count )
{
struct ts73xx_fpga_priv * priv = mgr - > priv ;
/* Reset the FPGA */
writeb ( 0 , priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
udelay ( TS73XX_FPGA_RESET_LOW_DELAY ) ;
writeb ( TS73XX_FPGA_RESET , priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
udelay ( TS73XX_FPGA_RESET_HIGH_DELAY ) ;
return 0 ;
}
static int ts73xx_fpga_write ( struct fpga_manager * mgr , const char * buf ,
size_t count )
{
struct ts73xx_fpga_priv * priv = mgr - > priv ;
size_t i = 0 ;
int ret ;
u8 reg ;
while ( count - - ) {
ret = readb_poll_timeout ( priv - > io_base + TS73XX_FPGA_CONFIG_REG ,
reg , ! ( reg & TS73XX_FPGA_WRITE_DONE ) ,
1 , TS73XX_FPGA_WRITE_DONE_TIMEOUT ) ;
if ( ret < 0 )
return ret ;
writeb ( buf [ i ] , priv - > io_base + TS73XX_FPGA_DATA_REG ) ;
i + + ;
}
return 0 ;
}
static int ts73xx_fpga_write_complete ( struct fpga_manager * mgr ,
struct fpga_image_info * info )
{
struct ts73xx_fpga_priv * priv = mgr - > priv ;
u8 reg ;
usleep_range ( 1000 , 2000 ) ;
reg = readb ( priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
reg | = TS73XX_FPGA_CONFIG_LOAD ;
writeb ( reg , priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
usleep_range ( 1000 , 2000 ) ;
reg = readb ( priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
reg & = ~ TS73XX_FPGA_CONFIG_LOAD ;
writeb ( reg , priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
reg = readb ( priv - > io_base + TS73XX_FPGA_CONFIG_REG ) ;
if ( ( reg & TS73XX_FPGA_LOAD_OK ) ! = TS73XX_FPGA_LOAD_OK )
return - ETIMEDOUT ;
return 0 ;
}
static const struct fpga_manager_ops ts73xx_fpga_ops = {
. write_init = ts73xx_fpga_write_init ,
. write = ts73xx_fpga_write ,
. write_complete = ts73xx_fpga_write_complete ,
} ;
static int ts73xx_fpga_probe ( struct platform_device * pdev )
{
struct device * kdev = & pdev - > dev ;
struct ts73xx_fpga_priv * priv ;
2018-05-17 02:49:55 +03:00
struct fpga_manager * mgr ;
2017-02-28 01:14:22 +03:00
priv = devm_kzalloc ( kdev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
priv - > dev = kdev ;
2023-07-05 12:46:54 +03:00
priv - > io_base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2019-09-10 12:26:56 +03:00
if ( IS_ERR ( priv - > io_base ) )
2017-02-28 01:14:22 +03:00
return PTR_ERR ( priv - > io_base ) ;
2021-11-19 04:55:51 +03:00
mgr = devm_fpga_mgr_register ( kdev , " TS-73xx FPGA Manager " ,
& ts73xx_fpga_ops , priv ) ;
return PTR_ERR_OR_ZERO ( mgr ) ;
2017-02-28 01:14:22 +03:00
}
static struct platform_driver ts73xx_fpga_driver = {
. driver = {
. name = " ts73xx-fpga-mgr " ,
} ,
. probe = ts73xx_fpga_probe ,
} ;
module_platform_driver ( ts73xx_fpga_driver ) ;
MODULE_AUTHOR ( " Florian Fainelli <f.fainelli@gmail.com> " ) ;
MODULE_DESCRIPTION ( " TS-73xx FPGA Manager driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;