|  | /* | 
|  | * (C) Copyright 2002 | 
|  | * MAZeT GmbH <www.mazet.de> | 
|  | * Stephan Linz <linz@mazet.de>, <linz@li-pro.net> | 
|  | * | 
|  | * The most stuff comes from PPCBoot and Linux. | 
|  | * | 
|  | * IMMS gGmbH <www.imms.de> | 
|  | * Thomas Elste <info@elste.org> | 
|  | * | 
|  | * Modifications for ModNET50 Board | 
|  | * | 
|  | * 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/arch/netarm_registers.h> | 
|  |  | 
|  | #define SCR   (*(volatile unsigned int *)(NETARM_GEN_MODULE_BASE + NETARM_GEN_SYSTEM_CONTROL)) | 
|  |  | 
|  | #define ALIGN_ABORT_OFF		SCR = SCR & ~NETARM_GEN_SYS_CFG_ALIGN_ABORT | 
|  | #define ALIGN_ABORT_ON		SCR = SCR |  NETARM_GEN_SYS_CFG_ALIGN_ABORT | 
|  |  | 
|  | #define PROG_ADDR		(0x555*2) | 
|  | #define SETUP_ADDR		(0x555*2) | 
|  | #define ID_ADDR			(0x555*2) | 
|  | #define UNLOCK_ADDR1		(0x555*2) | 
|  | #define UNLOCK_ADDR2		(0x2AA*2) | 
|  |  | 
|  | #define UNLOCK_CMD1		(0xAA) | 
|  | #define UNLOCK_CMD2		(0x55) | 
|  | #define ERASE_SUSPEND_CMD	(0xB0) | 
|  | #define ERASE_RESUME_CMD	(0x30) | 
|  | #define RESET_CMD		(0xF0) | 
|  | #define ID_CMD			(0x90) | 
|  | #define SECERASE_CMD		(0x30) | 
|  | #define CHIPERASE_CMD		(0x10) | 
|  | #define PROG_CMD		(0xa0) | 
|  | #define SETUP_CMD		(0x80) | 
|  |  | 
|  | #define DQ2			(0x04) | 
|  | #define DQ3			(DQ2*2) | 
|  | #define DQ5			(DQ3*4) | 
|  | #define DQ6			(DQ5*2) | 
|  |  | 
|  | #define WRITE_UNLOCK(addr) { \ | 
|  | *(volatile __u16*)(addr + UNLOCK_ADDR1) = (__u16)UNLOCK_CMD1; \ | 
|  | *(volatile __u16*)(addr + UNLOCK_ADDR2) = (__u16)UNLOCK_CMD2; \ | 
|  | } | 
|  |  | 
|  | #define CONFIG_AM29_RESERVED	(0) | 
|  | #define K			(1024) | 
|  | #define MB			(4) | 
|  |  | 
|  | #define CELL_SIZE		(64*K) | 
|  | #define DEVICE_SIZE		(MB*K*K) | 
|  | #define CELLS_PER_DEVICE	(DEVICE_SIZE/CELL_SIZE) | 
|  | #define RESERVED_CELLS		(CONFIG_AM29_RESERVED*K)/CELL_SIZE | 
|  | #define MAX_FLASH_DEVICES	(1) | 
|  | #define AVAIL_SIZE		(DEVICE_SIZE*MAX_FLASH_DEVICES - RESERVED_CELLS*CELL_SIZE) | 
|  |  | 
|  |  | 
|  | flash_info_t flash_info[CONFIG_SYS_MAX_FLASH_BANKS]; | 
|  | static __u16 toggling_bits; | 
|  |  | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | ulong flash_get_size (ulong baseaddr, flash_info_t * info) | 
|  | { | 
|  | short i; | 
|  | __u16 flashtest; | 
|  |  | 
|  | /* Write auto select command sequence and test FLASH answer */ | 
|  | WRITE_UNLOCK (baseaddr); | 
|  | *(volatile __u16 *) (baseaddr + ID_ADDR) = (__u16) ID_CMD; | 
|  | flashtest /* manufacturer ID */  = *(volatile __u16 *) (baseaddr); | 
|  | *(volatile __u16 *) (baseaddr + ID_ADDR) = (__u16) RESET_CMD; | 
|  |  | 
|  | switch ((__u32) ((flashtest << 16) + flashtest)) { | 
|  | case AMD_MANUFACT: | 
|  | info->flash_id = FLASH_MAN_AMD & FLASH_VENDMASK; | 
|  | break; | 
|  | case FUJ_MANUFACT: | 
|  | info->flash_id = FLASH_MAN_FUJ & FLASH_VENDMASK; | 
|  | break; | 
|  | default: | 
|  | info->flash_id = FLASH_UNKNOWN; | 
|  | info->sector_count = 0; | 
|  | info->size = 0; | 
|  | return (0);	/* no or unknown flash */ | 
|  | } | 
|  |  | 
|  | /* Write auto select command sequence and test FLASH answer */ | 
|  | WRITE_UNLOCK (baseaddr); | 
|  | *(volatile __u16 *) (baseaddr + ID_ADDR) = (__u16) ID_CMD; | 
|  | flashtest /* device ID */  = *(volatile __u16 *) (baseaddr + 2); | 
|  | *(volatile __u16 *) (baseaddr + ID_ADDR) = (__u16) RESET_CMD; | 
|  |  | 
|  | /* toggling_bits = (flashtest == TOSHIBA)?(DQ6):(DQ2|DQ6); */ | 
|  | toggling_bits = (DQ2 | DQ6); | 
|  |  | 
|  | switch ((__u32) ((flashtest << 16) + flashtest)) { | 
|  | case AMD_ID_LV160B: | 
|  | info->flash_id += | 
|  | (FLASH_AM160LV | FLASH_AM160B) & FLASH_TYPEMASK; | 
|  | info->sector_count = CONFIG_SYS_MAX_FLASH_SECT; | 
|  | info->size = CONFIG_SYS_FLASH_SIZE; | 
|  | /* 1*16K Boot Block | 
|  | 2*8K Parameter Block | 
|  | 1*32K Small Main Block */ | 
|  | info->start[0] = baseaddr; | 
|  | info->start[1] = baseaddr + 0x4000; | 
|  | info->start[2] = baseaddr + 0x6000; | 
|  | info->start[3] = baseaddr + 0x8000; | 
|  | for (i = 1; i < info->sector_count; i++) | 
|  | info->start[3 + i] = baseaddr + i * CONFIG_SYS_MAIN_SECT_SIZE; | 
|  | break; | 
|  | default: | 
|  | info->flash_id = FLASH_UNKNOWN; | 
|  | return (0);	/* no or unknown flash */ | 
|  | } | 
|  |  | 
|  | for (i = 0; i < info->sector_count; i++) { | 
|  | /* Write auto select command sequence and test FLASH answer */ | 
|  | WRITE_UNLOCK (info->start[i]); | 
|  | *(volatile __u16 *) (info->start[i] + ID_ADDR) = (__u16) ID_CMD; | 
|  | flashtest /* protected verify */  = *(volatile __u16 *) (info->start[i] + 4); | 
|  | *(volatile __u16 *) (info->start[i] + ID_ADDR) = (__u16) RESET_CMD; | 
|  | if (flashtest & 0x0001) { | 
|  | info->protect[i] = 1;	/* D0 = 1 if protected */ | 
|  | } else { | 
|  | info->protect[i] = 0; | 
|  | } | 
|  | } | 
|  | return (info->size); | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | ulong flash_init (void) | 
|  | { | 
|  | ulong size = 0; | 
|  | int i; | 
|  |  | 
|  | /* Init: no FLASHes known */ | 
|  | for (i = 0; i < CONFIG_SYS_MAX_FLASH_BANKS; ++i) { | 
|  | flash_info[i].flash_id = FLASH_UNKNOWN; | 
|  | } | 
|  |  | 
|  | /* Static FLASH Bank configuration here (only one bank) */ | 
|  | size = flash_get_size (CONFIG_SYS_FLASH_BASE, &flash_info[0]); | 
|  |  | 
|  | if (flash_info[0].flash_id == FLASH_UNKNOWN || size == 0) { | 
|  | printf ("## Unknown FLASH on Bank 0 - Size = 0x%08lx = %ld MB\n", | 
|  | size, size >> 20); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * protect monitor and environment sectors | 
|  | */ | 
|  | flash_protect (FLAG_PROTECT_SET, | 
|  | CONFIG_SYS_FLASH_BASE, | 
|  | CONFIG_SYS_FLASH_BASE + monitor_flash_len - 1, | 
|  | &flash_info[0]); | 
|  |  | 
|  | flash_protect (FLAG_PROTECT_SET, | 
|  | CONFIG_ENV_ADDR, | 
|  | CONFIG_ENV_ADDR + CONFIG_ENV_SIZE - 1, &flash_info[0]); | 
|  |  | 
|  | return size; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | void flash_print_info (flash_info_t * info) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | if (info->flash_id == FLASH_UNKNOWN) { | 
|  | printf ("missing or unknown FLASH type\n"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | switch (info->flash_id & FLASH_VENDMASK) { | 
|  | case FLASH_MAN_AMD: | 
|  | printf ("AMD "); | 
|  | break; | 
|  | case FLASH_MAN_FUJ: | 
|  | printf ("Fujitsu "); | 
|  | break; | 
|  | default: | 
|  | printf ("Unknown Vendor "); | 
|  | break; | 
|  | } | 
|  |  | 
|  | switch (info->flash_id & FLASH_TYPEMASK) { | 
|  | case FLASH_AMDL323B: | 
|  | printf ("29DL323B (32 M, bottom sector)\n"); | 
|  | break; | 
|  | case (FLASH_AM160LV | FLASH_AM160B): | 
|  | printf ("29LV160BE (1M x 16, bottom sector)\n"); | 
|  | break; | 
|  | default: | 
|  | printf ("Unknown Chip Type\n"); | 
|  | break; | 
|  | } | 
|  |  | 
|  | printf ("  Size: %ld MB in %d Sectors\n", | 
|  | info->size >> 20, info->sector_count); | 
|  | printf ("  Sector Start Addresses:"); | 
|  | for (i = 0; i < info->sector_count; i++) { | 
|  | if ((i % 4) == 0) | 
|  | printf ("\n   "); | 
|  | printf (" S%02d @ 0x%08lX%s", i, | 
|  | info->start[i], info->protect[i] ? " !" : "  "); | 
|  | } | 
|  | printf ("\n"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | int flash_check_protection (flash_info_t * info, int s_first, int s_last) | 
|  | { | 
|  | int sect, prot = 0; | 
|  |  | 
|  | for (sect = s_first; sect <= s_last; sect++) | 
|  | if (info->protect[sect]) | 
|  | prot++; | 
|  | if (prot) | 
|  | printf ("- can't erase %d protected sectors\n", prot); | 
|  | return prot; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | int flash_check_erase_amd (ulong start) | 
|  | { | 
|  | __u16 v1, v2; | 
|  |  | 
|  | v1 = *(volatile __u16 *) (start); | 
|  | v2 = *(volatile __u16 *) (start); | 
|  |  | 
|  | if (((v1 ^ v2) & toggling_bits) == toggling_bits) { | 
|  | if (((v1 | v2) & DQ5) == DQ5) { | 
|  | printf ("[DQ5] "); | 
|  | /* OOPS: exceeded timing limits */ | 
|  |  | 
|  | v1 = *(volatile __u16 *) (start); | 
|  | v2 = *(volatile __u16 *) (start); | 
|  |  | 
|  | if (((v1 ^ v2) & toggling_bits) == toggling_bits) { | 
|  |  | 
|  | printf ("[%s] ", | 
|  | ((toggling_bits & (DQ2 | DQ6)) == | 
|  | (DQ2 | DQ6)) ? "DQ2,DQ6" : "DQ6"); | 
|  |  | 
|  | /* OOPS: there is an erasure in progress, | 
|  | *       try to reset chip */ | 
|  | *(volatile __u16 *) (start) = | 
|  | (__u16) RESET_CMD; | 
|  |  | 
|  | return 1;	/* still busy */ | 
|  | } | 
|  | } | 
|  | return 1;	/* still busy */ | 
|  | } | 
|  | return 0;		/* be free */ | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | int flash_erase (flash_info_t * info, int s_first, int s_last) | 
|  | { | 
|  | int flag, sect, setup_offset = 0; | 
|  | int rc = ERR_OK; | 
|  |  | 
|  | if (info->flash_id == FLASH_UNKNOWN) { | 
|  | printf ("- missing\n"); | 
|  | return ERR_UNKNOWN_FLASH_TYPE; | 
|  | } | 
|  |  | 
|  | if ((s_first < 0) || (s_first > s_last)) { | 
|  | printf ("- no sectors to erase\n"); | 
|  | return ERR_INVAL; | 
|  | } | 
|  |  | 
|  | if (flash_check_protection (info, s_first, s_last)) | 
|  | return ERR_PROTECTED; | 
|  |  | 
|  | switch (info->flash_id & FLASH_VENDMASK) { | 
|  | case FLASH_MAN_FUJ: | 
|  | case FLASH_MAN_AMD: | 
|  | switch (info->flash_id & FLASH_TYPEMASK) { | 
|  | case (FLASH_AM160LV | FLASH_AM160B): | 
|  | setup_offset = UNLOCK_ADDR1;	/* just the adress for setup_cmd differs */ | 
|  | case FLASH_AMDL323B: | 
|  | /* | 
|  | * Disable interrupts which might cause a timeout | 
|  | * here. Remember that our exception vectors are | 
|  | * at address 0 in the flash, and we don't want a | 
|  | * (ticker) exception to happen while the flash | 
|  | * chip is in programming mode. | 
|  | */ | 
|  | flag = disable_interrupts (); | 
|  | /* Start erase on unprotected sectors */ | 
|  | for (sect = s_first; sect <= s_last && !ctrlc (); | 
|  | sect++) { | 
|  | printf ("Erasing sector %2d ... ", sect); | 
|  |  | 
|  | if (info->protect[sect] == 0) { | 
|  | /* not protected */ | 
|  | /* Write sector erase command sequence */ | 
|  | WRITE_UNLOCK (info->start[0]); | 
|  | *(volatile __u16 *) (info->start[0] + | 
|  | setup_offset) = | 
|  | (__u16) SETUP_CMD; | 
|  | WRITE_UNLOCK (info->start[0]); | 
|  | *(volatile __u16 *) (info-> | 
|  | start[sect]) = | 
|  | (__u16) SECERASE_CMD; | 
|  |  | 
|  | /* wait some time */ | 
|  | reset_timer_masked (); | 
|  | while (get_timer_masked () < 1000) { | 
|  | } | 
|  |  | 
|  | /* arm simple, non interrupt dependent timer */ | 
|  | reset_timer_masked (); | 
|  | while (flash_check_erase_amd (info->start[sect])) { | 
|  | if (get_timer_masked () > CONFIG_SYS_FLASH_ERASE_TOUT) { | 
|  | printf ("timeout!\n"); | 
|  | /* OOPS: reach timeout, | 
|  | * try to reset chip | 
|  | */ | 
|  | *(volatile __u16 *) (info-> start[sect]) = (__u16) RESET_CMD; | 
|  | rc = ERR_TIMOUT; | 
|  | goto outahere_323B; | 
|  | } | 
|  | } | 
|  | printf ("ok.\n"); | 
|  | } else { | 
|  | printf ("protected!\n"); | 
|  | } | 
|  | } | 
|  | if (ctrlc ()) | 
|  | printf ("User Interrupt!\n"); | 
|  | outahere_323B: | 
|  | /* allow flash to settle - wait 10 ms */ | 
|  | udelay_masked (10000); | 
|  | if (flag) | 
|  | enable_interrupts (); | 
|  | return rc; | 
|  | default: | 
|  | printf ("- unknown chip type\n"); | 
|  | return ERR_UNKNOWN_FLASH_TYPE; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | printf ("- unknown vendor "); | 
|  | return ERR_UNKNOWN_FLASH_VENDOR; | 
|  | } | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | */ | 
|  | int flash_check_write_amd (ulong dest) | 
|  | { | 
|  | __u16 v1, v2; | 
|  |  | 
|  | v1 = *(volatile __u16 *) (dest); | 
|  | v2 = *(volatile __u16 *) (dest); | 
|  |  | 
|  | /* DQ6 toggles during write */ | 
|  | if (((v1 ^ v2) & DQ6) == DQ6) { | 
|  | if (((v1 | v2) & DQ5) == DQ5) { | 
|  | printf ("[DQ5] @ %08lX\n", dest); | 
|  |  | 
|  | /* OOPS: exceeded timing limits, | 
|  | *       try to reset chip */ | 
|  | *(volatile __u16 *) (dest) = (__u16) RESET_CMD; | 
|  | return 0;	/* be free */ | 
|  | } | 
|  | return 1;	/* still busy */ | 
|  | } | 
|  |  | 
|  | return 0;		/* be free */ | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | * Copy memory to flash | 
|  | */ | 
|  | static int write_word (flash_info_t * info, ulong dest, ushort data) | 
|  | { | 
|  | int rc = ERR_OK; | 
|  | int flag; | 
|  |  | 
|  | /* Check if Flash is (sufficiently) erased */ | 
|  | if ((*(__u16 *) (dest) & data) != data) | 
|  | return ERR_NOT_ERASED; | 
|  |  | 
|  | /* | 
|  | * Disable interrupts which might cause a timeout | 
|  | * here. Remember that our exception vectors are | 
|  | * at address 0 in the flash, and we don't want a | 
|  | * (ticker) exception to happen while the flash | 
|  | * chip is in programming mode. | 
|  | */ | 
|  | flag = disable_interrupts (); | 
|  |  | 
|  | /* Write program command sequence */ | 
|  | WRITE_UNLOCK (info->start[0]); | 
|  |  | 
|  | /* Flash dependend program seqence */ | 
|  | switch (info->flash_id & FLASH_VENDMASK) { | 
|  | case FLASH_MAN_FUJ: | 
|  | case FLASH_MAN_AMD: | 
|  | switch (info->flash_id & FLASH_TYPEMASK) { | 
|  | case (FLASH_AM160LV | FLASH_AM160B): | 
|  | *(volatile __u16 *) (info->start[0] + UNLOCK_ADDR1) = | 
|  | (__u16) PROG_CMD; | 
|  | *(volatile __u16 *) (dest) = (__u16) data; | 
|  | break; | 
|  | case FLASH_AMDL323B: | 
|  | *(volatile __u16 *) (dest) = (__u16) PROG_CMD; | 
|  | *(volatile __u16 *) (dest) = (__u16) data; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* arm simple, non interrupt dependent timer */ | 
|  | reset_timer_masked (); | 
|  |  | 
|  | while (flash_check_write_amd (dest)) { | 
|  | if (get_timer_masked () > CONFIG_SYS_FLASH_WRITE_TOUT) { | 
|  | printf ("timeout! @ %08lX\n", dest); | 
|  | /* OOPS: reach timeout, | 
|  | *       try to reset chip */ | 
|  | *(volatile __u16 *) (dest) = (__u16) RESET_CMD; | 
|  |  | 
|  | rc = ERR_TIMOUT; | 
|  | goto outahere_323B; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Check if Flash was (accurately) written */ | 
|  | if (*(__u16 *) (dest) != data) | 
|  | rc = ERR_PROG_ERROR; | 
|  |  | 
|  | outahere_323B: | 
|  | if (flag) | 
|  | enable_interrupts (); | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------- | 
|  | * Copy memory to flash. | 
|  | */ | 
|  | int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt) | 
|  | { | 
|  | ulong cp, wp; | 
|  | ushort data; | 
|  | int l; | 
|  | int i, rc; | 
|  |  | 
|  | wp = (addr & ~1);	/* get lower word aligned address */ | 
|  |  | 
|  | /* | 
|  | * handle unaligned start bytes | 
|  | */ | 
|  | if ((l = addr - wp) != 0) { | 
|  | data = 0; | 
|  | for (i = 0, cp = wp; i < l; ++i, ++cp) { | 
|  | data = (data >> 8) | (*(uchar *) cp << 8); | 
|  | } | 
|  | for (; i < 2 && cnt > 0; ++i) { | 
|  | data = (data >> 8) | (*src++ << 8); | 
|  | --cnt; | 
|  | ++cp; | 
|  | } | 
|  | for (; cnt == 0 && i < 2; ++i, ++cp) { | 
|  | data = (data >> 8) | (*(uchar *) cp << 8); | 
|  | } | 
|  |  | 
|  | if ((rc = write_word (info, wp, data)) != 0) { | 
|  | return (rc); | 
|  | } | 
|  | wp += 2; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * handle word aligned part | 
|  | */ | 
|  | while (cnt >= 2) { | 
|  | data = *((ushort *) src); | 
|  | if ((rc = write_word (info, wp, data)) != 0) | 
|  | return (rc); | 
|  | src += 2; | 
|  | wp += 2; | 
|  | cnt -= 2; | 
|  | } | 
|  |  | 
|  | if (cnt == 0) | 
|  | return ERR_OK; | 
|  |  | 
|  | /* | 
|  | * handle unaligned tail bytes | 
|  | */ | 
|  | data = 0; | 
|  | for (i = 0, cp = wp; i < 2 && cnt > 0; ++i, ++cp) { | 
|  | data = (data >> 8) | (*src++ << 8); | 
|  | --cnt; | 
|  | } | 
|  | for (; i < 2; ++i, ++cp) { | 
|  | data = (data >> 8) | (*(uchar *) cp << 8); | 
|  | } | 
|  |  | 
|  | return write_word (info, wp, data); | 
|  | } |