]> git.ipfire.org Git - thirdparty/u-boot.git/commitdiff
fpga: zynq: Added support to load bit stream from SD/MMC
authorSiva Durga Prasad Paladugu <siva.durga.paladugu@xilinx.com>
Fri, 14 Mar 2014 11:05:37 +0000 (16:35 +0530)
committerMichal Simek <michal.simek@xilinx.com>
Mon, 17 Mar 2014 13:34:11 +0000 (14:34 +0100)
Added support to load a bitstream image in chunks by reading it in
chunks from SD/MMC.
Command format:
loadfs [dev] [address] [image size] [blocksize] <interface>
       [<dev[:part]>] <filename>
Example: fpga loadfs 0 1000000 3dbafc 4000 mmc 0 fpga.bin

Signed-off-by: Siva Durga Prasad Paladugu <sivadur@xilinx.com>
Signed-off-by: Michal Simek <michal.simek@xilinx.com>
common/cmd_fpga.c
drivers/fpga/fpga.c
drivers/fpga/xilinx.c
drivers/fpga/zynqpl.c
include/fpga.h
include/xilinx.h
include/zynqpl.h

index 010cd24e63dc21f86f0ae1910a4c4fac9d0e20e4..4d5caa041e70baf43c825d41e20b8706f3231a4f 100644 (file)
@@ -12,6 +12,9 @@
 #include <command.h>
 #include <fpga.h>
 #include <malloc.h>
+#ifdef CONFIG_FPGA_LOADFS
+#include <fs.h>
+#endif
 
 /* Local functions */
 static int fpga_get_op(char *opstr);
@@ -23,6 +26,7 @@ static int fpga_get_op(char *opstr);
 #define FPGA_LOADB  2
 #define FPGA_DUMP   3
 #define FPGA_LOADMK 4
+#define FPGA_LOADFS 5
 
 /* ------------------------------------------------------------------------- */
 /* command form:
@@ -45,6 +49,11 @@ int do_fpga(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
        const char *fit_uname = NULL;
        ulong fit_addr;
 #endif
+#ifdef CONFIG_FPGA_LOADFS
+       fpga_fs_info fpga_fsinfo;
+
+       fpga_fsinfo.fstype = FS_TYPE_ANY;
+#endif
 
        if (devstr)
                dev = (int) simple_strtoul(devstr, NULL, 16);
@@ -52,6 +61,17 @@ int do_fpga(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
                fpga_data = (void *)simple_strtoul(datastr, NULL, 16);
 
        switch (argc) {
+#ifdef CONFIG_FPGA_LOADFS
+       case 9:
+               fpga_fsinfo.blocksize = (unsigned int)
+                                            simple_strtoul(argv[5], NULL, 16);
+               fpga_fsinfo.interface = argv[6];
+               fpga_fsinfo.dev_part = argv[7];
+               fpga_fsinfo.filename = argv[8];
+
+               if (fpga_fsinfo.fstype == FS_TYPE_ANY)
+                       fpga_fsinfo.fstype = FS_TYPE_FAT;
+#endif
        case 5:         /* fpga <op> <dev> <data> <datasize> */
                data_size = simple_strtoul(argv[4], NULL, 16);
 
