123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499 |
- /*
- * Copyright 2013 Freescale Semiconductor, Inc.
- *
- * See file CREDITS for list of people who contributed to this
- * project.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of
- * the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that 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, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
- * MA 02111-1307 USA
- */
- #include <common.h>
- #include <asm/io.h>
- #include <command.h>
- #include <asm/imx-common/iomux-v3.h>
- #include <asm/arch-vf610/iomux-vf610.h>
- #include <asm/arch-vf610/crm_regs.h>
- #include <asm/arch/vf610_qspi.h>
- void quadspi_init(void);
- void quadspi_erase(void);
- void quadspi_erase_sector(unsigned int addr);
- void quadspi_program(unsigned int src, unsigned int base, unsigned int size);
- static void quadspi_setup_iomux(void);
- static void quadspi_setup_clocks(void);
- static void quadspi_config(void);
- static void quadspi_setup_lut(void);
- static void quadspi_wait_while_flash_busy(void);
- static void quadspi_enable_quadbit(void);
- static volatile struct quadspi *qspi = (struct quadspi *)QSPI0_BASE_ADDR;
- static int qspidev = 0;
- static int qspibase = QSPI0_FLASH_BASE_ADDR;
- static void quadspi_setup_iomux(void)
- {
- if (qspidev == 0) {
- imx_iomux_v3_setup_pad(VF610_PAD_PTD0__QSCKA);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD1__QCS0A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD2__QIO3A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD3__QIO2A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD4__QIO1A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD5__QIO0A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD7__QSCKB);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD8__QCS0B);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD9__QIO3B);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD10__QIO2B);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD11__QIO1B);
- imx_iomux_v3_setup_pad(VF610_PAD_PTD12__QIO0B);
- } else {
- imx_iomux_v3_setup_pad(VF610_PAD_PTA19__QSCKA);
- imx_iomux_v3_setup_pad(VF610_PAD_PTB0__QCS0A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTB1__QIO3A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTB2__QIO2A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTB3__QIO1A);
- imx_iomux_v3_setup_pad(VF610_PAD_PTB4__QIO0A);
- }
- }
- static void quadspi_setup_clocks(void)
- {
- struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
- debug ("Setup QuadSPI clocks\n");
- if (qspidev == 0) /* Ungate QSPI0 */
- ccm->ccgr2 |= 0x1 << 8;
- else /* Ungate QSPI1 */
- ccm->ccgr8 |= 0x1 << 8;
- /* Select pll1-pfd4 as source for both Qspi0 and Qsp1 */
- if (qspidev == 0)
- ccm->cscmr1 |= (0x3 << 22);
- else
- ccm->cscmr1 |= (0x3 << 24);
- /* QSPI Enable, QSPI_DIV(1),
- * QSPI_X2_DIV(2), QSPI_X4_DIV(4) = 33MHZ
- */
- if (qspidev == 0)
- ccm->cscdr3 |= 0x1F;
- else
- ccm->cscdr3 |= (0x1F << 8);
- }
- static void quadspi_config(void)
- {
- debug ("Config QuadSPI0\n");
- /* Support both SDR and DDR instructions;
- * Enable Qspi module
- */
- qspi->mcr &= ~0x4000;
- qspi->mcr |= 0x0080;
- qspi->buf0ind = 0x0;
- /* top address of FA1 */
- if (!qspi->sfa1ad)
- qspi->sfa1ad = (qspidev ? QSPI1_TOP_FLASH_A1_ADDR
- : QSPI0_TOP_FLASH_A1_ADDR);
- /* top address of FA2 */
- if (!qspi->sfa2ad)
- qspi->sfa2ad = qspi->sfa1ad;
- /* top address of FB1 */
- if (!qspi->sfb1ad)
- qspi->sfb1ad = (qspidev ? QSPI1_TOP_FLASH_B1_ADDR
- : QSPI0_TOP_FLASH_B1_ADDR);
- /* top address of FB2 */
- if (!qspi->sfb2ad)
- qspi->sfb2ad = qspi->sfb1ad;
- }
- static void quadspi_setup_lut (void)
- {
- debug ("quadspi_setup_lut\n");
- /* Unlock LUT */
- qspi->lutkey = 0x5af05af0;
- qspi->lckcr = 0x2;
- /* seqid 0 - quad read */
- qspi->lut[0] = 0x0a1804eb; /* quad read, 24 bit addresses */
- qspi->lut[1] = 0x0e0412a5; /* mode bits and 4 dummy cycles */
- qspi->lut[2] = 0x24011e80; /* read 128 bytes and jump to address */
- /* seqid 1 - write enable */
- qspi->lut[4] = 0x406;
- /* seqid 2 - bulk erase */
- qspi->lut[8] = 0x460;
- /* seqid 3 - read status */
- qspi->lut[12] = 0x1c010405;
- /* seqid 4 - page program */
- qspi->lut[16] = 0x08180402; /* 24bit address */
- qspi->lut[17] = 0x2004; /* default 4-byte write */
- /* seqid 5 - write config/status */
- qspi->lut[20] = 0x20020401; /*2-byte write */
- /* seqid 6 - read config */
- qspi->lut[24] = 0x1c010435; /*1-byte read */
- /* seqid 7 - sector erase */
- qspi->lut[28] = 0x081804d8;
- /* seqid 8 - quad ddr read */
- /* quad ddr read - 24 bit addresses */
- qspi->lut[32] = QuadSPI_LUT_INSTR0(CMD)|
- QuadSPI_LUT_PAD0(0x0)|
- QuadSPI_LUT_OPRND0(0xED)|
- QuadSPI_LUT_INSTR1(ADDR_DDR)|
- QuadSPI_LUT_PAD1(0x2) |
- QuadSPI_LUT_OPRND1(0x18);
- /*set mode and 6 dummy cycles */
- qspi->lut[33] = QuadSPI_LUT_INSTR0(MODE_DDR) |
- QuadSPI_LUT_PAD0(0x2) |
- QuadSPI_LUT_OPRND0(0xa5) |
- QuadSPI_LUT_INSTR1(DUMMY) |
- QuadSPI_LUT_PAD1(0x2) |
- QuadSPI_LUT_OPRND1(0x6);
- /* read 128 bytes */
- qspi->lut[34] = QuadSPI_LUT_INSTR0(READ_DDR) |
- QuadSPI_LUT_PAD0(0x2) |
- QuadSPI_LUT_OPRND0(0x80) |
- QuadSPI_LUT_INSTR1(JMP_ON_CS) |
- QuadSPI_LUT_PAD1(0x0) |
- QuadSPI_LUT_OPRND1(0x1);
- /* seqid 9 - ddr read - 24 bit addresses */
- qspi->lut[36] = QuadSPI_LUT_INSTR0(CMD) |
- QuadSPI_LUT_PAD0(0x0) |
- QuadSPI_LUT_OPRND0(0x0D) |
- QuadSPI_LUT_INSTR1(ADDR_DDR) |
- QuadSPI_LUT_PAD1(0x0) |
- QuadSPI_LUT_OPRND1(0x18);
- /*mode bits and 2 dummy cycles */
- qspi->lut[37] = QuadSPI_LUT_INSTR0(MODE_DDR) |
- QuadSPI_LUT_PAD0(0x0) |
- QuadSPI_LUT_OPRND0(0xFF) |
- QuadSPI_LUT_INSTR1(DUMMY)|
- QuadSPI_LUT_PAD1(0x0) |
- QuadSPI_LUT_OPRND1(0x2);
- qspi->lut[38] = 0x3880;
- /* lock lut */
- qspi->lutkey = 0x5af05af0;
- qspi->lckcr = 0x1;
- }
- static void quadspi_wait_while_flash_busy(void)
- {
- u32 status_value = 0x1;
- while ((status_value & 0x1)==0x1)
- {
- qspi->ipcr = 3 << 24;
- while(qspi->sr & 0x1);
- while(!(qspi->sr & (1 << 16)));
- status_value = QSPI_ARDB;
- /* read complete */
- qspi->fr = 0x10000;
- }
- }
- void quadspi_init(void)
- {
- quadspi_setup_clocks();
- /* port configuration */
- quadspi_setup_iomux();
- quadspi_config();
- quadspi_setup_lut();
- /* set quad ddr as xbar read instruction */
- qspi->bfgencr = 0x9000;
- /* set mdis bit */
- qspi->mcr = qspi->mcr | 0x4000;
- /* for 33MHz clock */
- qspi->smpr = 0x10000;
- /* clear mdis bit */
- qspi->mcr &= ~0x4000;
- /* clear tx fifo */
- qspi->mcr |= 0x800;
- qspi->sfar = (qspidev ? QSPI1_FLASH_BASE_ADDR : QSPI0_FLASH_BASE_ADDR);
- quadspi_enable_quadbit();
- qspi->sfar = (qspidev ? QSPI1_FLASH_BASE_ADDR : QSPI0_FLASH_BASE_ADDR);
- }
- static void quadspi_enable_quadbit(void)
- {
- /* write enable */
- qspi->ipcr = 1 << 24;
- while(qspi->sr & 0x1);
- /* write data to Tx Buffer */
- /* enable flash quad mode */
- qspi->tbdr = 0x00020000;
- /* send write command */
- qspi->ipcr = 5 << 24;
- while(qspi->sr & 0x1);
- quadspi_wait_while_flash_busy();
- /* Read config reg to ensure write was successful */
- qspi->ipcr = 6 << 24;
- while(qspi->sr & 0x1);
- /* read complete */
- qspi->fr = 0x10000;
- }
- void quadspi_erase(void)
- {
- qspi->sfar = qspibase;
- /*write enable*/
- qspi->ipcr = 1 << 24;
- while(qspi->sr & 0x1);
- /*send erase command */
- qspi->ipcr = 2 << 24;
- while(qspi->sr & 0x1);
- quadspi_wait_while_flash_busy();
- }
- void quadspi_erase_sector(unsigned int addr)
- {
- debug ("Erasing QuadSPI flash addr=0x%x\n",addr);
- qspi->sfar = addr;
- /*write enable */
- qspi->ipcr = 1 << 24;
- while(qspi->sr & 0x1);
- /*send erase sector command */
- qspi->ipcr = 7 << 24;
- while(qspi->sr & 0x1);
- quadspi_wait_while_flash_busy();
- }
- void quadspi_program(unsigned int src, unsigned int base, unsigned int size)
- {
- unsigned int *start_address = (unsigned int *)src;
- unsigned int *end_address = (unsigned int *)(src + size);
- unsigned int *page_address = start_address;
- unsigned int *flash_address = (unsigned int *)base;
- int page_size = QSPI_FLASH_PGSZ;
- unsigned int *current_address = start_address;
- unsigned int data_value;
- int i;
- /* 1024 offset for spansion */
- qspi->sfar = (unsigned int)flash_address;
- /* clear Tx fifo */
- qspi->mcr |= 0x800;
- page_address = start_address + (page_size >> 2);
- do {
- if ((unsigned int)flash_address % 0x10000 == 0)
- quadspi_erase_sector((unsigned int)flash_address);
- /* write enable */
- qspi->ipcr = 1 << 24;
- while(qspi->sr & 0x1);
- /* clear Tx fifo */
- qspi->mcr |= 0x800;
- while (current_address < page_address)
- {
- /* fill tx fifo (64 bytes) */
- for (i = 0; i < 16; i++)
- {
- data_value=*(current_address++);
- qspi->tbdr = __swap_32(data_value);
- }
- /*page program 256bytes */
- qspi->ipcr = (4 << 24) | page_size;
- for (i = 0; i < 48; i++)
- {
- /* while TX fifo Full */
- while(qspi->sr & 0x8000000u);
- data_value=*(current_address++);
- qspi->tbdr = __swap_32(data_value);
- }
- while(qspi->sr & 0x1);
- quadspi_wait_while_flash_busy();
- }
- page_address += (page_size >> 2);
- flash_address += (page_size >> 2);
- qspi->sfar = (unsigned int)flash_address;
- } while (current_address < end_address);
- }
- int quadspi_write_cmd(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
- {
- ulong length = 1;
- ulong dest, src;
- if (argc < 4)
- return CMD_RET_USAGE;
- /* Address is specified since argc > 1 */
- dest = simple_strtoul(argv[1], NULL, 16);
- src = simple_strtoul(argv[2], NULL, 16);
- length = simple_strtoul(argv[3], NULL, 16);
- if (!src || !length)
- return CMD_RET_USAGE;
- /* Check if in the QSPI address range */
- if (!((dest >= qspibase) && (dest + length <= qspi->sfb2ad)))
- return CMD_RET_USAGE;
- quadspi_program(src, dest, length);
- print_buffer(dest, (void*)dest, 2, 0x10, 8);
- return 0;
- }
- U_BOOT_CMD(
- qspiwrite, 4, 0, quadspi_write_cmd,
- "Write/programm the QSPI memory",
- "qspiwrite destAddr srcAddr length\n"
- " destAddr: is an address in the QSPI domain\n"
- " srcAddr: is the source address\n"
- " length: size in bytes\n"
- " Example:\n"
- " qspiwrite 0x20000000 0x3f800000 0x1000\n"
- ""
- );
- int quadspi_init_cmd(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
- {
- unsigned int addr;
- int i;
- if (argc < 2)
- return CMD_RET_USAGE;
- /* dev= must be first argument */
- if (!strncmp(argv[1], "dev=", 4)) {
- qspidev = simple_strtoul(argv[1] + 4, NULL, 16);
- switch (qspidev) {
- case 0:
- qspi = (struct quadspi *)QSPI0_BASE_ADDR;
- qspibase = QSPI0_FLASH_BASE_ADDR;
- break;
- case 1:
- qspi = (struct quadspi *)QSPI1_BASE_ADDR;
- qspibase = QSPI1_FLASH_BASE_ADDR;
- break;
- default:
- return CMD_RET_USAGE;
- }
- } else
- return CMD_RET_USAGE;
- for (i = 2; i < argc; i++) {
- addr = simple_strtoul(argv[i] + 4, NULL, 16);
- if (!strncmp(argv[i], "fa1=", 4)){
- qspi->sfa1ad = addr;
- /* assume there is no FA2 */
- qspi->sfa2ad = addr;
- /* assume there is no FB1 */
- qspi->sfb1ad = addr;
- } else if (!strncmp(argv[i], "fa2=", 4)) {
- /* FA2 can only be set if FA1 is already set */
- if (!qspi->sfa1ad)
- return CMD_RET_USAGE;
- qspi->sfa2ad = addr;
- /* assume there is no FB1 */
- qspi->sfb1ad = addr;
- } else if (!strncmp(argv[i], "fb1=", 4)) {
- /* FB1 can only be set if FA is already set */
- if (!qspi->sfa1ad)
- return CMD_RET_USAGE;
- qspi->sfb1ad = addr;
- /* assume there is no FB2 */
- qspi->sfb2ad = addr;
- } else if (!strncmp(argv[i], "fb2=", 4)) {
- if (!qspi->sfb1ad || !qspi->sfa1ad)
- return CMD_RET_USAGE;
- qspi->sfb2ad = addr;
- } else
- return CMD_RET_USAGE;
- }
- debug ("Init QuadSPI dev %d\n", qspidev);
- quadspi_init();
- return 0;
- }
- U_BOOT_CMD(
- qspiinit, 4, 0, quadspi_init_cmd,
- "initialize the QSPI",
- "qspiiint dev=X <fa1=A1> <fa2=A2> <fb1=B1> <fb2=B2>\n"
- " dev= set 0 for qspi0 and 1 for qsp1\n"
- " fa1= is the address of the top flash A1 device\n"
- " fa2= is the address of the top flash A2 device\n"
- " fb1= is the address of the top flash B1 device\n"
- " fb2= is the address of the top flash B2 device\n"
- " Note that fa1, fa2, fb1 and fb2 are optional but they must be listed\n "
- " in that order, you can not list fb1 before fa1 for instance\n"
- " Example1:\n"
- " Initialize QSPI0 using default values\n"
- " qspiinit dev=0\n"
- " Example2:\n"
- " Set FA1 and FB1\n"
- " qspiinit dev=0 fa1=0x21000000 fb1=0x22000000\n"
- ""
- );
- int quadspi_erase_cmd(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
- {
- ulong addr;
- if (argc < 2)
- return CMD_RET_USAGE;
- addr = simple_strtoul(argv[1], NULL, 16);
- quadspi_erase_sector(addr);
- return 0;
- }
- U_BOOT_CMD(
- qspierase, 2, 0, quadspi_erase_cmd,
- "Erase whole QSPI memory",
- "qspierase addr\n"
- " Example:\n"
- " qspierase 0x20040000\n"
- ""
- );
|