/* * (C) Copyright 2006 Embedded Artists AB * * 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 #include /* IAP commands use 32 bytes at the top of CPU internal sram, we use 512 bytes below that */ #define COPY_BUFFER_LOCATION 0x40003de0 #define IAP_LOCATION 0x7ffffff1 #define IAP_CMD_PREPARE 50 #define IAP_CMD_COPY 51 #define IAP_CMD_ERASE 52 #define IAP_CMD_CHECK 53 #define IAP_CMD_ID 54 #define IAP_CMD_VERSION 55 #define IAP_CMD_COMPARE 56 #define IAP_RET_CMD_SUCCESS 0 #define SST_BASEADDR 0x80000000 #define SST_ADDR1 ((volatile ushort*)(SST_BASEADDR + (0x5555 << 1))) #define SST_ADDR2 ((volatile ushort*)(SST_BASEADDR + (0x2AAA << 1))) static unsigned long command[5]; static unsigned long result[2]; flash_info_t flash_info[CFG_MAX_FLASH_BANKS]; extern void iap_entry(unsigned long * command, unsigned long * result); /*----------------------------------------------------------------------- * */ int get_flash_sector(flash_info_t * info, ulong flash_addr) { int i; for(i=1; i < (info->sector_count); i++) { if (flash_addr < (info->start[i])) break; } return (i-1); } /*----------------------------------------------------------------------- * This function assumes that flash_addr is aligned on 512 bytes boundary * in flash. This function also assumes that prepare have been called * for the sector in question. */ int copy_buffer_to_flash(flash_info_t * info, ulong flash_addr) { int first_sector; int last_sector; first_sector = get_flash_sector(info, flash_addr); last_sector = get_flash_sector(info, flash_addr + 512 - 1); /* prepare sectors for write */ command[0] = IAP_CMD_PREPARE; command[1] = first_sector; command[2] = last_sector; iap_entry(command, result); if (result[0] != IAP_RET_CMD_SUCCESS) { printf("IAP prepare failed\n"); return ERR_PROG_ERROR; } command[0] = IAP_CMD_COPY; command[1] = flash_addr; command[2] = COPY_BUFFER_LOCATION; command[3] = 512; command[4] = CFG_SYS_CLK_FREQ >> 10; iap_entry(command, result); if (result[0] != IAP_RET_CMD_SUCCESS) { printf("IAP copy failed\n"); return 1; } return 0; } /*----------------------------------------------------------------------- * */ void write_word_sst(ulong addr, ushort data) { ushort tmp; *SST_ADDR1 = 0x00AA; *SST_ADDR2 = 0x0055; *SST_ADDR1 = 0x00A0; *((volatile ushort*)addr) = data; /* do data polling */ do { tmp = *((volatile ushort*)addr); } while (tmp != data); } /*----------------------------------------------------------------------- */ ulong flash_init (void) { int j, k; ulong size = 0; ulong flashbase = 0; flash_info[0].flash_id = (PHILIPS_LPC2292 & FLASH_VENDMASK); flash_info[0].size = 0x003E000; /* 256 - 8 KB */ flash_info[0].sector_count = 17; memset (flash_info[0].protect, 0, 17); flashbase = 0x00000000; for (j = 0, k = 0; j < 8; j++, k++) { flash_info[0].start[k] = flashbase; flashbase += 0x00002000; } for (j = 0; j < 2; j++, k++) { flash_info[0].start[k] = flashbase; flashbase += 0x00010000; } for (j = 0; j < 7; j++, k++) { flash_info[0].start[k] = flashbase; flashbase += 0x00002000; } size += flash_info[0].size; flash_info[1].flash_id = (SST_MANUFACT & FLASH_VENDMASK); flash_info[1].size = 0x00200000; /* 2 MB */ flash_info[1].sector_count = 512; memset (flash_info[1].protect, 0, 512); flashbase = SST_BASEADDR; for (j=0; j<512; j++) { flash_info[1].start[j] = flashbase; flashbase += 0x1000; /* 4 KB sectors */ } size += flash_info[1].size; /* Protect monitor and environment sectors */ flash_protect (FLAG_PROTECT_SET, 0x0, 0x0 + monitor_flash_len - 1, &flash_info[0]); flash_protect (FLAG_PROTECT_SET, CFG_ENV_ADDR, CFG_ENV_ADDR + CFG_ENV_SIZE - 1, &flash_info[0]); return size; } /*----------------------------------------------------------------------- */ void flash_print_info (flash_info_t * info) { int i; int erased = 0; unsigned long j; unsigned long count; unsigned char *p; switch (info->flash_id & FLASH_VENDMASK) { case (SST_MANUFACT & FLASH_VENDMASK): printf("SST: "); break; case (PHILIPS_LPC2292 & FLASH_VENDMASK): printf("Philips: "); break; default: printf ("Unknown Vendor "); break; } printf (" Size: %ld KB in %d Sectors\n", info->size >> 10, info->sector_count); printf (" Sector Start Addresses:"); for (i = 0; i < info->sector_count; i++) { if ((i % 5) == 0) { printf ("\n "); } if (i < (info->sector_count - 1)) { count = info->start[i+1] - info->start[i]; } else { count = info->start[0] + info->size - info->start[i]; } p = (unsigned char*)(info->start[i]); erased = 1; for (j = 0; j < count; j++) { if (*p != 0xFF) { erased = 0; break; } p++; } printf (" %08lX%s%s", info->start[i], info->protect[i] ? " RO" : " ", erased ? " E" : " "); } printf ("\n"); } /*----------------------------------------------------------------------- */ int flash_erase_philips (flash_info_t * info, int s_first, int s_last) { int flag; int prot; int sect; prot = 0; for (sect = s_first; sect <= s_last; ++sect) { if (info->protect[sect]) { prot++; } } if (prot) return ERR_PROTECTED; flag = disable_interrupts(); printf ("Erasing %d sectors starting at sector %2d.\n" "This make take some time ... ", s_last - s_first + 1, s_first); command[0] = IAP_CMD_PREPARE; command[1] = s_first; command[2] = s_last; iap_entry(command, result); if (result[0] != IAP_RET_CMD_SUCCESS) { printf("IAP prepare failed\n"); return ERR_PROTECTED; } command[0] = IAP_CMD_ERASE; command[1] = s_first; command[2] = s_last; command[3] = CFG_SYS_CLK_FREQ >> 10; iap_entry(command, result); if (result[0] != IAP_RET_CMD_SUCCESS) { printf("IAP erase failed\n"); return ERR_PROTECTED; } if (flag) enable_interrupts(); return ERR_OK; } int flash_erase_sst (flash_info_t * info, int s_first, int s_last) { int i; for (i = s_first; i <= s_last; i++) { *SST_ADDR1 = 0x00AA; *SST_ADDR2 = 0x0055; *SST_ADDR1 = 0x0080; *SST_ADDR1 = 0x00AA; *SST_ADDR2 = 0x0055; *((volatile ushort*)(info->start[i])) = 0x0030; /* wait for erase to finish */ udelay(25000); } return ERR_OK; } int flash_erase (flash_info_t * info, int s_first, int s_last) { switch (info->flash_id & FLASH_VENDMASK) { case (SST_MANUFACT & FLASH_VENDMASK): return flash_erase_sst(info, s_first, s_last); case (PHILIPS_LPC2292 & FLASH_VENDMASK): return flash_erase_philips(info, s_first, s_last); default: return ERR_PROTECTED; } return ERR_PROTECTED; } /*----------------------------------------------------------------------- * Copy memory to flash. * * cnt is in bytes */ int write_buff_sst (flash_info_t * info, uchar * src, ulong addr, ulong cnt) { ushort tmp; ulong i; uchar* src_org; uchar* dst_org; ulong cnt_org = cnt; int ret = ERR_OK; src_org = src; dst_org = (uchar*)addr; if (addr & 1) { /* if odd address */ tmp = *((uchar*)(addr - 1)); /* little endian */ tmp |= (*src << 8); write_word_sst(addr - 1, tmp); addr += 1; cnt -= 1; src++; } while (cnt > 1) { tmp = ((*(src+1)) << 8) + (*src); /* little endian */ write_word_sst(addr, tmp); addr += 2; src += 2; cnt -= 2; } if (cnt > 0) { tmp = (*((uchar*)(addr + 1))) << 8; tmp |= *src; write_word_sst(addr, tmp); } for (i = 0; i < cnt_org; i++) { if (*dst_org != *src_org) { printf("Write failed. Byte %lX differs\n", i); ret = ERR_PROG_ERROR; break; } dst_org++; src_org++; } return ret; } int write_buff_philips (flash_info_t * info, uchar * src, ulong addr, ulong cnt) { int first_copy_size; int last_copy_size; int first_block; int last_block; int nbr_mid_blocks; uchar memmap_value; ulong i; uchar* src_org; uchar* dst_org; int ret = ERR_OK; src_org = src; dst_org = (uchar*)addr; first_block = addr / 512; last_block = (addr + cnt) / 512; nbr_mid_blocks = last_block - first_block - 1; first_copy_size = 512 - (addr % 512); last_copy_size = (addr + cnt) % 512; #if 0 printf("\ncopy first block: (1) %lX -> %lX 0x200 bytes, " "(2) %lX -> %lX 0x%X bytes, (3) %lX -> %lX 0x200 bytes\n", (ulong)(first_block * 512), (ulong)COPY_BUFFER_LOCATION, (ulong)src, (ulong)(COPY_BUFFER_LOCATION + 512 - first_copy_size), first_copy_size, (ulong)COPY_BUFFER_LOCATION, (ulong)(first_block * 512)); #endif /* copy first block */ memcpy((void*)COPY_BUFFER_LOCATION, (void*)(first_block * 512), 512); memcpy((void*)(COPY_BUFFER_LOCATION + 512 - first_copy_size), src, first_copy_size); copy_buffer_to_flash(info, first_block * 512); src += first_copy_size; addr += first_copy_size; /* copy middle blocks */ for (i = 0; i < nbr_mid_blocks; i++) { #if 0 printf("copy middle block: %lX -> %lX 512 bytes, " "%lX -> %lX 512 bytes\n", (ulong)src, (ulong)COPY_BUFFER_LOCATION, (ulong)COPY_BUFFER_LOCATION, (ulong)addr); #endif memcpy((void*)COPY_BUFFER_LOCATION, src, 512); copy_buffer_to_flash(info, addr); src += 512; addr += 512; } if (last_copy_size > 0) { #if 0 printf("copy last block: (1) %lX -> %lX 0x200 bytes, " "(2) %lX -> %lX 0x%X bytes, (3) %lX -> %lX x200 bytes\n", (ulong)(last_block * 512), (ulong)COPY_BUFFER_LOCATION, (ulong)src, (ulong)(COPY_BUFFER_LOCATION), last_copy_size, (ulong)COPY_BUFFER_LOCATION, (ulong)addr); #endif /* copy last block */ memcpy((void*)COPY_BUFFER_LOCATION, (void*)(last_block * 512), 512); memcpy((void*)COPY_BUFFER_LOCATION, src, last_copy_size); copy_buffer_to_flash(info, addr); } /* verify write */ memmap_value = GET8(MEMMAP); disable_interrupts(); PUT8(MEMMAP, 01); /* we must make sure that initial 64 bytes are taken from flash when we do the compare */ for (i = 0; i < cnt; i++) { if (*dst_org != *src_org){ printf("Write failed. Byte %lX differs\n", i); ret = ERR_PROG_ERROR; break; } dst_org++; src_org++; } PUT8(MEMMAP, memmap_value); enable_interrupts(); return ret; } int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt) { switch (info->flash_id & FLASH_VENDMASK) { case (SST_MANUFACT & FLASH_VENDMASK): return write_buff_sst(info, src, addr, cnt); case (PHILIPS_LPC2292 & FLASH_VENDMASK): return write_buff_philips(info, src, addr, cnt); default: return ERR_PROG_ERROR; } return ERR_PROG_ERROR; }