xref: /openbmc/linux/drivers/fpga/ts73xx-fpga.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
18e8e69d6SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
24348f7e2SFlorian Fainelli /*
34348f7e2SFlorian Fainelli  * Technologic Systems TS-73xx SBC FPGA loader
44348f7e2SFlorian Fainelli  *
54348f7e2SFlorian Fainelli  * Copyright (C) 2016 Florian Fainelli <f.fainelli@gmail.com>
64348f7e2SFlorian Fainelli  *
74348f7e2SFlorian Fainelli  * FPGA Manager Driver for the on-board Altera Cyclone II FPGA found on
84348f7e2SFlorian Fainelli  * TS-7300, heavily based on load_fpga.c in their vendor tree.
94348f7e2SFlorian Fainelli  */
104348f7e2SFlorian Fainelli 
114348f7e2SFlorian Fainelli #include <linux/delay.h>
124348f7e2SFlorian Fainelli #include <linux/io.h>
134348f7e2SFlorian Fainelli #include <linux/module.h>
144348f7e2SFlorian Fainelli #include <linux/platform_device.h>
154348f7e2SFlorian Fainelli #include <linux/string.h>
164348f7e2SFlorian Fainelli #include <linux/iopoll.h>
174348f7e2SFlorian Fainelli #include <linux/fpga/fpga-mgr.h>
184348f7e2SFlorian Fainelli 
194348f7e2SFlorian Fainelli #define TS73XX_FPGA_DATA_REG		0
204348f7e2SFlorian Fainelli #define TS73XX_FPGA_CONFIG_REG		1
214348f7e2SFlorian Fainelli 
224348f7e2SFlorian Fainelli #define TS73XX_FPGA_WRITE_DONE		0x1
234348f7e2SFlorian Fainelli #define TS73XX_FPGA_WRITE_DONE_TIMEOUT	1000	/* us */
244348f7e2SFlorian Fainelli #define TS73XX_FPGA_RESET		0x2
254348f7e2SFlorian Fainelli #define TS73XX_FPGA_RESET_LOW_DELAY	30	/* us */
264348f7e2SFlorian Fainelli #define TS73XX_FPGA_RESET_HIGH_DELAY	80	/* us */
274348f7e2SFlorian Fainelli #define TS73XX_FPGA_LOAD_OK		0x4
284348f7e2SFlorian Fainelli #define TS73XX_FPGA_CONFIG_LOAD		0x8
294348f7e2SFlorian Fainelli 
304348f7e2SFlorian Fainelli struct ts73xx_fpga_priv {
314348f7e2SFlorian Fainelli 	void __iomem	*io_base;
324348f7e2SFlorian Fainelli 	struct device	*dev;
334348f7e2SFlorian Fainelli };
344348f7e2SFlorian Fainelli 
ts73xx_fpga_write_init(struct fpga_manager * mgr,struct fpga_image_info * info,const char * buf,size_t count)354348f7e2SFlorian Fainelli static int ts73xx_fpga_write_init(struct fpga_manager *mgr,
364348f7e2SFlorian Fainelli 				  struct fpga_image_info *info,
374348f7e2SFlorian Fainelli 				  const char *buf, size_t count)
384348f7e2SFlorian Fainelli {
394348f7e2SFlorian Fainelli 	struct ts73xx_fpga_priv *priv = mgr->priv;
404348f7e2SFlorian Fainelli 
414348f7e2SFlorian Fainelli 	/* Reset the FPGA */
424348f7e2SFlorian Fainelli 	writeb(0, priv->io_base + TS73XX_FPGA_CONFIG_REG);
434348f7e2SFlorian Fainelli 	udelay(TS73XX_FPGA_RESET_LOW_DELAY);
444348f7e2SFlorian Fainelli 	writeb(TS73XX_FPGA_RESET, priv->io_base + TS73XX_FPGA_CONFIG_REG);
454348f7e2SFlorian Fainelli 	udelay(TS73XX_FPGA_RESET_HIGH_DELAY);
464348f7e2SFlorian Fainelli 
474348f7e2SFlorian Fainelli 	return 0;
484348f7e2SFlorian Fainelli }
494348f7e2SFlorian Fainelli 
ts73xx_fpga_write(struct fpga_manager * mgr,const char * buf,size_t count)504348f7e2SFlorian Fainelli static int ts73xx_fpga_write(struct fpga_manager *mgr, const char *buf,
514348f7e2SFlorian Fainelli 			     size_t count)
524348f7e2SFlorian Fainelli {
534348f7e2SFlorian Fainelli 	struct ts73xx_fpga_priv *priv = mgr->priv;
544348f7e2SFlorian Fainelli 	size_t i = 0;
554348f7e2SFlorian Fainelli 	int ret;
564348f7e2SFlorian Fainelli 	u8 reg;
574348f7e2SFlorian Fainelli 
584348f7e2SFlorian Fainelli 	while (count--) {
594348f7e2SFlorian Fainelli 		ret = readb_poll_timeout(priv->io_base + TS73XX_FPGA_CONFIG_REG,
604348f7e2SFlorian Fainelli 					 reg, !(reg & TS73XX_FPGA_WRITE_DONE),
614348f7e2SFlorian Fainelli 					 1, TS73XX_FPGA_WRITE_DONE_TIMEOUT);
624348f7e2SFlorian Fainelli 		if (ret < 0)
634348f7e2SFlorian Fainelli 			return ret;
644348f7e2SFlorian Fainelli 
654348f7e2SFlorian Fainelli 		writeb(buf[i], priv->io_base + TS73XX_FPGA_DATA_REG);
664348f7e2SFlorian Fainelli 		i++;
674348f7e2SFlorian Fainelli 	}
684348f7e2SFlorian Fainelli 
694348f7e2SFlorian Fainelli 	return 0;
704348f7e2SFlorian Fainelli }
714348f7e2SFlorian Fainelli 
ts73xx_fpga_write_complete(struct fpga_manager * mgr,struct fpga_image_info * info)724348f7e2SFlorian Fainelli static int ts73xx_fpga_write_complete(struct fpga_manager *mgr,
734348f7e2SFlorian Fainelli 				      struct fpga_image_info *info)
744348f7e2SFlorian Fainelli {
754348f7e2SFlorian Fainelli 	struct ts73xx_fpga_priv *priv = mgr->priv;
764348f7e2SFlorian Fainelli 	u8 reg;
774348f7e2SFlorian Fainelli 
784348f7e2SFlorian Fainelli 	usleep_range(1000, 2000);
794348f7e2SFlorian Fainelli 	reg = readb(priv->io_base + TS73XX_FPGA_CONFIG_REG);
804348f7e2SFlorian Fainelli 	reg |= TS73XX_FPGA_CONFIG_LOAD;
814348f7e2SFlorian Fainelli 	writeb(reg, priv->io_base + TS73XX_FPGA_CONFIG_REG);
824348f7e2SFlorian Fainelli 
834348f7e2SFlorian Fainelli 	usleep_range(1000, 2000);
844348f7e2SFlorian Fainelli 	reg = readb(priv->io_base + TS73XX_FPGA_CONFIG_REG);
854348f7e2SFlorian Fainelli 	reg &= ~TS73XX_FPGA_CONFIG_LOAD;
864348f7e2SFlorian Fainelli 	writeb(reg, priv->io_base + TS73XX_FPGA_CONFIG_REG);
874348f7e2SFlorian Fainelli 
884348f7e2SFlorian Fainelli 	reg = readb(priv->io_base + TS73XX_FPGA_CONFIG_REG);
894348f7e2SFlorian Fainelli 	if ((reg & TS73XX_FPGA_LOAD_OK) != TS73XX_FPGA_LOAD_OK)
904348f7e2SFlorian Fainelli 		return -ETIMEDOUT;
914348f7e2SFlorian Fainelli 
924348f7e2SFlorian Fainelli 	return 0;
934348f7e2SFlorian Fainelli }
944348f7e2SFlorian Fainelli 
954348f7e2SFlorian Fainelli static const struct fpga_manager_ops ts73xx_fpga_ops = {
964348f7e2SFlorian Fainelli 	.write_init	= ts73xx_fpga_write_init,
974348f7e2SFlorian Fainelli 	.write		= ts73xx_fpga_write,
984348f7e2SFlorian Fainelli 	.write_complete	= ts73xx_fpga_write_complete,
994348f7e2SFlorian Fainelli };
1004348f7e2SFlorian Fainelli 
ts73xx_fpga_probe(struct platform_device * pdev)1014348f7e2SFlorian Fainelli static int ts73xx_fpga_probe(struct platform_device *pdev)
1024348f7e2SFlorian Fainelli {
1034348f7e2SFlorian Fainelli 	struct device *kdev = &pdev->dev;
1044348f7e2SFlorian Fainelli 	struct ts73xx_fpga_priv *priv;
1057085e2a9SAlan Tull 	struct fpga_manager *mgr;
1064348f7e2SFlorian Fainelli 
1074348f7e2SFlorian Fainelli 	priv = devm_kzalloc(kdev, sizeof(*priv), GFP_KERNEL);
1084348f7e2SFlorian Fainelli 	if (!priv)
1094348f7e2SFlorian Fainelli 		return -ENOMEM;
1104348f7e2SFlorian Fainelli 
1114348f7e2SFlorian Fainelli 	priv->dev = kdev;
1124348f7e2SFlorian Fainelli 
113*533aae16SYangtao Li 	priv->io_base = devm_platform_ioremap_resource(pdev, 0);
1141d39387cSDing Xiang 	if (IS_ERR(priv->io_base))
1154348f7e2SFlorian Fainelli 		return PTR_ERR(priv->io_base);
1164348f7e2SFlorian Fainelli 
1174ba0b2c2SRuss Weight 	mgr = devm_fpga_mgr_register(kdev, "TS-73xx FPGA Manager",
1184348f7e2SFlorian Fainelli 				     &ts73xx_fpga_ops, priv);
1194ba0b2c2SRuss Weight 	return PTR_ERR_OR_ZERO(mgr);
1204348f7e2SFlorian Fainelli }
1214348f7e2SFlorian Fainelli 
1224348f7e2SFlorian Fainelli static struct platform_driver ts73xx_fpga_driver = {
1234348f7e2SFlorian Fainelli 	.driver	= {
1244348f7e2SFlorian Fainelli 		.name	= "ts73xx-fpga-mgr",
1254348f7e2SFlorian Fainelli 	},
1264348f7e2SFlorian Fainelli 	.probe	= ts73xx_fpga_probe,
1274348f7e2SFlorian Fainelli };
1284348f7e2SFlorian Fainelli module_platform_driver(ts73xx_fpga_driver);
1294348f7e2SFlorian Fainelli 
1304348f7e2SFlorian Fainelli MODULE_AUTHOR("Florian Fainelli <f.fainelli@gmail.com>");
1314348f7e2SFlorian Fainelli MODULE_DESCRIPTION("TS-73xx FPGA Manager driver");
1324348f7e2SFlorian Fainelli MODULE_LICENSE("GPL v2");
133