123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Copyright (C) 2017 Intel Corporation <www.intel.com>
- */
- #include <asm/io.h>
- #include <asm/arch/fpga_manager.h>
- #include <asm/arch/reset_manager.h>
- #include <asm/arch/system_manager.h>
- #include <asm/arch/sdram.h>
- #include <asm/arch/misc.h>
- #include <altera.h>
- #include <common.h>
- #include <errno.h>
- #include <wait_bit.h>
- #include <watchdog.h>
- #define CFGWDTH_32 1
- #define MIN_BITSTREAM_SIZECHECK 230
- #define ENCRYPTION_OFFSET 69
- #define COMPRESSION_OFFSET 229
- #define FPGA_TIMEOUT_MSEC 1000 /* timeout in ms */
- #define FPGA_TIMEOUT_CNT 0x1000000
- static const struct socfpga_fpga_manager *fpga_manager_base =
- (void *)SOCFPGA_FPGAMGRREGS_ADDRESS;
- static const struct socfpga_system_manager *system_manager_base =
- (void *)SOCFPGA_SYSMGR_ADDRESS;
- static void fpgamgr_set_cd_ratio(unsigned long ratio);
- static uint32_t fpgamgr_get_msel(void)
- {
- u32 reg;
- reg = readl(&fpga_manager_base->imgcfg_stat);
- reg = (reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_MSEL_SET_MSD) >>
- ALT_FPGAMGR_IMGCFG_STAT_F2S_MSEL0_LSB;
- return reg;
- }
- static void fpgamgr_set_cfgwdth(int width)
- {
- if (width)
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_CFGWIDTH_SET_MSK);
- else
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_CFGWIDTH_SET_MSK);
- }
- int is_fpgamgr_user_mode(void)
- {
- return (readl(&fpga_manager_base->imgcfg_stat) &
- ALT_FPGAMGR_IMGCFG_STAT_F2S_USERMODE_SET_MSK) != 0;
- }
- static int wait_for_user_mode(void)
- {
- return wait_for_bit_le32(&fpga_manager_base->imgcfg_stat,
- ALT_FPGAMGR_IMGCFG_STAT_F2S_USERMODE_SET_MSK,
- 1, FPGA_TIMEOUT_MSEC, false);
- }
- static int is_fpgamgr_early_user_mode(void)
- {
- return (readl(&fpga_manager_base->imgcfg_stat) &
- ALT_FPGAMGR_IMGCFG_STAT_F2S_EARLY_USERMODE_SET_MSK) != 0;
- }
- int fpgamgr_wait_early_user_mode(void)
- {
- u32 sync_data = 0xffffffff;
- u32 i = 0;
- unsigned start = get_timer(0);
- unsigned long cd_ratio;
- /* Getting existing CDRATIO */
- cd_ratio = (readl(&fpga_manager_base->imgcfg_ctrl_02) &
- ALT_FPGAMGR_IMGCFG_CTL_02_CDRATIO_SET_MSK) >>
- ALT_FPGAMGR_IMGCFG_CTL_02_CDRATIO_LSB;
- /* Using CDRATIO_X1 for better compatibility */
- fpgamgr_set_cd_ratio(CDRATIO_x1);
- while (!is_fpgamgr_early_user_mode()) {
- if (get_timer(start) > FPGA_TIMEOUT_MSEC)
- return -ETIMEDOUT;
- fpgamgr_program_write((const long unsigned int *)&sync_data,
- sizeof(sync_data));
- udelay(FPGA_TIMEOUT_MSEC);
- i++;
- }
- debug("Additional %i sync word needed\n", i);
- /* restoring original CDRATIO */
- fpgamgr_set_cd_ratio(cd_ratio);
- return 0;
- }
- /* Read f2s_nconfig_pin and f2s_nstatus_pin; loop until de-asserted */
- static int wait_for_nconfig_pin_and_nstatus_pin(void)
- {
- unsigned long mask = ALT_FPGAMGR_IMGCFG_STAT_F2S_NCONFIG_PIN_SET_MSK |
- ALT_FPGAMGR_IMGCFG_STAT_F2S_NSTATUS_PIN_SET_MSK;
- /*
- * Poll until f2s_nconfig_pin and f2s_nstatus_pin; loop until
- * de-asserted, timeout at 1000ms
- */
- return wait_for_bit_le32(&fpga_manager_base->imgcfg_stat, mask,
- true, FPGA_TIMEOUT_MSEC, false);
- }
- static int wait_for_f2s_nstatus_pin(unsigned long value)
- {
- /* Poll until f2s to specific value, timeout at 1000ms */
- return wait_for_bit_le32(&fpga_manager_base->imgcfg_stat,
- ALT_FPGAMGR_IMGCFG_STAT_F2S_NSTATUS_PIN_SET_MSK,
- value, FPGA_TIMEOUT_MSEC, false);
- }
- /* set CD ratio */
- static void fpgamgr_set_cd_ratio(unsigned long ratio)
- {
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_CDRATIO_SET_MSK);
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- (ratio << ALT_FPGAMGR_IMGCFG_CTL_02_CDRATIO_LSB) &
- ALT_FPGAMGR_IMGCFG_CTL_02_CDRATIO_SET_MSK);
- }
- /* get the MSEL value, verify we are set for FPP configuration mode */
- static int fpgamgr_verify_msel(void)
- {
- u32 msel = fpgamgr_get_msel();
- if (msel & ~BIT(0)) {
- printf("Fail: read msel=%d\n", msel);
- return -EPERM;
- }
- return 0;
- }
- /*
- * Write cdratio and cdwidth based on whether the bitstream is compressed
- * and/or encoded
- */
- static int fpgamgr_set_cdratio_cdwidth(unsigned int cfg_width, u32 *rbf_data,
- size_t rbf_size)
- {
- unsigned int cd_ratio;
- bool encrypt, compress;
- /*
- * According to the bitstream specification,
- * both encryption and compression status are
- * in location before offset 230 of the buffer.
- */
- if (rbf_size < MIN_BITSTREAM_SIZECHECK)
- return -EINVAL;
- encrypt = (rbf_data[ENCRYPTION_OFFSET] >> 2) & 3;
- encrypt = encrypt != 0;
- compress = (rbf_data[COMPRESSION_OFFSET] >> 1) & 1;
- compress = !compress;
- debug("header word %d = %08x\n", 69, rbf_data[69]);
- debug("header word %d = %08x\n", 229, rbf_data[229]);
- debug("read from rbf header: encrypt=%d compress=%d\n", encrypt, compress);
- /*
- * from the register map description of cdratio in imgcfg_ctrl_02:
- * Normal Configuration : 32bit Passive Parallel
- * Partial Reconfiguration : 16bit Passive Parallel
- */
- /*
- * cd ratio is dependent on cfg width and whether the bitstream
- * is encrypted and/or compressed.
- *
- * | width | encr. | compr. | cd ratio |
- * | 16 | 0 | 0 | 1 |
- * | 16 | 0 | 1 | 4 |
- * | 16 | 1 | 0 | 2 |
- * | 16 | 1 | 1 | 4 |
- * | 32 | 0 | 0 | 1 |
- * | 32 | 0 | 1 | 8 |
- * | 32 | 1 | 0 | 4 |
- * | 32 | 1 | 1 | 8 |
- */
- if (!compress && !encrypt) {
- cd_ratio = CDRATIO_x1;
- } else {
- if (compress)
- cd_ratio = CDRATIO_x4;
- else
- cd_ratio = CDRATIO_x2;
- /* if 32 bit, double the cd ratio (so register
- field setting is incremented) */
- if (cfg_width == CFGWDTH_32)
- cd_ratio += 1;
- }
- fpgamgr_set_cfgwdth(cfg_width);
- fpgamgr_set_cd_ratio(cd_ratio);
- return 0;
- }
- static int fpgamgr_reset(void)
- {
- unsigned long reg;
- /* S2F_NCONFIG = 0 */
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NCONFIG_SET_MSK);
- /* Wait for f2s_nstatus == 0 */
- if (wait_for_f2s_nstatus_pin(0))
- return -ETIME;
- /* S2F_NCONFIG = 1 */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NCONFIG_SET_MSK);
- /* Wait for f2s_nstatus == 1 */
- if (wait_for_f2s_nstatus_pin(1))
- return -ETIME;
- /* read and confirm f2s_condone_pin = 0 and f2s_condone_oe = 1 */
- reg = readl(&fpga_manager_base->imgcfg_stat);
- if ((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_CONDONE_PIN_SET_MSK) != 0)
- return -EPERM;
- if ((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_CONDONE_OE_SET_MSK) == 0)
- return -EPERM;
- return 0;
- }
- /* Start the FPGA programming by initialize the FPGA Manager */
- int fpgamgr_program_init(u32 * rbf_data, size_t rbf_size)
- {
- int ret;
- /* Step 1 */
- if (fpgamgr_verify_msel())
- return -EPERM;
- /* Step 2 */
- if (fpgamgr_set_cdratio_cdwidth(CFGWDTH_32, rbf_data, rbf_size))
- return -EPERM;
- /*
- * Step 3:
- * Make sure no other external devices are trying to interfere with
- * programming:
- */
- if (wait_for_nconfig_pin_and_nstatus_pin())
- return -ETIME;
- /*
- * Step 4:
- * Deassert the signal drives from HPS
- *
- * S2F_NCE = 1
- * S2F_PR_REQUEST = 0
- * EN_CFG_CTRL = 0
- * EN_CFG_DATA = 0
- * S2F_NCONFIG = 1
- * S2F_NSTATUS_OE = 0
- * S2F_CONDONE_OE = 0
- */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_NCE_SET_MSK);
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_PR_REQUEST_SET_MSK);
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_DATA_SET_MSK |
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_CTRL_SET_MSK);
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NCONFIG_SET_MSK);
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NSTATUS_OE_SET_MSK |
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_CONDONE_OE_SET_MSK);
- /*
- * Step 5:
- * Enable overrides
- * S2F_NENABLE_CONFIG = 0
- * S2F_NENABLE_NCONFIG = 0
- */
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_NENABLE_CONFIG_SET_MSK);
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NENABLE_NCONFIG_SET_MSK);
- /*
- * Disable driving signals that HPS doesn't need to drive.
- * S2F_NENABLE_NSTATUS = 1
- * S2F_NENABLE_CONDONE = 1
- */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NENABLE_NSTATUS_SET_MSK |
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NENABLE_CONDONE_SET_MSK);
- /*
- * Step 6:
- * Drive chip select S2F_NCE = 0
- */
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_NCE_SET_MSK);
- /* Step 7 */
- if (wait_for_nconfig_pin_and_nstatus_pin())
- return -ETIME;
- /* Step 8 */
- ret = fpgamgr_reset();
- if (ret)
- return ret;
- /*
- * Step 9:
- * EN_CFG_CTRL and EN_CFG_DATA = 1
- */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_DATA_SET_MSK |
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_CTRL_SET_MSK);
- return 0;
- }
- /* Ensure the FPGA entering config done */
- static int fpgamgr_program_poll_cd(void)
- {
- unsigned long reg, i;
- for (i = 0; i < FPGA_TIMEOUT_CNT; i++) {
- reg = readl(&fpga_manager_base->imgcfg_stat);
- if (reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_CONDONE_PIN_SET_MSK)
- return 0;
- if ((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_NSTATUS_PIN_SET_MSK) == 0) {
- printf("nstatus == 0 while waiting for condone\n");
- return -EPERM;
- }
- }
- if (i == FPGA_TIMEOUT_CNT)
- return -ETIME;
- return 0;
- }
- /* Ensure the FPGA entering user mode */
- static int fpgamgr_program_poll_usermode(void)
- {
- unsigned long reg;
- int ret = 0;
- if (fpgamgr_dclkcnt_set(0xf))
- return -ETIME;
- ret = wait_for_user_mode();
- if (ret < 0) {
- printf("%s: Failed to enter user mode with ", __func__);
- printf("error code %d\n", ret);
- return ret;
- }
- /*
- * Step 14:
- * Stop DATA path and Dclk
- * EN_CFG_CTRL and EN_CFG_DATA = 0
- */
- clrbits_le32(&fpga_manager_base->imgcfg_ctrl_02,
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_DATA_SET_MSK |
- ALT_FPGAMGR_IMGCFG_CTL_02_EN_CFG_CTRL_SET_MSK);
- /*
- * Step 15:
- * Disable overrides
- * S2F_NENABLE_CONFIG = 1
- * S2F_NENABLE_NCONFIG = 1
- */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_NENABLE_CONFIG_SET_MSK);
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_00,
- ALT_FPGAMGR_IMGCFG_CTL_00_S2F_NENABLE_NCONFIG_SET_MSK);
- /* Disable chip select S2F_NCE = 1 */
- setbits_le32(&fpga_manager_base->imgcfg_ctrl_01,
- ALT_FPGAMGR_IMGCFG_CTL_01_S2F_NCE_SET_MSK);
- /*
- * Step 16:
- * Final check
- */
- reg = readl(&fpga_manager_base->imgcfg_stat);
- if (((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_USERMODE_SET_MSK) !=
- ALT_FPGAMGR_IMGCFG_STAT_F2S_USERMODE_SET_MSK) ||
- ((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_CONDONE_PIN_SET_MSK) !=
- ALT_FPGAMGR_IMGCFG_STAT_F2S_CONDONE_PIN_SET_MSK) ||
- ((reg & ALT_FPGAMGR_IMGCFG_STAT_F2S_NSTATUS_PIN_SET_MSK) !=
- ALT_FPGAMGR_IMGCFG_STAT_F2S_NSTATUS_PIN_SET_MSK))
- return -EPERM;
- return 0;
- }
- int fpgamgr_program_finish(void)
- {
- /* Ensure the FPGA entering config done */
- int status = fpgamgr_program_poll_cd();
- if (status) {
- printf("FPGA: Poll CD failed with error code %d\n", status);
- return -EPERM;
- }
- WATCHDOG_RESET();
- /* Ensure the FPGA entering user mode */
- status = fpgamgr_program_poll_usermode();
- if (status) {
- printf("FPGA: Poll usermode failed with error code %d\n",
- status);
- return -EPERM;
- }
- printf("Full Configuration Succeeded.\n");
- return 0;
- }
- /*
- * FPGA Manager to program the FPGA. This is the interface used by FPGA driver.
- * Return 0 for sucess, non-zero for error.
- */
- int socfpga_load(Altera_desc *desc, const void *rbf_data, size_t rbf_size)
- {
- int status;
- /* disable all signals from hps peripheral controller to fpga */
- writel(0, &system_manager_base->fpgaintf_en_global);
- /* disable all axi bridge (hps2fpga, lwhps2fpga & fpga2hps) */
- socfpga_bridges_reset();
- /* Initialize the FPGA Manager */
- status = fpgamgr_program_init((u32 *)rbf_data, rbf_size);
- if (status)
- return status;
- /* Write the RBF data to FPGA Manager */
- fpgamgr_program_write(rbf_data, rbf_size);
- return fpgamgr_program_finish();
- }
|