@@ -120,6 +140,13 @@ int do_fpga(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
        case FPGA_NONE:
        case FPGA_INFO:
                break;
+#ifdef CONFIG_FPGA_LOADFS
+       case FPGA_LOADFS:
+               /* Blocksize can be zero */
+               if (!fpga_fsinfo.interface || !fpga_fsinfo.dev_part ||
+                   !fpga_fsinfo.filename || fpga_fsinfo.fstype == FS_TYPE_ANY)
+                       wrong_parms = 1;
+#endif
        case FPGA_LOAD:
        case FPGA_LOADB:
        case FPGA_DUMP:
@@ -152,7 +179,11 @@ int do_fpga(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
        case FPGA_LOADB:
                rc = fpga_loadbitstream(dev, fpga_data, data_size);
                break;
-
+#ifdef CONFIG_FPGA_LOADFS
+       case FPGA_LOADFS:
+               rc = fpga_fsload(dev, fpga_data, data_size, &fpga_fsinfo);
+               break;
+#endif
        case FPGA_LOADMK:
                switch (genimg_get_format(fpga_data)) {
                case IMAGE_FORMAT_LEGACY:
@@ -257,6 +288,10 @@ static int fpga_get_op(char *opstr)
                op = FPGA_LOADB;
        else if (!strcmp("load", opstr))
                op = FPGA_LOAD;
+#ifdef CONFIG_FPGA_LOADFS
+       else if (!strcmp("loadfs", opstr))
+               op = FPGA_LOADFS;
+#endif
        else if (!strcmp("loadmk", opstr))
                op = FPGA_LOADMK;
        else if (!strcmp("dump", opstr))
@@ -267,8 +302,11 @@ static int fpga_get_op(char *opstr)
 
        return op;
 }
-
+#ifdef CONFIG_FPGA_LOADFS
+U_BOOT_CMD(fpga, 9, 1, do_fpga,
+#else
 U_BOOT_CMD(fpga, 6, 1, do_fpga,
+#endif
           "loadable FPGA image support",
           "[operation type] [device number] [image address] [image size]\n"
           "fpga operations:\n"
@@ -281,6 +319,11 @@ U_BOOT_CMD(fpga, 6, 1, do_fpga,
 #if defined(CONFIG_FIT)
           "\n"
           "\tFor loadmk operating on FIT format uImage address must include\n"
-          "\tsubimage unit name in the form of addr:<subimg_uname>"
+          "\tsubimage unit name in the form of addr:<subimg_uname>\n"
+#endif
+#ifdef CONFIG_FPGA_LOADFS
+          "Load device from filesystem (FAT by default) (Xilinx only)\n"
+          "  loadfs [dev] [address] [image size] [blocksize] <interface>\n"
+          "        [<dev[:part]>] <filename>\n"
 #endif
 );
index 2ed2f959a1e04b61752bf2e0dbcd8f97bbf31e99..1f61e7d10dcc33d5c18ed6ad53f051e1f50da802 100644 (file)
@@ -179,6 +179,34 @@ int __weak fpga_loadbitstream(int devnum, char *fpgadata, size_t size)
        return FPGA_FAIL;
 }
 
+#ifdef CONFIG_FPGA_LOADFS
+int fpga_fsload(int devnum, const void *buf, size_t size,
+                fpga_fs_info *fpga_fsinfo)
+{
+       int ret_val = FPGA_FAIL;           /* assume failure */
+       const fpga_desc *desc = fpga_validate(devnum, buf, size,
+                                             (char *)__func__);
+
+       if (desc) {
+               switch (desc->devtype) {
+               case fpga_xilinx:
+#if defined(CONFIG_FPGA_XILINX)
+                       ret_val = xilinx_fsload(desc->devdesc, buf, size,
+                                                fpga_fsinfo);
+#else
+                       fpga_no_sup((char *)__func__, "Xilinx devices");
+#endif
+                       break;
+               default:
+                       printf("%s: Invalid or unsupported device type %d\n",
+                              __func__, desc->devtype);
+               }
+       }
+
+       return ret_val;
+}
+#endif
+
 /*
  * Generic multiplexing code
  */
index 2e0db535d474ca74c5f42d8072228a66f9b41cd5..8e9b1a2324d2f82b91f40a7fb11a6b60b8ce936c 100644 (file)
@@ -141,6 +141,37 @@ int fpga_loadbitstream(int devnum, char *fpgadata, size_t size)
        return fpga_load(devnum, dataptr, swapsize);
 }
 
+#ifdef CONFIG_FPGA_LOADFS
+int xilinx_fsload(Xilinx_desc *desc, const void *buf, size_t bsize,
+                  fpga_fs_info *fpga_fsinfo)
+{
+       int ret_val = FPGA_FAIL;        /* assume a failure */
+
+       if (!xilinx_validate(desc, (char *)__func__)) {
+               printf("%s: Invalid device descriptor\n", __func__);
+       } else {
+               switch (desc->family) {
+               case xilinx_zynq:
+#if defined(CONFIG_FPGA_ZYNQPL)
+                       PRINTF("%s: Launching the Zynq PL Loader...\n",
+                              __func__);
+                       ret_val = zynq_fsload(desc, buf, bsize,
+                                              fpga_fsinfo);
+#else
+                       printf("%s: No support for Zynq devices.\n",
+                              __func__);
+#endif
+                       break;
+
+               default:
+                       printf("%s: Unsupported family type, %d\n",
+                              __func__, desc->family);
+               }
+       }
+       return ret_val;
+}
+#endif
+
 int xilinx_load(Xilinx_desc *desc, const void *buf, size_t bsize)
 {
        int ret_val = FPGA_FAIL;        /* assume a failure */
index e53ab7641c00be1e773c2786b85ad709d6dd2b63..3572bc9c1fa1917a43b1a8460c44c5586b7557d2 100644 (file)
@@ -13,6 +13,9 @@
 #include <asm/sizes.h>
 #include <asm/arch/hardware.h>
 #include <asm/arch/sys_proto.h>
+#if defined(CONFIG_FPGA_LOADFS) && !defined(CONFIG_SPL_BUILD)
+#include <fs.h>
+#endif
 
 #define DEVCFG_CTRL_PCFG_PROG_B                0x40000000
 #define DEVCFG_CTRL_PCAP_RATE_EN_MASK  0x02000000
@@ -358,6 +361,87 @@ static int zynq_validate_bitstream(Xilinx_desc *desc, const void *buf,
        return 0;
 }
 
+#if defined(CONFIG_FPGA_LOADFS) && !defined(CONFIG_SPL_BUILD)
+int zynq_fsload(Xilinx_desc *desc, const void *buf, size_t bsize,
+                fpga_fs_info *fsinfo)
+{
+       unsigned long ts; /* Timestamp */
+       u32 isr_status, swap;
+       u32 partialbit = 0;
+       u32 blocksize;
+       u32 pos = 0;
+       int fstype;
+       char *interface, *dev_part, *filename;
+
+       blocksize = fsinfo->blocksize;
+       interface = fsinfo->interface;
+       dev_part = fsinfo->dev_part;
+       filename = fsinfo->filename;
+       fstype = fsinfo->fstype;
+
+       if (fs_set_blk_dev(interface, dev_part, fstype))
+               return FPGA_FAIL;
+
+       if (fs_read(filename, (u32) buf, pos, blocksize) < 0)
+               return FPGA_FAIL;
+
+       if (zynq_validate_bitstream(desc, buf, bsize, blocksize, &swap,
+                                   &partialbit))
+               return FPGA_FAIL;
+
+       dcache_disable();
+
+       do {
+               buf = zynq_align_dma_buffer((u32 *)buf, blocksize, swap);
+
+               if (zynq_dma_transfer((u32)buf | 1, blocksize >> 2,
+                                     0xffffffff, 0))
+                       return FPGA_FAIL;
+
+               bsize -= blocksize;
+               pos   += blocksize;
+
+               if (fs_set_blk_dev(interface, dev_part, FS_TYPE_FAT))
+                       return FPGA_FAIL;
+
+               if (bsize > blocksize) {
+                       if (fs_read(filename, (u32) buf, pos, blocksize) < 0)
+                               return FPGA_FAIL;
+               } else {
+                       if (fs_read(filename, (u32) buf, pos, bsize) < 0)
+                               return FPGA_FAIL;
+               }
+       } while (bsize > blocksize);
+
+       buf = zynq_align_dma_buffer((u32 *)buf, blocksize, swap);
+
+       if (zynq_dma_transfer((u32)buf | 1, bsize >> 2, 0xffffffff, 0))
+               return FPGA_FAIL;
+
+       dcache_enable();
+
+       isr_status = readl(&devcfg_base->int_sts);
+
+       /* Check FPGA configuration completion */
+       ts = get_timer(0);
+       while (!(isr_status & DEVCFG_ISR_PCFG_DONE)) {
+               if (get_timer(ts) > CONFIG_SYS_FPGA_WAIT) {
+                       printf("%s: Timeout wait for FPGA to config\n",
+                              __func__);
+                       return FPGA_FAIL;
+               }
+               isr_status = readl(&devcfg_base->int_sts);
+       }
+
+       debug("%s: FPGA config done\n", __func__);
+
+       if (!partialbit)
+               zynq_slcr_devcfg_enable();
+
+       return FPGA_SUCCESS;
+}
+#endif
+
 int zynq_load(Xilinx_desc *desc, const void *buf, size_t bsize)
 {
        unsigned long ts; /* Timestamp */
index acc9efc883b2fe5cc03e616935dd16db41aaa334..615d453963e8b16f06d9f55a5d2b9f2cde575c3f 100644 (file)
@@ -35,6 +35,15 @@ typedef struct {             /* typedef fpga_desc */
        void *devdesc;          /* real device descriptor */
 } fpga_desc;                   /* end, typedef fpga_desc */
 
+#ifdef CONFIG_FPGA_LOADFS
+typedef struct {                /* typedef fpga_desc */
+       unsigned int blocksize;
+       char *interface;
+       char *dev_part;
+       char *filename;
+       int fstype;
+} fpga_fs_info;
+#endif
 
 /* root function definitions */
 extern void fpga_init(void);
@@ -42,6 +51,10 @@ extern int fpga_add(fpga_type devtype, void *desc);
 extern int fpga_count(void);
 extern const fpga_desc *const fpga_get_desc(int devnum);
 extern int fpga_load(int devnum, const void *buf, size_t bsize);
+#ifdef CONFIG_FPGA_LOADFS
+extern int fpga_fsload(int devnum, const void *buf, size_t size,
+                       fpga_fs_info *fpga_fsinfo);
+#endif
 extern int fpga_loadbitstream(int devnum, char *fpgadata, size_t size);
 extern int fpga_dump(int devnum, const void *buf, size_t bsize);
 extern int fpga_info(int devnum);
index 00a585e5fc311650e7d981c507cd3310c2948d1e..2d8f0e225c8487e8fe0864bbce363ed8d99e3dd9 100644 (file)
@@ -46,6 +46,10 @@ typedef struct {             /* typedef Xilinx_desc */
 /* Generic Xilinx Functions
  *********************************************************************/
 extern int xilinx_load(Xilinx_desc *desc, const void *image, size_t size);
+#ifdef CONFIG_FPGA_LOADFS
+extern int xilinx_fsload(Xilinx_desc *desc, const void *buf, size_t bsize,
+                         fpga_fs_info *fpga_fsinfo);
+#endif
 extern int xilinx_dump(Xilinx_desc *desc, const void *buf, size_t bsize);
 extern int xilinx_info(Xilinx_desc *desc);
 
index 11806fdc39e163f9e46907fcbd5a8f1185608d94..224c841b550f0b92b22eeff7b22f93c04b9f0a8f 100644 (file)
 #include <xilinx.h>
 
 extern int zynq_load(Xilinx_desc *desc, const void *image, size_t size);
+#ifdef CONFIG_FPGA_LOADFS
+extern int zynq_fsload(Xilinx_desc *desc, const void *buf, size_t bsize,
+                       fpga_fs_info *fsinfo);
+#endif
 extern int zynq_dump(Xilinx_desc *desc, const void *buf, size_t bsize);
 extern int zynq_info(Xilinx_desc *desc);
 #ifdef CONFIG_CMD_ZYNQ_AES