123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617 |
- /*
- * FPGA Manager Driver for Altera SOCFPGA
- *
- * Copyright (C) 2013-2015 Altera Corporation
- *
- * This program is free software; you can redistribute it and/or modify it
- * under the terms and conditions of the GNU General Public License,
- * version 2, as published by the Free Software Foundation.
- *
- * This program is distributed in the hope it will be useful, but WITHOUT
- * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
- * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
- * more details.
- *
- * You should have received a copy of the GNU General Public License along with
- * this program. If not, see <http://www.gnu.org/licenses/>.
- */
- #include <linux/completion.h>
- #include <linux/delay.h>
- #include <linux/fpga/fpga-mgr.h>
- #include <linux/interrupt.h>
- #include <linux/io.h>
- #include <linux/module.h>
- #include <linux/of_address.h>
- #include <linux/of_irq.h>
- #include <linux/pm.h>
- /* Register offsets */
- #define SOCFPGA_FPGMGR_STAT_OFST 0x0
- #define SOCFPGA_FPGMGR_CTL_OFST 0x4
- #define SOCFPGA_FPGMGR_DCLKCNT_OFST 0x8
- #define SOCFPGA_FPGMGR_DCLKSTAT_OFST 0xc
- #define SOCFPGA_FPGMGR_GPIO_INTEN_OFST 0x830
- #define SOCFPGA_FPGMGR_GPIO_INTMSK_OFST 0x834
- #define SOCFPGA_FPGMGR_GPIO_INTTYPE_LEVEL_OFST 0x838
- #define SOCFPGA_FPGMGR_GPIO_INT_POL_OFST 0x83c
- #define SOCFPGA_FPGMGR_GPIO_INTSTAT_OFST 0x840
- #define SOCFPGA_FPGMGR_GPIO_RAW_INTSTAT_OFST 0x844
- #define SOCFPGA_FPGMGR_GPIO_PORTA_EOI_OFST 0x84c
- #define SOCFPGA_FPGMGR_GPIO_EXT_PORTA_OFST 0x850
- /* Register bit defines */
- /* SOCFPGA_FPGMGR_STAT register mode field values */
- #define SOCFPGA_FPGMGR_STAT_POWER_UP 0x0 /*ramping*/
- #define SOCFPGA_FPGMGR_STAT_RESET 0x1
- #define SOCFPGA_FPGMGR_STAT_CFG 0x2
- #define SOCFPGA_FPGMGR_STAT_INIT 0x3
- #define SOCFPGA_FPGMGR_STAT_USER_MODE 0x4
- #define SOCFPGA_FPGMGR_STAT_UNKNOWN 0x5
- #define SOCFPGA_FPGMGR_STAT_STATE_MASK 0x7
- /* This is a flag value that doesn't really happen in this register field */
- #define SOCFPGA_FPGMGR_STAT_POWER_OFF 0x0
- #define MSEL_PP16_FAST_NOAES_NODC 0x0
- #define MSEL_PP16_FAST_AES_NODC 0x1
- #define MSEL_PP16_FAST_AESOPT_DC 0x2
- #define MSEL_PP16_SLOW_NOAES_NODC 0x4
- #define MSEL_PP16_SLOW_AES_NODC 0x5
- #define MSEL_PP16_SLOW_AESOPT_DC 0x6
- #define MSEL_PP32_FAST_NOAES_NODC 0x8
- #define MSEL_PP32_FAST_AES_NODC 0x9
- #define MSEL_PP32_FAST_AESOPT_DC 0xa
- #define MSEL_PP32_SLOW_NOAES_NODC 0xc
- #define MSEL_PP32_SLOW_AES_NODC 0xd
- #define MSEL_PP32_SLOW_AESOPT_DC 0xe
- #define SOCFPGA_FPGMGR_STAT_MSEL_MASK 0x000000f8
- #define SOCFPGA_FPGMGR_STAT_MSEL_SHIFT 3
- /* SOCFPGA_FPGMGR_CTL register */
- #define SOCFPGA_FPGMGR_CTL_EN 0x00000001
- #define SOCFPGA_FPGMGR_CTL_NCE 0x00000002
- #define SOCFPGA_FPGMGR_CTL_NCFGPULL 0x00000004
- #define CDRATIO_X1 0x00000000
- #define CDRATIO_X2 0x00000040
- #define CDRATIO_X4 0x00000080
- #define CDRATIO_X8 0x000000c0
- #define SOCFPGA_FPGMGR_CTL_CDRATIO_MASK 0x000000c0
- #define SOCFPGA_FPGMGR_CTL_AXICFGEN 0x00000100
- #define CFGWDTH_16 0x00000000
- #define CFGWDTH_32 0x00000200
- #define SOCFPGA_FPGMGR_CTL_CFGWDTH_MASK 0x00000200
- /* SOCFPGA_FPGMGR_DCLKSTAT register */
- #define SOCFPGA_FPGMGR_DCLKSTAT_DCNTDONE_E_DONE 0x1
- /* SOCFPGA_FPGMGR_GPIO_* registers share the same bit positions */
- #define SOCFPGA_FPGMGR_MON_NSTATUS 0x0001
- #define SOCFPGA_FPGMGR_MON_CONF_DONE 0x0002
- #define SOCFPGA_FPGMGR_MON_INIT_DONE 0x0004
- #define SOCFPGA_FPGMGR_MON_CRC_ERROR 0x0008
- #define SOCFPGA_FPGMGR_MON_CVP_CONF_DONE 0x0010
- #define SOCFPGA_FPGMGR_MON_PR_READY 0x0020
- #define SOCFPGA_FPGMGR_MON_PR_ERROR 0x0040
- #define SOCFPGA_FPGMGR_MON_PR_DONE 0x0080
- #define SOCFPGA_FPGMGR_MON_NCONFIG_PIN 0x0100
- #define SOCFPGA_FPGMGR_MON_NSTATUS_PIN 0x0200
- #define SOCFPGA_FPGMGR_MON_CONF_DONE_PIN 0x0400
- #define SOCFPGA_FPGMGR_MON_FPGA_POWER_ON 0x0800
- #define SOCFPGA_FPGMGR_MON_STATUS_MASK 0x0fff
- #define SOCFPGA_FPGMGR_NUM_SUPPLIES 3
- #define SOCFPGA_RESUME_TIMEOUT 3
- /* In power-up order. Reverse for power-down. */
- static const char *supply_names[SOCFPGA_FPGMGR_NUM_SUPPLIES] __maybe_unused = {
- "FPGA-1.5V",
- "FPGA-1.1V",
- "FPGA-2.5V",
- };
- struct socfpga_fpga_priv {
- void __iomem *fpga_base_addr;
- void __iomem *fpga_data_addr;
- struct completion status_complete;
- int irq;
- };
- struct cfgmgr_mode {
- /* Values to set in the CTRL register */
- u32 ctrl;
- /* flag that this table entry is a valid mode */
- bool valid;
- };
- /* For SOCFPGA_FPGMGR_STAT_MSEL field */
- static struct cfgmgr_mode cfgmgr_modes[] = {
- [MSEL_PP16_FAST_NOAES_NODC] = { CFGWDTH_16 | CDRATIO_X1, 1 },
- [MSEL_PP16_FAST_AES_NODC] = { CFGWDTH_16 | CDRATIO_X2, 1 },
- [MSEL_PP16_FAST_AESOPT_DC] = { CFGWDTH_16 | CDRATIO_X4, 1 },
- [MSEL_PP16_SLOW_NOAES_NODC] = { CFGWDTH_16 | CDRATIO_X1, 1 },
- [MSEL_PP16_SLOW_AES_NODC] = { CFGWDTH_16 | CDRATIO_X2, 1 },
- [MSEL_PP16_SLOW_AESOPT_DC] = { CFGWDTH_16 | CDRATIO_X4, 1 },
- [MSEL_PP32_FAST_NOAES_NODC] = { CFGWDTH_32 | CDRATIO_X1, 1 },
- [MSEL_PP32_FAST_AES_NODC] = { CFGWDTH_32 | CDRATIO_X4, 1 },
- [MSEL_PP32_FAST_AESOPT_DC] = { CFGWDTH_32 | CDRATIO_X8, 1 },
- [MSEL_PP32_SLOW_NOAES_NODC] = { CFGWDTH_32 | CDRATIO_X1, 1 },
- [MSEL_PP32_SLOW_AES_NODC] = { CFGWDTH_32 | CDRATIO_X4, 1 },
- [MSEL_PP32_SLOW_AESOPT_DC] = { CFGWDTH_32 | CDRATIO_X8, 1 },
- };
- static u32 socfpga_fpga_readl(struct socfpga_fpga_priv *priv, u32 reg_offset)
- {
- return readl(priv->fpga_base_addr + reg_offset);
- }
- static void socfpga_fpga_writel(struct socfpga_fpga_priv *priv, u32 reg_offset,
- u32 value)
- {
- writel(value, priv->fpga_base_addr + reg_offset);
- }
- static u32 socfpga_fpga_raw_readl(struct socfpga_fpga_priv *priv,
- u32 reg_offset)
- {
- return __raw_readl(priv->fpga_base_addr + reg_offset);
- }
- static void socfpga_fpga_raw_writel(struct socfpga_fpga_priv *priv,
- u32 reg_offset, u32 value)
- {
- __raw_writel(value, priv->fpga_base_addr + reg_offset);
- }
- static void socfpga_fpga_data_writel(struct socfpga_fpga_priv *priv, u32 value)
- {
- writel(value, priv->fpga_data_addr);
- }
- static inline void socfpga_fpga_set_bitsl(struct socfpga_fpga_priv *priv,
- u32 offset, u32 bits)
- {
- u32 val;
- val = socfpga_fpga_readl(priv, offset);
- val |= bits;
- socfpga_fpga_writel(priv, offset, val);
- }
- static inline void socfpga_fpga_clr_bitsl(struct socfpga_fpga_priv *priv,
- u32 offset, u32 bits)
- {
- u32 val;
- val = socfpga_fpga_readl(priv, offset);
- val &= ~bits;
- socfpga_fpga_writel(priv, offset, val);
- }
- static u32 socfpga_fpga_mon_status_get(struct socfpga_fpga_priv *priv)
- {
- return socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_GPIO_EXT_PORTA_OFST) &
- SOCFPGA_FPGMGR_MON_STATUS_MASK;
- }
- static u32 socfpga_fpga_state_get(struct socfpga_fpga_priv *priv)
- {
- u32 status = socfpga_fpga_mon_status_get(priv);
- if ((status & SOCFPGA_FPGMGR_MON_FPGA_POWER_ON) == 0)
- return SOCFPGA_FPGMGR_STAT_POWER_OFF;
- return socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_STAT_OFST) &
- SOCFPGA_FPGMGR_STAT_STATE_MASK;
- }
- static void socfpga_fpga_clear_done_status(struct socfpga_fpga_priv *priv)
- {
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_DCLKSTAT_OFST,
- SOCFPGA_FPGMGR_DCLKSTAT_DCNTDONE_E_DONE);
- }
- /*
- * Set the DCLKCNT, wait for DCLKSTAT to report the count completed, and clear
- * the complete status.
- */
- static int socfpga_fpga_dclk_set_and_wait_clear(struct socfpga_fpga_priv *priv,
- u32 count)
- {
- int timeout = 2;
- u32 done;
- /* Clear any existing DONE status. */
- if (socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_DCLKSTAT_OFST))
- socfpga_fpga_clear_done_status(priv);
- /* Issue the DCLK count. */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_DCLKCNT_OFST, count);
- /* Poll DCLKSTAT to see if it completed in the timeout period. */
- do {
- done = socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_DCLKSTAT_OFST);
- if (done == SOCFPGA_FPGMGR_DCLKSTAT_DCNTDONE_E_DONE) {
- socfpga_fpga_clear_done_status(priv);
- return 0;
- }
- udelay(1);
- } while (timeout--);
- return -ETIMEDOUT;
- }
- static int socfpga_fpga_wait_for_state(struct socfpga_fpga_priv *priv,
- u32 state)
- {
- int timeout = 2;
- /*
- * HW doesn't support an interrupt for changes in state, so poll to see
- * if it matches the requested state within the timeout period.
- */
- do {
- if ((socfpga_fpga_state_get(priv) & state) != 0)
- return 0;
- msleep(20);
- } while (timeout--);
- return -ETIMEDOUT;
- }
- static void socfpga_fpga_enable_irqs(struct socfpga_fpga_priv *priv, u32 irqs)
- {
- /* set irqs to level sensitive */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_INTTYPE_LEVEL_OFST, 0);
- /* set interrupt polarity */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_INT_POL_OFST, irqs);
- /* clear irqs */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_PORTA_EOI_OFST, irqs);
- /* unmask interrupts */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_INTMSK_OFST, 0);
- /* enable interrupts */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_INTEN_OFST, irqs);
- }
- static void socfpga_fpga_disable_irqs(struct socfpga_fpga_priv *priv)
- {
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_INTEN_OFST, 0);
- }
- static irqreturn_t socfpga_fpga_isr(int irq, void *dev_id)
- {
- struct socfpga_fpga_priv *priv = dev_id;
- u32 irqs, st;
- bool conf_done, nstatus;
- /* clear irqs */
- irqs = socfpga_fpga_raw_readl(priv, SOCFPGA_FPGMGR_GPIO_INTSTAT_OFST);
- socfpga_fpga_raw_writel(priv, SOCFPGA_FPGMGR_GPIO_PORTA_EOI_OFST, irqs);
- st = socfpga_fpga_raw_readl(priv, SOCFPGA_FPGMGR_GPIO_EXT_PORTA_OFST);
- conf_done = (st & SOCFPGA_FPGMGR_MON_CONF_DONE) != 0;
- nstatus = (st & SOCFPGA_FPGMGR_MON_NSTATUS) != 0;
- /* success */
- if (conf_done && nstatus) {
- /* disable irqs */
- socfpga_fpga_raw_writel(priv,
- SOCFPGA_FPGMGR_GPIO_INTEN_OFST, 0);
- complete(&priv->status_complete);
- }
- return IRQ_HANDLED;
- }
- static int socfpga_fpga_wait_for_config_done(struct socfpga_fpga_priv *priv)
- {
- int timeout, ret = 0;
- socfpga_fpga_disable_irqs(priv);
- init_completion(&priv->status_complete);
- socfpga_fpga_enable_irqs(priv, SOCFPGA_FPGMGR_MON_CONF_DONE);
- timeout = wait_for_completion_interruptible_timeout(
- &priv->status_complete,
- msecs_to_jiffies(10));
- if (timeout == 0)
- ret = -ETIMEDOUT;
- socfpga_fpga_disable_irqs(priv);
- return ret;
- }
- static int socfpga_fpga_cfg_mode_get(struct socfpga_fpga_priv *priv)
- {
- u32 msel;
- msel = socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_STAT_OFST);
- msel &= SOCFPGA_FPGMGR_STAT_MSEL_MASK;
- msel >>= SOCFPGA_FPGMGR_STAT_MSEL_SHIFT;
- /* Check that this MSEL setting is supported */
- if ((msel >= ARRAY_SIZE(cfgmgr_modes)) || !cfgmgr_modes[msel].valid)
- return -EINVAL;
- return msel;
- }
- static int socfpga_fpga_cfg_mode_set(struct socfpga_fpga_priv *priv)
- {
- u32 ctrl_reg;
- int mode;
- /* get value from MSEL pins */
- mode = socfpga_fpga_cfg_mode_get(priv);
- if (mode < 0)
- return mode;
- /* Adjust CTRL for the CDRATIO */
- ctrl_reg = socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_CTL_OFST);
- ctrl_reg &= ~SOCFPGA_FPGMGR_CTL_CDRATIO_MASK;
- ctrl_reg &= ~SOCFPGA_FPGMGR_CTL_CFGWDTH_MASK;
- ctrl_reg |= cfgmgr_modes[mode].ctrl;
- /* Set NCE to 0. */
- ctrl_reg &= ~SOCFPGA_FPGMGR_CTL_NCE;
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_CTL_OFST, ctrl_reg);
- return 0;
- }
- static int socfpga_fpga_reset(struct fpga_manager *mgr)
- {
- struct socfpga_fpga_priv *priv = mgr->priv;
- u32 ctrl_reg, status;
- int ret;
- /*
- * Step 1:
- * - Set CTRL.CFGWDTH, CTRL.CDRATIO to match cfg mode
- * - Set CTRL.NCE to 0
- */
- ret = socfpga_fpga_cfg_mode_set(priv);
- if (ret)
- return ret;
- /* Step 2: Set CTRL.EN to 1 */
- socfpga_fpga_set_bitsl(priv, SOCFPGA_FPGMGR_CTL_OFST,
- SOCFPGA_FPGMGR_CTL_EN);
- /* Step 3: Set CTRL.NCONFIGPULL to 1 to put FPGA in reset */
- ctrl_reg = socfpga_fpga_readl(priv, SOCFPGA_FPGMGR_CTL_OFST);
- ctrl_reg |= SOCFPGA_FPGMGR_CTL_NCFGPULL;
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_CTL_OFST, ctrl_reg);
- /* Step 4: Wait for STATUS.MODE to report FPGA is in reset phase */
- status = socfpga_fpga_wait_for_state(priv, SOCFPGA_FPGMGR_STAT_RESET);
- /* Step 5: Set CONTROL.NCONFIGPULL to 0 to release FPGA from reset */
- ctrl_reg &= ~SOCFPGA_FPGMGR_CTL_NCFGPULL;
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_CTL_OFST, ctrl_reg);
- /* Timeout waiting for reset */
- if (status)
- return -ETIMEDOUT;
- return 0;
- }
- /*
- * Prepare the FPGA to receive the configuration data.
- */
- static int socfpga_fpga_ops_configure_init(struct fpga_manager *mgr, u32 flags,
- const char *buf, size_t count)
- {
- struct socfpga_fpga_priv *priv = mgr->priv;
- int ret;
- if (flags & FPGA_MGR_PARTIAL_RECONFIG) {
- dev_err(&mgr->dev, "Partial reconfiguration not supported.\n");
- return -EINVAL;
- }
- /* Steps 1 - 5: Reset the FPGA */
- ret = socfpga_fpga_reset(mgr);
- if (ret)
- return ret;
- /* Step 6: Wait for FPGA to enter configuration phase */
- if (socfpga_fpga_wait_for_state(priv, SOCFPGA_FPGMGR_STAT_CFG))
- return -ETIMEDOUT;
- /* Step 7: Clear nSTATUS interrupt */
- socfpga_fpga_writel(priv, SOCFPGA_FPGMGR_GPIO_PORTA_EOI_OFST,
- SOCFPGA_FPGMGR_MON_NSTATUS);
- /* Step 8: Set CTRL.AXICFGEN to 1 to enable transfer of config data */
- socfpga_fpga_set_bitsl(priv, SOCFPGA_FPGMGR_CTL_OFST,
- SOCFPGA_FPGMGR_CTL_AXICFGEN);
- return 0;
- }
- /*
- * Step 9: write data to the FPGA data register
- */
- static int socfpga_fpga_ops_configure_write(struct fpga_manager *mgr,
- const char *buf, size_t count)
- {
- struct socfpga_fpga_priv *priv = mgr->priv;
- u32 *buffer_32 = (u32 *)buf;
- size_t i = 0;
- if (count <= 0)
- return -EINVAL;
- /* Write out the complete 32-bit chunks. */
- while (count >= sizeof(u32)) {
- socfpga_fpga_data_writel(priv, buffer_32[i++]);
- count -= sizeof(u32);
- }
- /* Write out remaining non 32-bit chunks. */
- switch (count) {
- case 3:
- socfpga_fpga_data_writel(priv, buffer_32[i++] & 0x00ffffff);
- break;
- case 2:
- socfpga_fpga_data_writel(priv, buffer_32[i++] & 0x0000ffff);
- break;
- case 1:
- socfpga_fpga_data_writel(priv, buffer_32[i++] & 0x000000ff);
- break;
- case 0:
- break;
- default:
- /* This will never happen. */
- return -EFAULT;
- }
- return 0;
- }
- static int socfpga_fpga_ops_configure_complete(struct fpga_manager *mgr,
- u32 flags)
- {
- struct socfpga_fpga_priv *priv = mgr->priv;
- u32 status;
- /*
- * Step 10:
- * - Observe CONF_DONE and nSTATUS (active low)
- * - if CONF_DONE = 1 and nSTATUS = 1, configuration was successful
- * - if CONF_DONE = 0 and nSTATUS = 0, configuration failed
- */
- status = socfpga_fpga_wait_for_config_done(priv);
- if (status)
- return status;
- /* Step 11: Clear CTRL.AXICFGEN to disable transfer of config data */
- socfpga_fpga_clr_bitsl(priv, SOCFPGA_FPGMGR_CTL_OFST,
- SOCFPGA_FPGMGR_CTL_AXICFGEN);
- /*
- * Step 12:
- * - Write 4 to DCLKCNT
- * - Wait for STATUS.DCNTDONE = 1
- * - Clear W1C bit in STATUS.DCNTDONE
- */
- if (socfpga_fpga_dclk_set_and_wait_clear(priv, 4))
- return -ETIMEDOUT;
- /* Step 13: Wait for STATUS.MODE to report USER MODE */
- if (socfpga_fpga_wait_for_state(priv, SOCFPGA_FPGMGR_STAT_USER_MODE))
- return -ETIMEDOUT;
- /* Step 14: Set CTRL.EN to 0 */
- socfpga_fpga_clr_bitsl(priv, SOCFPGA_FPGMGR_CTL_OFST,
- SOCFPGA_FPGMGR_CTL_EN);
- return 0;
- }
- /* Translate state register values to FPGA framework state */
- static const enum fpga_mgr_states socfpga_state_to_framework_state[] = {
- [SOCFPGA_FPGMGR_STAT_POWER_OFF] = FPGA_MGR_STATE_POWER_OFF,
- [SOCFPGA_FPGMGR_STAT_RESET] = FPGA_MGR_STATE_RESET,
- [SOCFPGA_FPGMGR_STAT_CFG] = FPGA_MGR_STATE_WRITE_INIT,
- [SOCFPGA_FPGMGR_STAT_INIT] = FPGA_MGR_STATE_WRITE_INIT,
- [SOCFPGA_FPGMGR_STAT_USER_MODE] = FPGA_MGR_STATE_OPERATING,
- [SOCFPGA_FPGMGR_STAT_UNKNOWN] = FPGA_MGR_STATE_UNKNOWN,
- };
- static enum fpga_mgr_states socfpga_fpga_ops_state(struct fpga_manager *mgr)
- {
- struct socfpga_fpga_priv *priv = mgr->priv;
- enum fpga_mgr_states ret;
- u32 state;
- state = socfpga_fpga_state_get(priv);
- if (state < ARRAY_SIZE(socfpga_state_to_framework_state))
- ret = socfpga_state_to_framework_state[state];
- else
- ret = FPGA_MGR_STATE_UNKNOWN;
- return ret;
- }
- static const struct fpga_manager_ops socfpga_fpga_ops = {
- .state = socfpga_fpga_ops_state,
- .write_init = socfpga_fpga_ops_configure_init,
- .write = socfpga_fpga_ops_configure_write,
- .write_complete = socfpga_fpga_ops_configure_complete,
- };
- static int socfpga_fpga_probe(struct platform_device *pdev)
- {
- struct device *dev = &pdev->dev;
- struct socfpga_fpga_priv *priv;
- struct resource *res;
- int ret;
- priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
- if (!priv)
- return -ENOMEM;
- res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
- priv->fpga_base_addr = devm_ioremap_resource(dev, res);
- if (IS_ERR(priv->fpga_base_addr))
- return PTR_ERR(priv->fpga_base_addr);
- res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
- priv->fpga_data_addr = devm_ioremap_resource(dev, res);
- if (IS_ERR(priv->fpga_data_addr))
- return PTR_ERR(priv->fpga_data_addr);
- priv->irq = platform_get_irq(pdev, 0);
- if (priv->irq < 0)
- return priv->irq;
- ret = devm_request_irq(dev, priv->irq, socfpga_fpga_isr, 0,
- dev_name(dev), priv);
- if (ret)
- return ret;
- return fpga_mgr_register(dev, "Altera SOCFPGA FPGA Manager",
- &socfpga_fpga_ops, priv);
- }
- static int socfpga_fpga_remove(struct platform_device *pdev)
- {
- fpga_mgr_unregister(&pdev->dev);
- return 0;
- }
- #ifdef CONFIG_OF
- static const struct of_device_id socfpga_fpga_of_match[] = {
- { .compatible = "altr,socfpga-fpga-mgr", },
- {},
- };
- MODULE_DEVICE_TABLE(of, socfpga_fpga_of_match);
- #endif
- static struct platform_driver socfpga_fpga_driver = {
- .probe = socfpga_fpga_probe,
- .remove = socfpga_fpga_remove,
- .driver = {
- .name = "socfpga_fpga_manager",
- .of_match_table = of_match_ptr(socfpga_fpga_of_match),
- },
- };
- module_platform_driver(socfpga_fpga_driver);
- MODULE_AUTHOR("Alan Tull <atull@opensource.altera.com>");
- MODULE_DESCRIPTION("Altera SOCFPGA FPGA Manager");
- MODULE_LICENSE("GPL v2");
|