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