blob: c40bf661052082b2d1424d8f70e0f44ad86602bc [file] [log] [blame]
wdenk5653fc32004-02-08 22:55:38 +00001/*
wdenkbf9e3b32004-02-12 00:47:09 +00002 * (C) Copyright 2002-2004
wdenk5653fc32004-02-08 22:55:38 +00003 * Brad Kemp, Seranoa Networks, Brad.Kemp@seranoa.com
4 *
5 * Copyright (C) 2003 Arabella Software Ltd.
6 * Yuli Barcohen <yuli@arabellasw.com>
wdenk5653fc32004-02-08 22:55:38 +00007 *
wdenkbf9e3b32004-02-12 00:47:09 +00008 * Copyright (C) 2004
9 * Ed Okerson
Stefan Roese260421a2006-11-13 13:55:24 +010010 *
11 * Copyright (C) 2006
12 * Tolunay Orkun <listmember@orkun.us>
wdenkbf9e3b32004-02-12 00:47:09 +000013 *
wdenk5653fc32004-02-08 22:55:38 +000014 * See file CREDITS for list of people who contributed to this
15 * project.
16 *
17 * This program is free software; you can redistribute it and/or
18 * modify it under the terms of the GNU General Public License as
19 * published by the Free Software Foundation; either version 2 of
20 * the License, or (at your option) any later version.
21 *
22 * This program is distributed in the hope that it will be useful,
23 * but WITHOUT ANY WARRANTY; without even the implied warranty of
24 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
25 * GNU General Public License for more details.
26 *
27 * You should have received a copy of the GNU General Public License
28 * along with this program; if not, write to the Free Software
29 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
30 * MA 02111-1307 USA
31 *
wdenk5653fc32004-02-08 22:55:38 +000032 */
33
34/* The DEBUG define must be before common to enable debugging */
wdenk2d1a5372004-02-23 19:30:57 +000035/* #define DEBUG */
36
wdenk5653fc32004-02-08 22:55:38 +000037#include <common.h>
38#include <asm/processor.h>
Haiying Wang3a197b22007-02-21 16:52:31 +010039#include <asm/io.h>
wdenk4c0d4c32004-06-09 17:34:58 +000040#include <asm/byteorder.h>
wdenk2a8af182005-04-13 10:02:42 +000041#include <environment.h>
wdenk028ab6b2004-02-23 23:54:43 +000042
wdenk5653fc32004-02-08 22:55:38 +000043/*
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +010044 * This file implements a Common Flash Interface (CFI) driver for
45 * U-Boot.
46 *
47 * The width of the port and the width of the chips are determined at
48 * initialization. These widths are used to calculate the address for
49 * access CFI data structures.
wdenk5653fc32004-02-08 22:55:38 +000050 *
51 * References
52 * JEDEC Standard JESD68 - Common Flash Interface (CFI)
53 * JEDEC Standard JEP137-A Common Flash Interface (CFI) ID Codes
54 * Intel Application Note 646 Common Flash Interface (CFI) and Command Sets
55 * Intel 290667-008 3 Volt Intel StrataFlash Memory datasheet
Stefan Roese260421a2006-11-13 13:55:24 +010056 * AMD CFI Specification, Release 2.0 December 1, 2001
57 * AMD/Spansion Application Note: Migration from Single-byte to Three-byte
58 * Device IDs, Publication Number 25538 Revision A, November 8, 2001
wdenk5653fc32004-02-08 22:55:38 +000059 *
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020060 * Define CONFIG_SYS_WRITE_SWAPPED_DATA, if you have to swap the Bytes between
Heiko Schocherd0b6e142007-01-19 18:05:26 +010061 * reading and writing ... (yes there is such a Hardware).
wdenk5653fc32004-02-08 22:55:38 +000062 */
63
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020064#ifndef CONFIG_SYS_FLASH_BANKS_LIST
65#define CONFIG_SYS_FLASH_BANKS_LIST { CONFIG_SYS_FLASH_BASE }
wdenkbf9e3b32004-02-12 00:47:09 +000066#endif
67
wdenk5653fc32004-02-08 22:55:38 +000068#define FLASH_CMD_CFI 0x98
69#define FLASH_CMD_READ_ID 0x90
70#define FLASH_CMD_RESET 0xff
71#define FLASH_CMD_BLOCK_ERASE 0x20
72#define FLASH_CMD_ERASE_CONFIRM 0xD0
73#define FLASH_CMD_WRITE 0x40
74#define FLASH_CMD_PROTECT 0x60
75#define FLASH_CMD_PROTECT_SET 0x01
76#define FLASH_CMD_PROTECT_CLEAR 0xD0
77#define FLASH_CMD_CLEAR_STATUS 0x50
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +040078#define FLASH_CMD_READ_STATUS 0x70
wdenkbf9e3b32004-02-12 00:47:09 +000079#define FLASH_CMD_WRITE_TO_BUFFER 0xE8
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +040080#define FLASH_CMD_WRITE_BUFFER_PROG 0xE9
wdenkbf9e3b32004-02-12 00:47:09 +000081#define FLASH_CMD_WRITE_BUFFER_CONFIRM 0xD0
wdenk5653fc32004-02-08 22:55:38 +000082
83#define FLASH_STATUS_DONE 0x80
84#define FLASH_STATUS_ESS 0x40
85#define FLASH_STATUS_ECLBS 0x20
86#define FLASH_STATUS_PSLBS 0x10
87#define FLASH_STATUS_VPENS 0x08
88#define FLASH_STATUS_PSS 0x04
89#define FLASH_STATUS_DPS 0x02
90#define FLASH_STATUS_R 0x01
91#define FLASH_STATUS_PROTECT 0x01
92
93#define AMD_CMD_RESET 0xF0
94#define AMD_CMD_WRITE 0xA0
95#define AMD_CMD_ERASE_START 0x80
96#define AMD_CMD_ERASE_SECTOR 0x30
wdenk855a4962004-03-14 18:23:55 +000097#define AMD_CMD_UNLOCK_START 0xAA
98#define AMD_CMD_UNLOCK_ACK 0x55
Stefan Roese79b4cda2006-02-28 15:29:58 +010099#define AMD_CMD_WRITE_TO_BUFFER 0x25
100#define AMD_CMD_WRITE_BUFFER_CONFIRM 0x29
wdenk5653fc32004-02-08 22:55:38 +0000101
102#define AMD_STATUS_TOGGLE 0x40
103#define AMD_STATUS_ERROR 0x20
Stefan Roese79b4cda2006-02-28 15:29:58 +0100104
Rafael Camposbc9019e2008-07-31 10:22:20 +0200105#define ATM_CMD_UNLOCK_SECT 0x70
106#define ATM_CMD_SOFTLOCK_START 0x80
107#define ATM_CMD_LOCK_SECT 0x40
108
Stefan Roese260421a2006-11-13 13:55:24 +0100109#define FLASH_OFFSET_MANUFACTURER_ID 0x00
110#define FLASH_OFFSET_DEVICE_ID 0x01
111#define FLASH_OFFSET_DEVICE_ID2 0x0E
112#define FLASH_OFFSET_DEVICE_ID3 0x0F
wdenk5653fc32004-02-08 22:55:38 +0000113#define FLASH_OFFSET_CFI 0x55
Wolfgang Denk92eb7292006-12-27 01:26:13 +0100114#define FLASH_OFFSET_CFI_ALT 0x555
wdenk5653fc32004-02-08 22:55:38 +0000115#define FLASH_OFFSET_CFI_RESP 0x10
wdenkbf9e3b32004-02-12 00:47:09 +0000116#define FLASH_OFFSET_PRIMARY_VENDOR 0x13
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +0100117/* extended query table primary address */
118#define FLASH_OFFSET_EXT_QUERY_T_P_ADDR 0x15
wdenk5653fc32004-02-08 22:55:38 +0000119#define FLASH_OFFSET_WTOUT 0x1F
wdenkbf9e3b32004-02-12 00:47:09 +0000120#define FLASH_OFFSET_WBTOUT 0x20
wdenk5653fc32004-02-08 22:55:38 +0000121#define FLASH_OFFSET_ETOUT 0x21
wdenkbf9e3b32004-02-12 00:47:09 +0000122#define FLASH_OFFSET_CETOUT 0x22
wdenk5653fc32004-02-08 22:55:38 +0000123#define FLASH_OFFSET_WMAX_TOUT 0x23
wdenkbf9e3b32004-02-12 00:47:09 +0000124#define FLASH_OFFSET_WBMAX_TOUT 0x24
wdenk5653fc32004-02-08 22:55:38 +0000125#define FLASH_OFFSET_EMAX_TOUT 0x25
wdenkbf9e3b32004-02-12 00:47:09 +0000126#define FLASH_OFFSET_CEMAX_TOUT 0x26
wdenk5653fc32004-02-08 22:55:38 +0000127#define FLASH_OFFSET_SIZE 0x27
wdenkbf9e3b32004-02-12 00:47:09 +0000128#define FLASH_OFFSET_INTERFACE 0x28
129#define FLASH_OFFSET_BUFFER_SIZE 0x2A
wdenk5653fc32004-02-08 22:55:38 +0000130#define FLASH_OFFSET_NUM_ERASE_REGIONS 0x2C
131#define FLASH_OFFSET_ERASE_REGIONS 0x2D
132#define FLASH_OFFSET_PROTECT 0x02
wdenkbf9e3b32004-02-12 00:47:09 +0000133#define FLASH_OFFSET_USER_PROTECTION 0x85
134#define FLASH_OFFSET_INTEL_PROTECTION 0x81
wdenk5653fc32004-02-08 22:55:38 +0000135
Stefan Roese260421a2006-11-13 13:55:24 +0100136#define CFI_CMDSET_NONE 0
137#define CFI_CMDSET_INTEL_EXTENDED 1
138#define CFI_CMDSET_AMD_STANDARD 2
139#define CFI_CMDSET_INTEL_STANDARD 3
140#define CFI_CMDSET_AMD_EXTENDED 4
141#define CFI_CMDSET_MITSU_STANDARD 256
142#define CFI_CMDSET_MITSU_EXTENDED 257
143#define CFI_CMDSET_SST 258
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400144#define CFI_CMDSET_INTEL_PROG_REGIONS 512
wdenk5653fc32004-02-08 22:55:38 +0000145
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200146#ifdef CONFIG_SYS_FLASH_CFI_AMD_RESET /* needed for STM_ID_29W320DB on UC100 */
wdenkf7d15722004-12-18 22:35:43 +0000147# undef FLASH_CMD_RESET
Stefan Roese260421a2006-11-13 13:55:24 +0100148# define FLASH_CMD_RESET AMD_CMD_RESET /* use AMD-Reset instead */
wdenkf7d15722004-12-18 22:35:43 +0000149#endif
150
wdenk5653fc32004-02-08 22:55:38 +0000151typedef union {
152 unsigned char c;
153 unsigned short w;
154 unsigned long l;
155 unsigned long long ll;
156} cfiword_t;
157
Stefan Roese260421a2006-11-13 13:55:24 +0100158#define NUM_ERASE_REGIONS 4 /* max. number of erase regions */
wdenk5653fc32004-02-08 22:55:38 +0000159
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +0100160static uint flash_offset_cfi[2] = { FLASH_OFFSET_CFI, FLASH_OFFSET_CFI_ALT };
Wolfgang Denk92eb7292006-12-27 01:26:13 +0100161
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200162/* use CONFIG_SYS_MAX_FLASH_BANKS_DETECT if defined */
163#ifdef CONFIG_SYS_MAX_FLASH_BANKS_DETECT
164# define CFI_MAX_FLASH_BANKS CONFIG_SYS_MAX_FLASH_BANKS_DETECT
Marian Balakowicze6f2e902005-10-11 19:09:42 +0200165#else
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200166# define CFI_MAX_FLASH_BANKS CONFIG_SYS_MAX_FLASH_BANKS
Marian Balakowicze6f2e902005-10-11 19:09:42 +0200167#endif
wdenk5653fc32004-02-08 22:55:38 +0000168
Wolfgang Denk2a112b22008-08-08 16:39:54 +0200169flash_info_t flash_info[CFI_MAX_FLASH_BANKS]; /* FLASH chips info */
170
Stefan Roese79b4cda2006-02-28 15:29:58 +0100171/*
172 * Check if chip width is defined. If not, start detecting with 8bit.
173 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200174#ifndef CONFIG_SYS_FLASH_CFI_WIDTH
175#define CONFIG_SYS_FLASH_CFI_WIDTH FLASH_CFI_8BIT
Stefan Roese79b4cda2006-02-28 15:29:58 +0100176#endif
177
wdenk5653fc32004-02-08 22:55:38 +0000178typedef unsigned long flash_sect_t;
179
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100180/* CFI standard query structure */
181struct cfi_qry {
182 u8 qry[3];
183 u16 p_id;
184 u16 p_adr;
185 u16 a_id;
186 u16 a_adr;
187 u8 vcc_min;
188 u8 vcc_max;
189 u8 vpp_min;
190 u8 vpp_max;
191 u8 word_write_timeout_typ;
192 u8 buf_write_timeout_typ;
193 u8 block_erase_timeout_typ;
194 u8 chip_erase_timeout_typ;
195 u8 word_write_timeout_max;
196 u8 buf_write_timeout_max;
197 u8 block_erase_timeout_max;
198 u8 chip_erase_timeout_max;
199 u8 dev_size;
200 u16 interface_desc;
201 u16 max_buf_write_size;
202 u8 num_erase_regions;
203 u32 erase_region_info[NUM_ERASE_REGIONS];
204} __attribute__((packed));
205
206struct cfi_pri_hdr {
207 u8 pri[3];
208 u8 major_version;
209 u8 minor_version;
210} __attribute__((packed));
211
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100212static void flash_write8(u8 value, void *addr)
213{
214 __raw_writeb(value, addr);
215}
216
217static void flash_write16(u16 value, void *addr)
218{
219 __raw_writew(value, addr);
220}
221
222static void flash_write32(u32 value, void *addr)
223{
224 __raw_writel(value, addr);
225}
226
227static void flash_write64(u64 value, void *addr)
228{
229 /* No architectures currently implement __raw_writeq() */
230 *(volatile u64 *)addr = value;
231}
232
233static u8 flash_read8(void *addr)
234{
235 return __raw_readb(addr);
236}
237
238static u16 flash_read16(void *addr)
239{
240 return __raw_readw(addr);
241}
242
243static u32 flash_read32(void *addr)
244{
245 return __raw_readl(addr);
246}
247
Daniel Hellstrom97bf85d2008-03-28 20:40:19 +0100248static u64 __flash_read64(void *addr)
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100249{
250 /* No architectures currently implement __raw_readq() */
251 return *(volatile u64 *)addr;
252}
253
Daniel Hellstrom97bf85d2008-03-28 20:40:19 +0100254u64 flash_read64(void *addr)__attribute__((weak, alias("__flash_read64")));
255
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200256/*-----------------------------------------------------------------------
257 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200258#if defined(CONFIG_ENV_IS_IN_FLASH) || defined(CONFIG_ENV_ADDR_REDUND) || (CONFIG_SYS_MONITOR_BASE >= CONFIG_SYS_FLASH_BASE)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200259static flash_info_t *flash_get_info(ulong base)
260{
261 int i;
262 flash_info_t * info = 0;
263
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200264 for (i = 0; i < CONFIG_SYS_MAX_FLASH_BANKS; i++) {
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200265 info = & flash_info[i];
266 if (info->size && info->start[0] <= base &&
267 base <= info->start[0] + info->size - 1)
268 break;
269 }
270
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200271 return i == CONFIG_SYS_MAX_FLASH_BANKS ? 0 : info;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200272}
wdenk5653fc32004-02-08 22:55:38 +0000273#endif
274
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100275unsigned long flash_sector_size(flash_info_t *info, flash_sect_t sect)
276{
277 if (sect != (info->sector_count - 1))
278 return info->start[sect + 1] - info->start[sect];
279 else
280 return info->start[0] + info->size - info->start[sect];
281}
282
wdenk5653fc32004-02-08 22:55:38 +0000283/*-----------------------------------------------------------------------
284 * create an address based on the offset and the port width
285 */
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100286static inline void *
287flash_map (flash_info_t * info, flash_sect_t sect, uint offset)
wdenk5653fc32004-02-08 22:55:38 +0000288{
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100289 unsigned int byte_offset = offset * info->portwidth;
290
291 return map_physmem(info->start[sect] + byte_offset,
292 flash_sector_size(info, sect) - byte_offset,
293 MAP_NOCACHE);
294}
295
296static inline void flash_unmap(flash_info_t *info, flash_sect_t sect,
297 unsigned int offset, void *addr)
298{
299 unsigned int byte_offset = offset * info->portwidth;
300
301 unmap_physmem(addr, flash_sector_size(info, sect) - byte_offset);
wdenk5653fc32004-02-08 22:55:38 +0000302}
wdenkbf9e3b32004-02-12 00:47:09 +0000303
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200304/*-----------------------------------------------------------------------
305 * make a proper sized command based on the port and chip widths
306 */
Sebastian Siewior7288f972008-07-15 13:35:23 +0200307static void flash_make_cmd(flash_info_t *info, u32 cmd, void *cmdbuf)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200308{
309 int i;
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400310 int cword_offset;
311 int cp_offset;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200312#if defined(__LITTLE_ENDIAN) || defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Sebastian Siewior340ccb22008-07-16 20:04:49 +0200313 u32 cmd_le = cpu_to_le32(cmd);
314#endif
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400315 uchar val;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200316 uchar *cp = (uchar *) cmdbuf;
317
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400318 for (i = info->portwidth; i > 0; i--){
319 cword_offset = (info->portwidth-i)%info->chipwidth;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200320#if defined(__LITTLE_ENDIAN) || defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400321 cp_offset = info->portwidth - i;
Sebastian Siewior340ccb22008-07-16 20:04:49 +0200322 val = *((uchar*)&cmd_le + cword_offset);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200323#else
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400324 cp_offset = i - 1;
Sebastian Siewior7288f972008-07-15 13:35:23 +0200325 val = *((uchar*)&cmd + sizeof(u32) - cword_offset - 1);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200326#endif
Sebastian Siewior7288f972008-07-15 13:35:23 +0200327 cp[cp_offset] = (cword_offset >= sizeof(u32)) ? 0x00 : val;
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400328 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200329}
330
wdenkbf9e3b32004-02-12 00:47:09 +0000331#ifdef DEBUG
332/*-----------------------------------------------------------------------
333 * Debug support
334 */
Haavard Skinnemoen30557932007-12-13 12:56:29 +0100335static void print_longlong (char *str, unsigned long long data)
wdenkbf9e3b32004-02-12 00:47:09 +0000336{
337 int i;
338 char *cp;
339
340 cp = (unsigned char *) &data;
341 for (i = 0; i < 8; i++)
342 sprintf (&str[i * 2], "%2.2x", *cp++);
343}
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200344
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100345static void flash_printqry (struct cfi_qry *qry)
wdenkbf9e3b32004-02-12 00:47:09 +0000346{
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100347 u8 *p = (u8 *)qry;
wdenkbf9e3b32004-02-12 00:47:09 +0000348 int x, y;
349
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100350 for (x = 0; x < sizeof(struct cfi_qry); x += 16) {
351 debug("%02x : ", x);
352 for (y = 0; y < 16; y++)
353 debug("%2.2x ", p[x + y]);
354 debug(" ");
wdenkbf9e3b32004-02-12 00:47:09 +0000355 for (y = 0; y < 16; y++) {
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100356 unsigned char c = p[x + y];
357 if (c >= 0x20 && c <= 0x7e)
358 debug("%c", c);
359 else
360 debug(".");
wdenkbf9e3b32004-02-12 00:47:09 +0000361 }
Haavard Skinnemoene23741f2007-12-14 15:36:16 +0100362 debug("\n");
wdenkbf9e3b32004-02-12 00:47:09 +0000363 }
364}
wdenkbf9e3b32004-02-12 00:47:09 +0000365#endif
366
367
wdenk5653fc32004-02-08 22:55:38 +0000368/*-----------------------------------------------------------------------
369 * read a character at a port width address
370 */
Haavard Skinnemoen30557932007-12-13 12:56:29 +0100371static inline uchar flash_read_uchar (flash_info_t * info, uint offset)
wdenk5653fc32004-02-08 22:55:38 +0000372{
373 uchar *cp;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100374 uchar retval;
wdenkbf9e3b32004-02-12 00:47:09 +0000375
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100376 cp = flash_map (info, 0, offset);
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200377#if defined(__LITTLE_ENDIAN) || defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100378 retval = flash_read8(cp);
wdenkbf9e3b32004-02-12 00:47:09 +0000379#else
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100380 retval = flash_read8(cp + info->portwidth - 1);
wdenkbf9e3b32004-02-12 00:47:09 +0000381#endif
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100382 flash_unmap (info, 0, offset, cp);
383 return retval;
wdenk5653fc32004-02-08 22:55:38 +0000384}
385
386/*-----------------------------------------------------------------------
Tor Krill90447ec2008-03-28 11:29:10 +0100387 * read a word at a port width address, assume 16bit bus
388 */
389static inline ushort flash_read_word (flash_info_t * info, uint offset)
390{
391 ushort *addr, retval;
392
393 addr = flash_map (info, 0, offset);
394 retval = flash_read16 (addr);
395 flash_unmap (info, 0, offset, addr);
396 return retval;
397}
398
399
400/*-----------------------------------------------------------------------
Stefan Roese260421a2006-11-13 13:55:24 +0100401 * read a long word by picking the least significant byte of each maximum
wdenk5653fc32004-02-08 22:55:38 +0000402 * port size word. Swap for ppc format.
403 */
Haavard Skinnemoen30557932007-12-13 12:56:29 +0100404static ulong flash_read_long (flash_info_t * info, flash_sect_t sect,
405 uint offset)
wdenk5653fc32004-02-08 22:55:38 +0000406{
wdenkbf9e3b32004-02-12 00:47:09 +0000407 uchar *addr;
408 ulong retval;
wdenk5653fc32004-02-08 22:55:38 +0000409
wdenkbf9e3b32004-02-12 00:47:09 +0000410#ifdef DEBUG
411 int x;
412#endif
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100413 addr = flash_map (info, sect, offset);
wdenkbf9e3b32004-02-12 00:47:09 +0000414
415#ifdef DEBUG
416 debug ("long addr is at %p info->portwidth = %d\n", addr,
417 info->portwidth);
418 for (x = 0; x < 4 * info->portwidth; x++) {
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100419 debug ("addr[%x] = 0x%x\n", x, flash_read8(addr + x));
wdenkbf9e3b32004-02-12 00:47:09 +0000420 }
421#endif
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200422#if defined(__LITTLE_ENDIAN) || defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100423 retval = ((flash_read8(addr) << 16) |
424 (flash_read8(addr + info->portwidth) << 24) |
425 (flash_read8(addr + 2 * info->portwidth)) |
426 (flash_read8(addr + 3 * info->portwidth) << 8));
wdenkbf9e3b32004-02-12 00:47:09 +0000427#else
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100428 retval = ((flash_read8(addr + 2 * info->portwidth - 1) << 24) |
429 (flash_read8(addr + info->portwidth - 1) << 16) |
430 (flash_read8(addr + 4 * info->portwidth - 1) << 8) |
431 (flash_read8(addr + 3 * info->portwidth - 1)));
wdenkbf9e3b32004-02-12 00:47:09 +0000432#endif
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100433 flash_unmap(info, sect, offset, addr);
434
wdenkbf9e3b32004-02-12 00:47:09 +0000435 return retval;
wdenk5653fc32004-02-08 22:55:38 +0000436}
437
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200438/*
439 * Write a proper sized command to the correct address
440 */
441static void flash_write_cmd (flash_info_t * info, flash_sect_t sect,
Sebastian Siewior7288f972008-07-15 13:35:23 +0200442 uint offset, u32 cmd)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200443{
Stefan Roese79b4cda2006-02-28 15:29:58 +0100444
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100445 void *addr;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200446 cfiword_t cword;
447
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100448 addr = flash_map (info, sect, offset);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200449 flash_make_cmd (info, cmd, &cword);
450 switch (info->portwidth) {
451 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100452 debug ("fwc addr %p cmd %x %x 8bit x %d bit\n", addr, cmd,
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200453 cword.c, info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100454 flash_write8(cword.c, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200455 break;
456 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100457 debug ("fwc addr %p cmd %x %4.4x 16bit x %d bit\n", addr,
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200458 cmd, cword.w,
459 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100460 flash_write16(cword.w, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200461 break;
462 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100463 debug ("fwc addr %p cmd %x %8.8lx 32bit x %d bit\n", addr,
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200464 cmd, cword.l,
465 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100466 flash_write32(cword.l, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200467 break;
468 case FLASH_CFI_64BIT:
469#ifdef DEBUG
470 {
471 char str[20];
472
473 print_longlong (str, cword.ll);
474
475 debug ("fwrite addr %p cmd %x %s 64 bit x %d bit\n",
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100476 addr, cmd, str,
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200477 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
478 }
479#endif
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100480 flash_write64(cword.ll, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200481 break;
482 }
483
484 /* Ensure all the instructions are fully finished */
485 sync();
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100486
487 flash_unmap(info, sect, offset, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200488}
489
490static void flash_unlock_seq (flash_info_t * info, flash_sect_t sect)
491{
492 flash_write_cmd (info, sect, info->addr_unlock1, AMD_CMD_UNLOCK_START);
493 flash_write_cmd (info, sect, info->addr_unlock2, AMD_CMD_UNLOCK_ACK);
494}
495
496/*-----------------------------------------------------------------------
497 */
498static int flash_isequal (flash_info_t * info, flash_sect_t sect,
499 uint offset, uchar cmd)
500{
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100501 void *addr;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200502 cfiword_t cword;
503 int retval;
504
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100505 addr = flash_map (info, sect, offset);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200506 flash_make_cmd (info, cmd, &cword);
507
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100508 debug ("is= cmd %x(%c) addr %p ", cmd, cmd, addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200509 switch (info->portwidth) {
510 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100511 debug ("is= %x %x\n", flash_read8(addr), cword.c);
512 retval = (flash_read8(addr) == cword.c);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200513 break;
514 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100515 debug ("is= %4.4x %4.4x\n", flash_read16(addr), cword.w);
516 retval = (flash_read16(addr) == cword.w);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200517 break;
518 case FLASH_CFI_32BIT:
Andrew Klossner52514692008-08-21 07:12:26 -0700519 debug ("is= %8.8x %8.8lx\n", flash_read32(addr), cword.l);
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100520 retval = (flash_read32(addr) == cword.l);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200521 break;
522 case FLASH_CFI_64BIT:
523#ifdef DEBUG
524 {
525 char str1[20];
526 char str2[20];
527
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100528 print_longlong (str1, flash_read64(addr));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200529 print_longlong (str2, cword.ll);
530 debug ("is= %s %s\n", str1, str2);
531 }
532#endif
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100533 retval = (flash_read64(addr) == cword.ll);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200534 break;
535 default:
536 retval = 0;
537 break;
538 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100539 flash_unmap(info, sect, offset, addr);
540
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200541 return retval;
542}
543
544/*-----------------------------------------------------------------------
545 */
546static int flash_isset (flash_info_t * info, flash_sect_t sect,
547 uint offset, uchar cmd)
548{
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100549 void *addr;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200550 cfiword_t cword;
551 int retval;
552
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100553 addr = flash_map (info, sect, offset);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200554 flash_make_cmd (info, cmd, &cword);
555 switch (info->portwidth) {
556 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100557 retval = ((flash_read8(addr) & cword.c) == cword.c);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200558 break;
559 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100560 retval = ((flash_read16(addr) & cword.w) == cword.w);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200561 break;
562 case FLASH_CFI_32BIT:
Stefan Roese47cc23c2008-01-02 14:05:37 +0100563 retval = ((flash_read32(addr) & cword.l) == cword.l);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200564 break;
565 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100566 retval = ((flash_read64(addr) & cword.ll) == cword.ll);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200567 break;
568 default:
569 retval = 0;
570 break;
571 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100572 flash_unmap(info, sect, offset, addr);
573
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200574 return retval;
575}
576
577/*-----------------------------------------------------------------------
578 */
579static int flash_toggle (flash_info_t * info, flash_sect_t sect,
580 uint offset, uchar cmd)
581{
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100582 void *addr;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200583 cfiword_t cword;
584 int retval;
585
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100586 addr = flash_map (info, sect, offset);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200587 flash_make_cmd (info, cmd, &cword);
588 switch (info->portwidth) {
589 case FLASH_CFI_8BIT:
Stefan Roesefb8c0612008-06-16 10:40:02 +0200590 retval = flash_read8(addr) != flash_read8(addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200591 break;
592 case FLASH_CFI_16BIT:
Stefan Roesefb8c0612008-06-16 10:40:02 +0200593 retval = flash_read16(addr) != flash_read16(addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200594 break;
595 case FLASH_CFI_32BIT:
Stefan Roesefb8c0612008-06-16 10:40:02 +0200596 retval = flash_read32(addr) != flash_read32(addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200597 break;
598 case FLASH_CFI_64BIT:
Stefan Roesefb8c0612008-06-16 10:40:02 +0200599 retval = flash_read64(addr) != flash_read64(addr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200600 break;
601 default:
602 retval = 0;
603 break;
604 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100605 flash_unmap(info, sect, offset, addr);
606
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200607 return retval;
608}
609
610/*
611 * flash_is_busy - check to see if the flash is busy
612 *
613 * This routine checks the status of the chip and returns true if the
614 * chip is busy.
615 */
616static int flash_is_busy (flash_info_t * info, flash_sect_t sect)
617{
618 int retval;
619
620 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400621 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200622 case CFI_CMDSET_INTEL_STANDARD:
623 case CFI_CMDSET_INTEL_EXTENDED:
624 retval = !flash_isset (info, sect, 0, FLASH_STATUS_DONE);
625 break;
626 case CFI_CMDSET_AMD_STANDARD:
627 case CFI_CMDSET_AMD_EXTENDED:
Michael Schwingen81b20cc2007-12-07 23:35:02 +0100628#ifdef CONFIG_FLASH_CFI_LEGACY
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200629 case CFI_CMDSET_AMD_LEGACY:
630#endif
631 retval = flash_toggle (info, sect, 0, AMD_STATUS_TOGGLE);
632 break;
633 default:
634 retval = 0;
Michael Schwingen81b20cc2007-12-07 23:35:02 +0100635 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200636 debug ("flash_is_busy: %d\n", retval);
637 return retval;
Michael Schwingen81b20cc2007-12-07 23:35:02 +0100638}
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200639
640/*-----------------------------------------------------------------------
641 * wait for XSR.7 to be set. Time out with an error if it does not.
642 * This routine does not set the flash to read-array mode.
643 */
644static int flash_status_check (flash_info_t * info, flash_sect_t sector,
645 ulong tout, char *prompt)
646{
647 ulong start;
648
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200649#if CONFIG_SYS_HZ != 1000
650 tout *= CONFIG_SYS_HZ/1000;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200651#endif
652
653 /* Wait for command completion */
654 start = get_timer (0);
655 while (flash_is_busy (info, sector)) {
656 if (get_timer (start) > tout) {
657 printf ("Flash %s timeout at address %lx data %lx\n",
658 prompt, info->start[sector],
659 flash_read_long (info, sector, 0));
660 flash_write_cmd (info, sector, 0, info->cmd_reset);
661 return ERR_TIMOUT;
662 }
663 udelay (1); /* also triggers watchdog */
664 }
665 return ERR_OK;
666}
667
668/*-----------------------------------------------------------------------
669 * Wait for XSR.7 to be set, if it times out print an error, otherwise
670 * do a full status check.
671 *
672 * This routine sets the flash to read-array mode.
673 */
674static int flash_full_status_check (flash_info_t * info, flash_sect_t sector,
675 ulong tout, char *prompt)
676{
677 int retcode;
678
679 retcode = flash_status_check (info, sector, tout, prompt);
680 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400681 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200682 case CFI_CMDSET_INTEL_EXTENDED:
683 case CFI_CMDSET_INTEL_STANDARD:
Ed Swarthout0d01f662008-10-09 01:26:36 -0500684 if ((retcode != ERR_OK)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200685 && !flash_isequal (info, sector, 0, FLASH_STATUS_DONE)) {
686 retcode = ERR_INVAL;
687 printf ("Flash %s error at address %lx\n", prompt,
688 info->start[sector]);
689 if (flash_isset (info, sector, 0, FLASH_STATUS_ECLBS |
690 FLASH_STATUS_PSLBS)) {
691 puts ("Command Sequence Error.\n");
692 } else if (flash_isset (info, sector, 0,
693 FLASH_STATUS_ECLBS)) {
694 puts ("Block Erase Error.\n");
695 retcode = ERR_NOT_ERASED;
696 } else if (flash_isset (info, sector, 0,
697 FLASH_STATUS_PSLBS)) {
698 puts ("Locking Error\n");
699 }
700 if (flash_isset (info, sector, 0, FLASH_STATUS_DPS)) {
701 puts ("Block locked.\n");
702 retcode = ERR_PROTECTED;
703 }
704 if (flash_isset (info, sector, 0, FLASH_STATUS_VPENS))
705 puts ("Vpp Low Error.\n");
706 }
707 flash_write_cmd (info, sector, 0, info->cmd_reset);
708 break;
709 default:
710 break;
711 }
712 return retcode;
713}
714
715/*-----------------------------------------------------------------------
716 */
717static void flash_add_byte (flash_info_t * info, cfiword_t * cword, uchar c)
718{
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200719#if defined(__LITTLE_ENDIAN) && !defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200720 unsigned short w;
721 unsigned int l;
722 unsigned long long ll;
723#endif
724
725 switch (info->portwidth) {
726 case FLASH_CFI_8BIT:
727 cword->c = c;
728 break;
729 case FLASH_CFI_16BIT:
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200730#if defined(__LITTLE_ENDIAN) && !defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200731 w = c;
732 w <<= 8;
733 cword->w = (cword->w >> 8) | w;
Michael Schwingen81b20cc2007-12-07 23:35:02 +0100734#else
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200735 cword->w = (cword->w << 8) | c;
Michael Schwingen81b20cc2007-12-07 23:35:02 +0100736#endif
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200737 break;
738 case FLASH_CFI_32BIT:
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200739#if defined(__LITTLE_ENDIAN) && !defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200740 l = c;
741 l <<= 24;
742 cword->l = (cword->l >> 8) | l;
743#else
744 cword->l = (cword->l << 8) | c;
Stefan Roese2662b402006-04-01 13:41:03 +0200745#endif
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200746 break;
747 case FLASH_CFI_64BIT:
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200748#if defined(__LITTLE_ENDIAN) && !defined(CONFIG_SYS_WRITE_SWAPPED_DATA)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200749 ll = c;
750 ll <<= 56;
751 cword->ll = (cword->ll >> 8) | ll;
752#else
753 cword->ll = (cword->ll << 8) | c;
754#endif
755 break;
wdenk5653fc32004-02-08 22:55:38 +0000756 }
wdenk5653fc32004-02-08 22:55:38 +0000757}
758
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200759/* loop through the sectors from the highest address when the passed
760 * address is greater or equal to the sector address we have a match
wdenk5653fc32004-02-08 22:55:38 +0000761 */
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200762static flash_sect_t find_sector (flash_info_t * info, ulong addr)
wdenk7680c142005-05-16 15:23:22 +0000763{
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200764 flash_sect_t sector;
wdenk7680c142005-05-16 15:23:22 +0000765
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200766 for (sector = info->sector_count - 1; sector >= 0; sector--) {
767 if (addr >= info->start[sector])
wdenk7680c142005-05-16 15:23:22 +0000768 break;
769 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200770 return sector;
wdenk7680c142005-05-16 15:23:22 +0000771}
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200772
773/*-----------------------------------------------------------------------
774 */
775static int flash_write_cfiword (flash_info_t * info, ulong dest,
776 cfiword_t cword)
777{
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100778 void *dstaddr;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200779 int flag;
Ed Swarthout0d01f662008-10-09 01:26:36 -0500780 flash_sect_t sect;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200781
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100782 dstaddr = map_physmem(dest, info->portwidth, MAP_NOCACHE);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200783
784 /* Check if Flash is (sufficiently) erased */
785 switch (info->portwidth) {
786 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100787 flag = ((flash_read8(dstaddr) & cword.c) == cword.c);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200788 break;
789 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100790 flag = ((flash_read16(dstaddr) & cword.w) == cword.w);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200791 break;
792 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100793 flag = ((flash_read32(dstaddr) & cword.l) == cword.l);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200794 break;
795 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100796 flag = ((flash_read64(dstaddr) & cword.ll) == cword.ll);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200797 break;
798 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100799 flag = 0;
800 break;
801 }
802 if (!flag) {
803 unmap_physmem(dstaddr, info->portwidth);
Stefan Roese0dc80e22007-12-27 07:50:54 +0100804 return ERR_NOT_ERASED;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200805 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200806
807 /* Disable interrupts which might cause a timeout here */
808 flag = disable_interrupts ();
809
810 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400811 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200812 case CFI_CMDSET_INTEL_EXTENDED:
813 case CFI_CMDSET_INTEL_STANDARD:
814 flash_write_cmd (info, 0, 0, FLASH_CMD_CLEAR_STATUS);
815 flash_write_cmd (info, 0, 0, FLASH_CMD_WRITE);
816 break;
817 case CFI_CMDSET_AMD_EXTENDED:
818 case CFI_CMDSET_AMD_STANDARD:
819#ifdef CONFIG_FLASH_CFI_LEGACY
820 case CFI_CMDSET_AMD_LEGACY:
Wolfgang Denk080bdb72005-10-05 01:51:29 +0200821#endif
Ed Swarthout0d01f662008-10-09 01:26:36 -0500822 sect = find_sector(info, dest);
823 flash_unlock_seq (info, sect);
824 flash_write_cmd (info, sect, info->addr_unlock1, AMD_CMD_WRITE);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200825 break;
826 }
827
828 switch (info->portwidth) {
829 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100830 flash_write8(cword.c, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200831 break;
832 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100833 flash_write16(cword.w, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200834 break;
835 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100836 flash_write32(cword.l, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200837 break;
838 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100839 flash_write64(cword.ll, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200840 break;
841 }
842
843 /* re-enable interrupts if necessary */
844 if (flag)
845 enable_interrupts ();
846
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100847 unmap_physmem(dstaddr, info->portwidth);
848
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200849 return flash_full_status_check (info, find_sector (info, dest),
850 info->write_tout, "write");
851}
852
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200853#ifdef CONFIG_SYS_FLASH_USE_BUFFER_WRITE
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200854
855static int flash_write_cfibuffer (flash_info_t * info, ulong dest, uchar * cp,
856 int len)
857{
858 flash_sect_t sector;
859 int cnt;
860 int retcode;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100861 void *src = cp;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100862 void *dst = map_physmem(dest, len, MAP_NOCACHE);
Stefan Roese0dc80e22007-12-27 07:50:54 +0100863 void *dst2 = dst;
864 int flag = 0;
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200865 uint offset = 0;
866 unsigned int shift;
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400867 uchar write_cmd;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100868
Stefan Roese0dc80e22007-12-27 07:50:54 +0100869 switch (info->portwidth) {
870 case FLASH_CFI_8BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200871 shift = 0;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100872 break;
873 case FLASH_CFI_16BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200874 shift = 1;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100875 break;
876 case FLASH_CFI_32BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200877 shift = 2;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100878 break;
879 case FLASH_CFI_64BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200880 shift = 3;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100881 break;
882 default:
883 retcode = ERR_INVAL;
884 goto out_unmap;
885 }
886
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200887 cnt = len >> shift;
888
Stefan Roese0dc80e22007-12-27 07:50:54 +0100889 while ((cnt-- > 0) && (flag == 0)) {
890 switch (info->portwidth) {
891 case FLASH_CFI_8BIT:
892 flag = ((flash_read8(dst2) & flash_read8(src)) ==
893 flash_read8(src));
894 src += 1, dst2 += 1;
895 break;
896 case FLASH_CFI_16BIT:
897 flag = ((flash_read16(dst2) & flash_read16(src)) ==
898 flash_read16(src));
899 src += 2, dst2 += 2;
900 break;
901 case FLASH_CFI_32BIT:
902 flag = ((flash_read32(dst2) & flash_read32(src)) ==
903 flash_read32(src));
904 src += 4, dst2 += 4;
905 break;
906 case FLASH_CFI_64BIT:
907 flag = ((flash_read64(dst2) & flash_read64(src)) ==
908 flash_read64(src));
909 src += 8, dst2 += 8;
910 break;
911 }
912 }
913 if (!flag) {
914 retcode = ERR_NOT_ERASED;
915 goto out_unmap;
916 }
917
918 src = cp;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100919 sector = find_sector (info, dest);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200920
921 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400922 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200923 case CFI_CMDSET_INTEL_STANDARD:
924 case CFI_CMDSET_INTEL_EXTENDED:
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400925 write_cmd = (info->vendor == CFI_CMDSET_INTEL_PROG_REGIONS) ?
926 FLASH_CMD_WRITE_BUFFER_PROG : FLASH_CMD_WRITE_TO_BUFFER;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200927 flash_write_cmd (info, sector, 0, FLASH_CMD_CLEAR_STATUS);
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400928 flash_write_cmd (info, sector, 0, FLASH_CMD_READ_STATUS);
929 flash_write_cmd (info, sector, 0, write_cmd);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200930 retcode = flash_status_check (info, sector,
931 info->buffer_write_tout,
932 "write to buffer");
933 if (retcode == ERR_OK) {
934 /* reduce the number of loops by the width of
935 * the port */
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200936 cnt = len >> shift;
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400937 flash_write_cmd (info, sector, 0, cnt - 1);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200938 while (cnt-- > 0) {
939 switch (info->portwidth) {
940 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100941 flash_write8(flash_read8(src), dst);
942 src += 1, dst += 1;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200943 break;
944 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100945 flash_write16(flash_read16(src), dst);
946 src += 2, dst += 2;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200947 break;
948 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100949 flash_write32(flash_read32(src), dst);
950 src += 4, dst += 4;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200951 break;
952 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100953 flash_write64(flash_read64(src), dst);
954 src += 8, dst += 8;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200955 break;
956 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100957 retcode = ERR_INVAL;
958 goto out_unmap;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200959 }
960 }
961 flash_write_cmd (info, sector, 0,
962 FLASH_CMD_WRITE_BUFFER_CONFIRM);
963 retcode = flash_full_status_check (
964 info, sector, info->buffer_write_tout,
965 "buffer write");
966 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100967
968 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200969
970 case CFI_CMDSET_AMD_STANDARD:
971 case CFI_CMDSET_AMD_EXTENDED:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200972 flash_unlock_seq(info,0);
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200973
974#ifdef CONFIG_FLASH_SPANSION_S29WS_N
975 offset = ((unsigned long)dst - info->start[sector]) >> shift;
976#endif
977 flash_write_cmd(info, sector, offset, AMD_CMD_WRITE_TO_BUFFER);
978 cnt = len >> shift;
979 flash_write_cmd(info, sector, offset, (uchar)cnt - 1);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200980
981 switch (info->portwidth) {
982 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100983 while (cnt-- > 0) {
984 flash_write8(flash_read8(src), dst);
985 src += 1, dst += 1;
986 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200987 break;
988 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100989 while (cnt-- > 0) {
990 flash_write16(flash_read16(src), dst);
991 src += 2, dst += 2;
992 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200993 break;
994 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100995 while (cnt-- > 0) {
996 flash_write32(flash_read32(src), dst);
997 src += 4, dst += 4;
998 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200999 break;
1000 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +01001001 while (cnt-- > 0) {
1002 flash_write64(flash_read64(src), dst);
1003 src += 8, dst += 8;
1004 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001005 break;
1006 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001007 retcode = ERR_INVAL;
1008 goto out_unmap;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001009 }
1010
1011 flash_write_cmd (info, sector, 0, AMD_CMD_WRITE_BUFFER_CONFIRM);
1012 retcode = flash_full_status_check (info, sector,
1013 info->buffer_write_tout,
1014 "buffer write");
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001015 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001016
1017 default:
1018 debug ("Unknown Command Set\n");
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001019 retcode = ERR_INVAL;
1020 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001021 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001022
1023out_unmap:
1024 unmap_physmem(dst, len);
1025 return retcode;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001026}
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001027#endif /* CONFIG_SYS_FLASH_USE_BUFFER_WRITE */
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001028
wdenk7680c142005-05-16 15:23:22 +00001029
1030/*-----------------------------------------------------------------------
1031 */
wdenkbf9e3b32004-02-12 00:47:09 +00001032int flash_erase (flash_info_t * info, int s_first, int s_last)
wdenk5653fc32004-02-08 22:55:38 +00001033{
1034 int rcode = 0;
1035 int prot;
1036 flash_sect_t sect;
1037
wdenkbf9e3b32004-02-12 00:47:09 +00001038 if (info->flash_id != FLASH_MAN_CFI) {
wdenk4b9206e2004-03-23 22:14:11 +00001039 puts ("Can't erase unknown flash type - aborted\n");
wdenk5653fc32004-02-08 22:55:38 +00001040 return 1;
1041 }
1042 if ((s_first < 0) || (s_first > s_last)) {
wdenk4b9206e2004-03-23 22:14:11 +00001043 puts ("- no sectors to erase\n");
wdenk5653fc32004-02-08 22:55:38 +00001044 return 1;
1045 }
1046
1047 prot = 0;
wdenkbf9e3b32004-02-12 00:47:09 +00001048 for (sect = s_first; sect <= s_last; ++sect) {
wdenk5653fc32004-02-08 22:55:38 +00001049 if (info->protect[sect]) {
1050 prot++;
1051 }
1052 }
1053 if (prot) {
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001054 printf ("- Warning: %d protected sectors will not be erased!\n",
1055 prot);
wdenk5653fc32004-02-08 22:55:38 +00001056 } else {
wdenk4b9206e2004-03-23 22:14:11 +00001057 putc ('\n');
wdenk5653fc32004-02-08 22:55:38 +00001058 }
1059
1060
wdenkbf9e3b32004-02-12 00:47:09 +00001061 for (sect = s_first; sect <= s_last; sect++) {
wdenk5653fc32004-02-08 22:55:38 +00001062 if (info->protect[sect] == 0) { /* not protected */
wdenkbf9e3b32004-02-12 00:47:09 +00001063 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001064 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenk5653fc32004-02-08 22:55:38 +00001065 case CFI_CMDSET_INTEL_STANDARD:
1066 case CFI_CMDSET_INTEL_EXTENDED:
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001067 flash_write_cmd (info, sect, 0,
1068 FLASH_CMD_CLEAR_STATUS);
1069 flash_write_cmd (info, sect, 0,
1070 FLASH_CMD_BLOCK_ERASE);
1071 flash_write_cmd (info, sect, 0,
1072 FLASH_CMD_ERASE_CONFIRM);
wdenk5653fc32004-02-08 22:55:38 +00001073 break;
1074 case CFI_CMDSET_AMD_STANDARD:
1075 case CFI_CMDSET_AMD_EXTENDED:
wdenkbf9e3b32004-02-12 00:47:09 +00001076 flash_unlock_seq (info, sect);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001077 flash_write_cmd (info, sect,
1078 info->addr_unlock1,
1079 AMD_CMD_ERASE_START);
wdenkbf9e3b32004-02-12 00:47:09 +00001080 flash_unlock_seq (info, sect);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001081 flash_write_cmd (info, sect, 0,
1082 AMD_CMD_ERASE_SECTOR);
wdenk5653fc32004-02-08 22:55:38 +00001083 break;
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001084#ifdef CONFIG_FLASH_CFI_LEGACY
1085 case CFI_CMDSET_AMD_LEGACY:
1086 flash_unlock_seq (info, 0);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001087 flash_write_cmd (info, 0, info->addr_unlock1,
1088 AMD_CMD_ERASE_START);
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001089 flash_unlock_seq (info, 0);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001090 flash_write_cmd (info, sect, 0,
1091 AMD_CMD_ERASE_SECTOR);
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001092 break;
1093#endif
wdenk5653fc32004-02-08 22:55:38 +00001094 default:
wdenkbf9e3b32004-02-12 00:47:09 +00001095 debug ("Unkown flash vendor %d\n",
1096 info->vendor);
wdenk5653fc32004-02-08 22:55:38 +00001097 break;
1098 }
1099
wdenkbf9e3b32004-02-12 00:47:09 +00001100 if (flash_full_status_check
1101 (info, sect, info->erase_blk_tout, "erase")) {
wdenk5653fc32004-02-08 22:55:38 +00001102 rcode = 1;
1103 } else
wdenk4b9206e2004-03-23 22:14:11 +00001104 putc ('.');
wdenk5653fc32004-02-08 22:55:38 +00001105 }
1106 }
wdenk4b9206e2004-03-23 22:14:11 +00001107 puts (" done\n");
wdenk5653fc32004-02-08 22:55:38 +00001108 return rcode;
1109}
1110
1111/*-----------------------------------------------------------------------
1112 */
wdenkbf9e3b32004-02-12 00:47:09 +00001113void flash_print_info (flash_info_t * info)
wdenk5653fc32004-02-08 22:55:38 +00001114{
1115 int i;
1116
1117 if (info->flash_id != FLASH_MAN_CFI) {
wdenk4b9206e2004-03-23 22:14:11 +00001118 puts ("missing or unknown FLASH type\n");
wdenk5653fc32004-02-08 22:55:38 +00001119 return;
1120 }
1121
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001122 printf ("%s FLASH (%d x %d)",
1123 info->name,
wdenkbf9e3b32004-02-12 00:47:09 +00001124 (info->portwidth << 3), (info->chipwidth << 3));
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001125 if (info->size < 1024*1024)
1126 printf (" Size: %ld kB in %d Sectors\n",
1127 info->size >> 10, info->sector_count);
1128 else
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001129 printf (" Size: %ld MB in %d Sectors\n",
1130 info->size >> 20, info->sector_count);
Stefan Roese260421a2006-11-13 13:55:24 +01001131 printf (" ");
1132 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001133 case CFI_CMDSET_INTEL_PROG_REGIONS:
1134 printf ("Intel Prog Regions");
1135 break;
Stefan Roese260421a2006-11-13 13:55:24 +01001136 case CFI_CMDSET_INTEL_STANDARD:
1137 printf ("Intel Standard");
1138 break;
1139 case CFI_CMDSET_INTEL_EXTENDED:
1140 printf ("Intel Extended");
1141 break;
1142 case CFI_CMDSET_AMD_STANDARD:
1143 printf ("AMD Standard");
1144 break;
1145 case CFI_CMDSET_AMD_EXTENDED:
1146 printf ("AMD Extended");
1147 break;
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001148#ifdef CONFIG_FLASH_CFI_LEGACY
1149 case CFI_CMDSET_AMD_LEGACY:
1150 printf ("AMD Legacy");
1151 break;
1152#endif
Stefan Roese260421a2006-11-13 13:55:24 +01001153 default:
1154 printf ("Unknown (%d)", info->vendor);
1155 break;
1156 }
1157 printf (" command set, Manufacturer ID: 0x%02X, Device ID: 0x%02X",
1158 info->manufacturer_id, info->device_id);
1159 if (info->device_id == 0x7E) {
1160 printf("%04X", info->device_id2);
1161 }
1162 printf ("\n Erase timeout: %ld ms, write timeout: %ld ms\n",
wdenk028ab6b2004-02-23 23:54:43 +00001163 info->erase_blk_tout,
Stefan Roese260421a2006-11-13 13:55:24 +01001164 info->write_tout);
1165 if (info->buffer_size > 1) {
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001166 printf (" Buffer write timeout: %ld ms, "
1167 "buffer size: %d bytes\n",
wdenk028ab6b2004-02-23 23:54:43 +00001168 info->buffer_write_tout,
1169 info->buffer_size);
Stefan Roese260421a2006-11-13 13:55:24 +01001170 }
wdenk5653fc32004-02-08 22:55:38 +00001171
Stefan Roese260421a2006-11-13 13:55:24 +01001172 puts ("\n Sector Start Addresses:");
wdenkbf9e3b32004-02-12 00:47:09 +00001173 for (i = 0; i < info->sector_count; ++i) {
Stefan Roese260421a2006-11-13 13:55:24 +01001174 if ((i % 5) == 0)
1175 printf ("\n");
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001176#ifdef CONFIG_SYS_FLASH_EMPTY_INFO
wdenk5653fc32004-02-08 22:55:38 +00001177 int k;
1178 int size;
1179 int erased;
1180 volatile unsigned long *flash;
1181
1182 /*
1183 * Check if whole sector is erased
1184 */
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001185 size = flash_sector_size(info, i);
wdenk5653fc32004-02-08 22:55:38 +00001186 erased = 1;
wdenkbf9e3b32004-02-12 00:47:09 +00001187 flash = (volatile unsigned long *) info->start[i];
1188 size = size >> 2; /* divide by 4 for longword access */
1189 for (k = 0; k < size; k++) {
1190 if (*flash++ != 0xffffffff) {
1191 erased = 0;
1192 break;
1193 }
1194 }
wdenk5653fc32004-02-08 22:55:38 +00001195
wdenk5653fc32004-02-08 22:55:38 +00001196 /* print empty and read-only info */
Stefan Roese260421a2006-11-13 13:55:24 +01001197 printf (" %08lX %c %s ",
wdenk5653fc32004-02-08 22:55:38 +00001198 info->start[i],
Stefan Roese260421a2006-11-13 13:55:24 +01001199 erased ? 'E' : ' ',
1200 info->protect[i] ? "RO" : " ");
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001201#else /* ! CONFIG_SYS_FLASH_EMPTY_INFO */
Stefan Roese260421a2006-11-13 13:55:24 +01001202 printf (" %08lX %s ",
1203 info->start[i],
1204 info->protect[i] ? "RO" : " ");
wdenk5653fc32004-02-08 22:55:38 +00001205#endif
1206 }
wdenk4b9206e2004-03-23 22:14:11 +00001207 putc ('\n');
wdenk5653fc32004-02-08 22:55:38 +00001208 return;
1209}
1210
1211/*-----------------------------------------------------------------------
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001212 * This is used in a few places in write_buf() to show programming
1213 * progress. Making it a function is nasty because it needs to do side
1214 * effect updates to digit and dots. Repeated code is nasty too, so
1215 * we define it once here.
1216 */
Stefan Roesef0105722008-03-19 07:09:26 +01001217#ifdef CONFIG_FLASH_SHOW_PROGRESS
1218#define FLASH_SHOW_PROGRESS(scale, dots, digit, dots_sub) \
1219 dots -= dots_sub; \
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001220 if ((scale > 0) && (dots <= 0)) { \
1221 if ((digit % 5) == 0) \
1222 printf ("%d", digit / 5); \
1223 else \
1224 putc ('.'); \
1225 digit--; \
1226 dots += scale; \
1227 }
Stefan Roesef0105722008-03-19 07:09:26 +01001228#else
1229#define FLASH_SHOW_PROGRESS(scale, dots, digit, dots_sub)
1230#endif
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001231
1232/*-----------------------------------------------------------------------
wdenk5653fc32004-02-08 22:55:38 +00001233 * Copy memory to flash, returns:
1234 * 0 - OK
1235 * 1 - write timeout
1236 * 2 - Flash not erased
1237 */
wdenkbf9e3b32004-02-12 00:47:09 +00001238int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt)
wdenk5653fc32004-02-08 22:55:38 +00001239{
1240 ulong wp;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001241 uchar *p;
wdenk5653fc32004-02-08 22:55:38 +00001242 int aln;
1243 cfiword_t cword;
1244 int i, rc;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001245#ifdef CONFIG_SYS_FLASH_USE_BUFFER_WRITE
wdenkbf9e3b32004-02-12 00:47:09 +00001246 int buffered_size;
1247#endif
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001248#ifdef CONFIG_FLASH_SHOW_PROGRESS
1249 int digit = CONFIG_FLASH_SHOW_PROGRESS;
1250 int scale = 0;
1251 int dots = 0;
1252
1253 /*
1254 * Suppress if there are fewer than CONFIG_FLASH_SHOW_PROGRESS writes.
1255 */
1256 if (cnt >= CONFIG_FLASH_SHOW_PROGRESS) {
1257 scale = (int)((cnt + CONFIG_FLASH_SHOW_PROGRESS - 1) /
1258 CONFIG_FLASH_SHOW_PROGRESS);
1259 }
1260#endif
1261
wdenkbf9e3b32004-02-12 00:47:09 +00001262 /* get lower aligned address */
wdenk5653fc32004-02-08 22:55:38 +00001263 wp = (addr & ~(info->portwidth - 1));
1264
1265 /* handle unaligned start */
wdenkbf9e3b32004-02-12 00:47:09 +00001266 if ((aln = addr - wp) != 0) {
wdenk5653fc32004-02-08 22:55:38 +00001267 cword.l = 0;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001268 p = map_physmem(wp, info->portwidth, MAP_NOCACHE);
1269 for (i = 0; i < aln; ++i)
1270 flash_add_byte (info, &cword, flash_read8(p + i));
wdenk5653fc32004-02-08 22:55:38 +00001271
wdenkbf9e3b32004-02-12 00:47:09 +00001272 for (; (i < info->portwidth) && (cnt > 0); i++) {
1273 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001274 cnt--;
wdenk5653fc32004-02-08 22:55:38 +00001275 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001276 for (; (cnt == 0) && (i < info->portwidth); ++i)
1277 flash_add_byte (info, &cword, flash_read8(p + i));
1278
1279 rc = flash_write_cfiword (info, wp, cword);
1280 unmap_physmem(p, info->portwidth);
1281 if (rc != 0)
wdenk5653fc32004-02-08 22:55:38 +00001282 return rc;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001283
1284 wp += i;
Stefan Roesef0105722008-03-19 07:09:26 +01001285 FLASH_SHOW_PROGRESS(scale, dots, digit, i);
wdenk5653fc32004-02-08 22:55:38 +00001286 }
1287
wdenkbf9e3b32004-02-12 00:47:09 +00001288 /* handle the aligned part */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001289#ifdef CONFIG_SYS_FLASH_USE_BUFFER_WRITE
wdenkbf9e3b32004-02-12 00:47:09 +00001290 buffered_size = (info->portwidth / info->chipwidth);
1291 buffered_size *= info->buffer_size;
1292 while (cnt >= info->portwidth) {
Stefan Roese79b4cda2006-02-28 15:29:58 +01001293 /* prohibit buffer write when buffer_size is 1 */
1294 if (info->buffer_size == 1) {
1295 cword.l = 0;
1296 for (i = 0; i < info->portwidth; i++)
1297 flash_add_byte (info, &cword, *src++);
1298 if ((rc = flash_write_cfiword (info, wp, cword)) != 0)
1299 return rc;
1300 wp += info->portwidth;
1301 cnt -= info->portwidth;
1302 continue;
1303 }
1304
1305 /* write buffer until next buffered_size aligned boundary */
1306 i = buffered_size - (wp % buffered_size);
1307 if (i > cnt)
1308 i = cnt;
wdenkbf9e3b32004-02-12 00:47:09 +00001309 if ((rc = flash_write_cfibuffer (info, wp, src, i)) != ERR_OK)
wdenk5653fc32004-02-08 22:55:38 +00001310 return rc;
Wolfgang Denk8d4ba3d2005-08-12 22:35:59 +02001311 i -= i & (info->portwidth - 1);
wdenk5653fc32004-02-08 22:55:38 +00001312 wp += i;
1313 src += i;
wdenkbf9e3b32004-02-12 00:47:09 +00001314 cnt -= i;
Stefan Roesef0105722008-03-19 07:09:26 +01001315 FLASH_SHOW_PROGRESS(scale, dots, digit, i);
wdenk5653fc32004-02-08 22:55:38 +00001316 }
1317#else
wdenkbf9e3b32004-02-12 00:47:09 +00001318 while (cnt >= info->portwidth) {
wdenk5653fc32004-02-08 22:55:38 +00001319 cword.l = 0;
wdenkbf9e3b32004-02-12 00:47:09 +00001320 for (i = 0; i < info->portwidth; i++) {
1321 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001322 }
wdenkbf9e3b32004-02-12 00:47:09 +00001323 if ((rc = flash_write_cfiword (info, wp, cword)) != 0)
wdenk5653fc32004-02-08 22:55:38 +00001324 return rc;
1325 wp += info->portwidth;
1326 cnt -= info->portwidth;
Stefan Roesef0105722008-03-19 07:09:26 +01001327 FLASH_SHOW_PROGRESS(scale, dots, digit, info->portwidth);
wdenk5653fc32004-02-08 22:55:38 +00001328 }
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001329#endif /* CONFIG_SYS_FLASH_USE_BUFFER_WRITE */
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001330
wdenk5653fc32004-02-08 22:55:38 +00001331 if (cnt == 0) {
1332 return (0);
1333 }
1334
1335 /*
1336 * handle unaligned tail bytes
1337 */
1338 cword.l = 0;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001339 p = map_physmem(wp, info->portwidth, MAP_NOCACHE);
1340 for (i = 0; (i < info->portwidth) && (cnt > 0); ++i) {
wdenkbf9e3b32004-02-12 00:47:09 +00001341 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001342 --cnt;
1343 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001344 for (; i < info->portwidth; ++i)
1345 flash_add_byte (info, &cword, flash_read8(p + i));
1346 unmap_physmem(p, info->portwidth);
wdenk5653fc32004-02-08 22:55:38 +00001347
wdenkbf9e3b32004-02-12 00:47:09 +00001348 return flash_write_cfiword (info, wp, cword);
wdenk5653fc32004-02-08 22:55:38 +00001349}
1350
1351/*-----------------------------------------------------------------------
1352 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001353#ifdef CONFIG_SYS_FLASH_PROTECTION
wdenk5653fc32004-02-08 22:55:38 +00001354
wdenkbf9e3b32004-02-12 00:47:09 +00001355int flash_real_protect (flash_info_t * info, long sector, int prot)
wdenk5653fc32004-02-08 22:55:38 +00001356{
1357 int retcode = 0;
1358
Rafael Camposbc9019e2008-07-31 10:22:20 +02001359 switch (info->vendor) {
1360 case CFI_CMDSET_INTEL_PROG_REGIONS:
1361 case CFI_CMDSET_INTEL_STANDARD:
Nick Spence9e8e63c2008-08-19 22:21:16 -07001362 case CFI_CMDSET_INTEL_EXTENDED:
Rafael Camposbc9019e2008-07-31 10:22:20 +02001363 flash_write_cmd (info, sector, 0,
1364 FLASH_CMD_CLEAR_STATUS);
1365 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT);
1366 if (prot)
1367 flash_write_cmd (info, sector, 0,
1368 FLASH_CMD_PROTECT_SET);
1369 else
1370 flash_write_cmd (info, sector, 0,
1371 FLASH_CMD_PROTECT_CLEAR);
1372 break;
1373 case CFI_CMDSET_AMD_EXTENDED:
1374 case CFI_CMDSET_AMD_STANDARD:
Rafael Camposbc9019e2008-07-31 10:22:20 +02001375 /* U-Boot only checks the first byte */
1376 if (info->manufacturer_id == (uchar)ATM_MANUFACT) {
1377 if (prot) {
1378 flash_unlock_seq (info, 0);
1379 flash_write_cmd (info, 0,
1380 info->addr_unlock1,
1381 ATM_CMD_SOFTLOCK_START);
1382 flash_unlock_seq (info, 0);
1383 flash_write_cmd (info, sector, 0,
1384 ATM_CMD_LOCK_SECT);
1385 } else {
1386 flash_write_cmd (info, 0,
1387 info->addr_unlock1,
1388 AMD_CMD_UNLOCK_START);
1389 if (info->device_id == ATM_ID_BV6416)
1390 flash_write_cmd (info, sector,
1391 0, ATM_CMD_UNLOCK_SECT);
1392 }
1393 }
1394 break;
TsiChung Liew4e00acd2008-08-19 16:53:39 +00001395#ifdef CONFIG_FLASH_CFI_LEGACY
1396 case CFI_CMDSET_AMD_LEGACY:
1397 flash_write_cmd (info, sector, 0, FLASH_CMD_CLEAR_STATUS);
1398 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT);
1399 if (prot)
1400 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT_SET);
1401 else
1402 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT_CLEAR);
1403#endif
Rafael Camposbc9019e2008-07-31 10:22:20 +02001404 };
wdenk5653fc32004-02-08 22:55:38 +00001405
wdenkbf9e3b32004-02-12 00:47:09 +00001406 if ((retcode =
1407 flash_full_status_check (info, sector, info->erase_blk_tout,
1408 prot ? "protect" : "unprotect")) == 0) {
wdenk5653fc32004-02-08 22:55:38 +00001409
1410 info->protect[sector] = prot;
Stefan Roese2662b402006-04-01 13:41:03 +02001411
1412 /*
1413 * On some of Intel's flash chips (marked via legacy_unlock)
1414 * unprotect unprotects all locking.
1415 */
1416 if ((prot == 0) && (info->legacy_unlock)) {
wdenk5653fc32004-02-08 22:55:38 +00001417 flash_sect_t i;
wdenkbf9e3b32004-02-12 00:47:09 +00001418
1419 for (i = 0; i < info->sector_count; i++) {
1420 if (info->protect[i])
1421 flash_real_protect (info, i, 1);
wdenk5653fc32004-02-08 22:55:38 +00001422 }
1423 }
1424 }
wdenk5653fc32004-02-08 22:55:38 +00001425 return retcode;
wdenkbf9e3b32004-02-12 00:47:09 +00001426}
1427
wdenk5653fc32004-02-08 22:55:38 +00001428/*-----------------------------------------------------------------------
1429 * flash_read_user_serial - read the OneTimeProgramming cells
1430 */
wdenkbf9e3b32004-02-12 00:47:09 +00001431void flash_read_user_serial (flash_info_t * info, void *buffer, int offset,
1432 int len)
wdenk5653fc32004-02-08 22:55:38 +00001433{
wdenkbf9e3b32004-02-12 00:47:09 +00001434 uchar *src;
1435 uchar *dst;
wdenk5653fc32004-02-08 22:55:38 +00001436
1437 dst = buffer;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001438 src = flash_map (info, 0, FLASH_OFFSET_USER_PROTECTION);
wdenkbf9e3b32004-02-12 00:47:09 +00001439 flash_write_cmd (info, 0, 0, FLASH_CMD_READ_ID);
1440 memcpy (dst, src + offset, len);
Wolfgang Denkdb421e62005-09-25 16:41:22 +02001441 flash_write_cmd (info, 0, 0, info->cmd_reset);
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001442 flash_unmap(info, 0, FLASH_OFFSET_USER_PROTECTION, src);
wdenk5653fc32004-02-08 22:55:38 +00001443}
wdenkbf9e3b32004-02-12 00:47:09 +00001444
wdenk5653fc32004-02-08 22:55:38 +00001445/*
1446 * flash_read_factory_serial - read the device Id from the protection area
1447 */
wdenkbf9e3b32004-02-12 00:47:09 +00001448void flash_read_factory_serial (flash_info_t * info, void *buffer, int offset,
1449 int len)
wdenk5653fc32004-02-08 22:55:38 +00001450{
wdenkbf9e3b32004-02-12 00:47:09 +00001451 uchar *src;
wdenkcd37d9e2004-02-10 00:03:41 +00001452
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001453 src = flash_map (info, 0, FLASH_OFFSET_INTEL_PROTECTION);
wdenkbf9e3b32004-02-12 00:47:09 +00001454 flash_write_cmd (info, 0, 0, FLASH_CMD_READ_ID);
1455 memcpy (buffer, src + offset, len);
Wolfgang Denkdb421e62005-09-25 16:41:22 +02001456 flash_write_cmd (info, 0, 0, info->cmd_reset);
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001457 flash_unmap(info, 0, FLASH_OFFSET_INTEL_PROTECTION, src);
wdenk5653fc32004-02-08 22:55:38 +00001458}
1459
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001460#endif /* CONFIG_SYS_FLASH_PROTECTION */
wdenk5653fc32004-02-08 22:55:38 +00001461
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001462/*-----------------------------------------------------------------------
1463 * Reverse the order of the erase regions in the CFI QRY structure.
1464 * This is needed for chips that are either a) correctly detected as
1465 * top-boot, or b) buggy.
1466 */
1467static void cfi_reverse_geometry(struct cfi_qry *qry)
1468{
1469 unsigned int i, j;
1470 u32 tmp;
1471
1472 for (i = 0, j = qry->num_erase_regions - 1; i < j; i++, j--) {
1473 tmp = qry->erase_region_info[i];
1474 qry->erase_region_info[i] = qry->erase_region_info[j];
1475 qry->erase_region_info[j] = tmp;
1476 }
1477}
wdenk5653fc32004-02-08 22:55:38 +00001478
1479/*-----------------------------------------------------------------------
Stefan Roese260421a2006-11-13 13:55:24 +01001480 * read jedec ids from device and set corresponding fields in info struct
1481 *
1482 * Note: assume cfi->vendor, cfi->portwidth and cfi->chipwidth are correct
1483 *
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001484 */
1485static void cmdset_intel_read_jedec_ids(flash_info_t *info)
1486{
1487 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1488 flash_write_cmd(info, 0, 0, FLASH_CMD_READ_ID);
1489 udelay(1000); /* some flash are slow to respond */
1490 info->manufacturer_id = flash_read_uchar (info,
1491 FLASH_OFFSET_MANUFACTURER_ID);
1492 info->device_id = flash_read_uchar (info,
1493 FLASH_OFFSET_DEVICE_ID);
1494 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1495}
1496
1497static int cmdset_intel_init(flash_info_t *info, struct cfi_qry *qry)
1498{
1499 info->cmd_reset = FLASH_CMD_RESET;
1500
1501 cmdset_intel_read_jedec_ids(info);
1502 flash_write_cmd(info, 0, info->cfi_offset, FLASH_CMD_CFI);
1503
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001504#ifdef CONFIG_SYS_FLASH_PROTECTION
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001505 /* read legacy lock/unlock bit from intel flash */
1506 if (info->ext_addr) {
1507 info->legacy_unlock = flash_read_uchar (info,
1508 info->ext_addr + 5) & 0x08;
1509 }
1510#endif
1511
1512 return 0;
1513}
1514
1515static void cmdset_amd_read_jedec_ids(flash_info_t *info)
1516{
1517 flash_write_cmd(info, 0, 0, AMD_CMD_RESET);
1518 flash_unlock_seq(info, 0);
1519 flash_write_cmd(info, 0, info->addr_unlock1, FLASH_CMD_READ_ID);
1520 udelay(1000); /* some flash are slow to respond */
Tor Krill90447ec2008-03-28 11:29:10 +01001521
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001522 info->manufacturer_id = flash_read_uchar (info,
1523 FLASH_OFFSET_MANUFACTURER_ID);
Tor Krill90447ec2008-03-28 11:29:10 +01001524
1525 switch (info->chipwidth){
1526 case FLASH_CFI_8BIT:
1527 info->device_id = flash_read_uchar (info,
1528 FLASH_OFFSET_DEVICE_ID);
1529 if (info->device_id == 0x7E) {
1530 /* AMD 3-byte (expanded) device ids */
1531 info->device_id2 = flash_read_uchar (info,
1532 FLASH_OFFSET_DEVICE_ID2);
1533 info->device_id2 <<= 8;
1534 info->device_id2 |= flash_read_uchar (info,
1535 FLASH_OFFSET_DEVICE_ID3);
1536 }
1537 break;
1538 case FLASH_CFI_16BIT:
1539 info->device_id = flash_read_word (info,
1540 FLASH_OFFSET_DEVICE_ID);
1541 break;
1542 default:
1543 break;
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001544 }
1545 flash_write_cmd(info, 0, 0, AMD_CMD_RESET);
1546}
1547
1548static int cmdset_amd_init(flash_info_t *info, struct cfi_qry *qry)
1549{
1550 info->cmd_reset = AMD_CMD_RESET;
1551
1552 cmdset_amd_read_jedec_ids(info);
1553 flash_write_cmd(info, 0, info->cfi_offset, FLASH_CMD_CFI);
1554
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001555 return 0;
1556}
1557
1558#ifdef CONFIG_FLASH_CFI_LEGACY
Stefan Roese260421a2006-11-13 13:55:24 +01001559static void flash_read_jedec_ids (flash_info_t * info)
1560{
1561 info->manufacturer_id = 0;
1562 info->device_id = 0;
1563 info->device_id2 = 0;
1564
1565 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001566 case CFI_CMDSET_INTEL_PROG_REGIONS:
Stefan Roese260421a2006-11-13 13:55:24 +01001567 case CFI_CMDSET_INTEL_STANDARD:
1568 case CFI_CMDSET_INTEL_EXTENDED:
Michael Schwingen8225d1e2008-01-12 20:29:47 +01001569 cmdset_intel_read_jedec_ids(info);
Stefan Roese260421a2006-11-13 13:55:24 +01001570 break;
1571 case CFI_CMDSET_AMD_STANDARD:
1572 case CFI_CMDSET_AMD_EXTENDED:
Michael Schwingen8225d1e2008-01-12 20:29:47 +01001573 cmdset_amd_read_jedec_ids(info);
Stefan Roese260421a2006-11-13 13:55:24 +01001574 break;
1575 default:
1576 break;
1577 }
1578}
1579
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001580/*-----------------------------------------------------------------------
1581 * Call board code to request info about non-CFI flash.
1582 * board_flash_get_legacy needs to fill in at least:
1583 * info->portwidth, info->chipwidth and info->interface for Jedec probing.
1584 */
1585static int flash_detect_legacy(ulong base, int banknum)
1586{
1587 flash_info_t *info = &flash_info[banknum];
1588
1589 if (board_flash_get_legacy(base, banknum, info)) {
1590 /* board code may have filled info completely. If not, we
1591 use JEDEC ID probing. */
1592 if (!info->vendor) {
1593 int modes[] = {
1594 CFI_CMDSET_AMD_STANDARD,
1595 CFI_CMDSET_INTEL_STANDARD
1596 };
1597 int i;
1598
1599 for (i = 0; i < sizeof(modes) / sizeof(modes[0]); i++) {
1600 info->vendor = modes[i];
1601 info->start[0] = base;
1602 if (info->portwidth == FLASH_CFI_8BIT
1603 && info->interface == FLASH_CFI_X8X16) {
1604 info->addr_unlock1 = 0x2AAA;
1605 info->addr_unlock2 = 0x5555;
1606 } else {
1607 info->addr_unlock1 = 0x5555;
1608 info->addr_unlock2 = 0x2AAA;
1609 }
1610 flash_read_jedec_ids(info);
1611 debug("JEDEC PROBE: ID %x %x %x\n",
1612 info->manufacturer_id,
1613 info->device_id,
1614 info->device_id2);
1615 if (jedec_flash_match(info, base))
1616 break;
1617 }
1618 }
1619
1620 switch(info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001621 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001622 case CFI_CMDSET_INTEL_STANDARD:
1623 case CFI_CMDSET_INTEL_EXTENDED:
1624 info->cmd_reset = FLASH_CMD_RESET;
1625 break;
1626 case CFI_CMDSET_AMD_STANDARD:
1627 case CFI_CMDSET_AMD_EXTENDED:
1628 case CFI_CMDSET_AMD_LEGACY:
1629 info->cmd_reset = AMD_CMD_RESET;
1630 break;
1631 }
1632 info->flash_id = FLASH_MAN_CFI;
1633 return 1;
1634 }
1635 return 0; /* use CFI */
1636}
1637#else
1638static inline int flash_detect_legacy(ulong base, int banknum)
1639{
1640 return 0; /* use CFI */
1641}
1642#endif
1643
Stefan Roese260421a2006-11-13 13:55:24 +01001644/*-----------------------------------------------------------------------
wdenk5653fc32004-02-08 22:55:38 +00001645 * detect if flash is compatible with the Common Flash Interface (CFI)
1646 * http://www.jedec.org/download/search/jesd68.pdf
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001647 */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001648static void flash_read_cfi (flash_info_t *info, void *buf,
1649 unsigned int start, size_t len)
1650{
1651 u8 *p = buf;
1652 unsigned int i;
1653
1654 for (i = 0; i < len; i++)
1655 p[i] = flash_read_uchar(info, start + i);
1656}
1657
1658static int __flash_detect_cfi (flash_info_t * info, struct cfi_qry *qry)
wdenk5653fc32004-02-08 22:55:38 +00001659{
Wolfgang Denk92eb7292006-12-27 01:26:13 +01001660 int cfi_offset;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001661
Michael Schwingen1ba639d2008-02-18 23:16:35 +01001662 /* We do not yet know what kind of commandset to use, so we issue
1663 the reset command in both Intel and AMD variants, in the hope
1664 that AMD flash roms ignore the Intel command. */
1665 flash_write_cmd (info, 0, 0, AMD_CMD_RESET);
1666 flash_write_cmd (info, 0, 0, FLASH_CMD_RESET);
1667
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001668 for (cfi_offset=0;
1669 cfi_offset < sizeof(flash_offset_cfi) / sizeof(uint);
1670 cfi_offset++) {
1671 flash_write_cmd (info, 0, flash_offset_cfi[cfi_offset],
1672 FLASH_CMD_CFI);
1673 if (flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP, 'Q')
1674 && flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP + 1, 'R')
1675 && flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP + 2, 'Y')) {
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001676 flash_read_cfi(info, qry, FLASH_OFFSET_CFI_RESP,
1677 sizeof(struct cfi_qry));
1678 info->interface = le16_to_cpu(qry->interface_desc);
1679
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001680 info->cfi_offset = flash_offset_cfi[cfi_offset];
1681 debug ("device interface is %d\n",
1682 info->interface);
1683 debug ("found port %d chip %d ",
1684 info->portwidth, info->chipwidth);
1685 debug ("port %d bits chip %d bits\n",
1686 info->portwidth << CFI_FLASH_SHIFT_WIDTH,
1687 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
1688
1689 /* calculate command offsets as in the Linux driver */
1690 info->addr_unlock1 = 0x555;
1691 info->addr_unlock2 = 0x2aa;
1692
1693 /*
1694 * modify the unlock address if we are
1695 * in compatibility mode
1696 */
1697 if ( /* x8/x16 in x8 mode */
1698 ((info->chipwidth == FLASH_CFI_BY8) &&
1699 (info->interface == FLASH_CFI_X8X16)) ||
1700 /* x16/x32 in x16 mode */
1701 ((info->chipwidth == FLASH_CFI_BY16) &&
1702 (info->interface == FLASH_CFI_X16X32)))
1703 {
1704 info->addr_unlock1 = 0xaaa;
1705 info->addr_unlock2 = 0x555;
1706 }
1707
1708 info->name = "CFI conformant";
1709 return 1;
1710 }
1711 }
1712
1713 return 0;
1714}
1715
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001716static int flash_detect_cfi (flash_info_t * info, struct cfi_qry *qry)
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001717{
wdenkbf9e3b32004-02-12 00:47:09 +00001718 debug ("flash detect cfi\n");
wdenk5653fc32004-02-08 22:55:38 +00001719
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001720 for (info->portwidth = CONFIG_SYS_FLASH_CFI_WIDTH;
wdenkbf9e3b32004-02-12 00:47:09 +00001721 info->portwidth <= FLASH_CFI_64BIT; info->portwidth <<= 1) {
1722 for (info->chipwidth = FLASH_CFI_BY8;
1723 info->chipwidth <= info->portwidth;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001724 info->chipwidth <<= 1)
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001725 if (__flash_detect_cfi(info, qry))
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001726 return 1;
wdenk5653fc32004-02-08 22:55:38 +00001727 }
wdenkbf9e3b32004-02-12 00:47:09 +00001728 debug ("not found\n");
wdenk5653fc32004-02-08 22:55:38 +00001729 return 0;
1730}
wdenkbf9e3b32004-02-12 00:47:09 +00001731
wdenk5653fc32004-02-08 22:55:38 +00001732/*
Haavard Skinnemoen467bcee2007-12-14 15:36:18 +01001733 * Manufacturer-specific quirks. Add workarounds for geometry
1734 * reversal, etc. here.
1735 */
1736static void flash_fixup_amd(flash_info_t *info, struct cfi_qry *qry)
1737{
1738 /* check if flash geometry needs reversal */
1739 if (qry->num_erase_regions > 1) {
1740 /* reverse geometry if top boot part */
1741 if (info->cfi_version < 0x3131) {
1742 /* CFI < 1.1, try to guess from device id */
1743 if ((info->device_id & 0x80) != 0)
1744 cfi_reverse_geometry(qry);
1745 } else if (flash_read_uchar(info, info->ext_addr + 0xf) == 3) {
1746 /* CFI >= 1.1, deduct from top/bottom flag */
1747 /* note: ext_addr is valid since cfi_version > 0 */
1748 cfi_reverse_geometry(qry);
1749 }
1750 }
1751}
1752
1753static void flash_fixup_atmel(flash_info_t *info, struct cfi_qry *qry)
1754{
1755 int reverse_geometry = 0;
1756
1757 /* Check the "top boot" bit in the PRI */
1758 if (info->ext_addr && !(flash_read_uchar(info, info->ext_addr + 6) & 1))
1759 reverse_geometry = 1;
1760
1761 /* AT49BV6416(T) list the erase regions in the wrong order.
1762 * However, the device ID is identical with the non-broken
1763 * AT49BV642D since u-boot only reads the low byte (they
1764 * differ in the high byte.) So leave out this fixup for now.
1765 */
1766#if 0
1767 if (info->device_id == 0xd6 || info->device_id == 0xd2)
1768 reverse_geometry = !reverse_geometry;
1769#endif
1770
1771 if (reverse_geometry)
1772 cfi_reverse_geometry(qry);
1773}
1774
1775/*
wdenk5653fc32004-02-08 22:55:38 +00001776 * The following code cannot be run from FLASH!
1777 *
1778 */
Marian Balakowicze6f2e902005-10-11 19:09:42 +02001779ulong flash_get_size (ulong base, int banknum)
wdenk5653fc32004-02-08 22:55:38 +00001780{
wdenkbf9e3b32004-02-12 00:47:09 +00001781 flash_info_t *info = &flash_info[banknum];
wdenk5653fc32004-02-08 22:55:38 +00001782 int i, j;
1783 flash_sect_t sect_cnt;
1784 unsigned long sector;
1785 unsigned long tmp;
1786 int size_ratio;
1787 uchar num_erase_regions;
wdenkbf9e3b32004-02-12 00:47:09 +00001788 int erase_region_size;
1789 int erase_region_count;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001790 struct cfi_qry qry;
Stefan Roese260421a2006-11-13 13:55:24 +01001791
Kumar Galaf9796902008-05-15 15:13:08 -05001792 memset(&qry, 0, sizeof(qry));
1793
Stefan Roese260421a2006-11-13 13:55:24 +01001794 info->ext_addr = 0;
1795 info->cfi_version = 0;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001796#ifdef CONFIG_SYS_FLASH_PROTECTION
Stefan Roese2662b402006-04-01 13:41:03 +02001797 info->legacy_unlock = 0;
1798#endif
wdenk5653fc32004-02-08 22:55:38 +00001799
1800 info->start[0] = base;
1801
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001802 if (flash_detect_cfi (info, &qry)) {
1803 info->vendor = le16_to_cpu(qry.p_id);
1804 info->ext_addr = le16_to_cpu(qry.p_adr);
1805 num_erase_regions = qry.num_erase_regions;
1806
Stefan Roese260421a2006-11-13 13:55:24 +01001807 if (info->ext_addr) {
1808 info->cfi_version = (ushort) flash_read_uchar (info,
1809 info->ext_addr + 3) << 8;
1810 info->cfi_version |= (ushort) flash_read_uchar (info,
1811 info->ext_addr + 4);
1812 }
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001813
wdenkbf9e3b32004-02-12 00:47:09 +00001814#ifdef DEBUG
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001815 flash_printqry (&qry);
wdenkbf9e3b32004-02-12 00:47:09 +00001816#endif
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001817
wdenkbf9e3b32004-02-12 00:47:09 +00001818 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001819 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenk5653fc32004-02-08 22:55:38 +00001820 case CFI_CMDSET_INTEL_STANDARD:
1821 case CFI_CMDSET_INTEL_EXTENDED:
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001822 cmdset_intel_init(info, &qry);
wdenk5653fc32004-02-08 22:55:38 +00001823 break;
1824 case CFI_CMDSET_AMD_STANDARD:
1825 case CFI_CMDSET_AMD_EXTENDED:
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001826 cmdset_amd_init(info, &qry);
wdenk5653fc32004-02-08 22:55:38 +00001827 break;
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001828 default:
1829 printf("CFI: Unknown command set 0x%x\n",
1830 info->vendor);
1831 /*
1832 * Unfortunately, this means we don't know how
1833 * to get the chip back to Read mode. Might
1834 * as well try an Intel-style reset...
1835 */
1836 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1837 return 0;
wdenk5653fc32004-02-08 22:55:38 +00001838 }
wdenkcd37d9e2004-02-10 00:03:41 +00001839
Haavard Skinnemoen467bcee2007-12-14 15:36:18 +01001840 /* Do manufacturer-specific fixups */
1841 switch (info->manufacturer_id) {
1842 case 0x0001:
1843 flash_fixup_amd(info, &qry);
1844 break;
1845 case 0x001f:
1846 flash_fixup_atmel(info, &qry);
1847 break;
1848 }
1849
wdenkbf9e3b32004-02-12 00:47:09 +00001850 debug ("manufacturer is %d\n", info->vendor);
Stefan Roese260421a2006-11-13 13:55:24 +01001851 debug ("manufacturer id is 0x%x\n", info->manufacturer_id);
1852 debug ("device id is 0x%x\n", info->device_id);
1853 debug ("device id2 is 0x%x\n", info->device_id2);
1854 debug ("cfi version is 0x%04x\n", info->cfi_version);
1855
wdenk5653fc32004-02-08 22:55:38 +00001856 size_ratio = info->portwidth / info->chipwidth;
wdenkbf9e3b32004-02-12 00:47:09 +00001857 /* if the chip is x8/x16 reduce the ratio by half */
1858 if ((info->interface == FLASH_CFI_X8X16)
1859 && (info->chipwidth == FLASH_CFI_BY8)) {
1860 size_ratio >>= 1;
1861 }
wdenkbf9e3b32004-02-12 00:47:09 +00001862 debug ("size_ratio %d port %d bits chip %d bits\n",
1863 size_ratio, info->portwidth << CFI_FLASH_SHIFT_WIDTH,
1864 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
1865 debug ("found %d erase regions\n", num_erase_regions);
wdenk5653fc32004-02-08 22:55:38 +00001866 sect_cnt = 0;
1867 sector = base;
wdenkbf9e3b32004-02-12 00:47:09 +00001868 for (i = 0; i < num_erase_regions; i++) {
1869 if (i > NUM_ERASE_REGIONS) {
wdenk028ab6b2004-02-23 23:54:43 +00001870 printf ("%d erase regions found, only %d used\n",
1871 num_erase_regions, NUM_ERASE_REGIONS);
wdenk5653fc32004-02-08 22:55:38 +00001872 break;
1873 }
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001874
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001875 tmp = le32_to_cpu(qry.erase_region_info[i]);
1876 debug("erase region %u: 0x%08lx\n", i, tmp);
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001877
1878 erase_region_count = (tmp & 0xffff) + 1;
1879 tmp >>= 16;
wdenkbf9e3b32004-02-12 00:47:09 +00001880 erase_region_size =
1881 (tmp & 0xffff) ? ((tmp & 0xffff) * 256) : 128;
wdenk4c0d4c32004-06-09 17:34:58 +00001882 debug ("erase_region_count = %d erase_region_size = %d\n",
wdenk028ab6b2004-02-23 23:54:43 +00001883 erase_region_count, erase_region_size);
wdenkbf9e3b32004-02-12 00:47:09 +00001884 for (j = 0; j < erase_region_count; j++) {
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001885 if (sect_cnt >= CONFIG_SYS_MAX_FLASH_SECT) {
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001886 printf("ERROR: too many flash sectors\n");
1887 break;
1888 }
wdenk5653fc32004-02-08 22:55:38 +00001889 info->start[sect_cnt] = sector;
1890 sector += (erase_region_size * size_ratio);
wdenka1191902005-01-09 17:12:27 +00001891
1892 /*
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001893 * Only read protection status from
1894 * supported devices (intel...)
wdenka1191902005-01-09 17:12:27 +00001895 */
1896 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001897 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenka1191902005-01-09 17:12:27 +00001898 case CFI_CMDSET_INTEL_EXTENDED:
1899 case CFI_CMDSET_INTEL_STANDARD:
1900 info->protect[sect_cnt] =
1901 flash_isset (info, sect_cnt,
1902 FLASH_OFFSET_PROTECT,
1903 FLASH_STATUS_PROTECT);
1904 break;
1905 default:
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001906 /* default: not protected */
1907 info->protect[sect_cnt] = 0;
wdenka1191902005-01-09 17:12:27 +00001908 }
1909
wdenk5653fc32004-02-08 22:55:38 +00001910 sect_cnt++;
1911 }
1912 }
1913
1914 info->sector_count = sect_cnt;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001915 info->size = 1 << qry.dev_size;
wdenk5653fc32004-02-08 22:55:38 +00001916 /* multiply the size by the number of chips */
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001917 info->size *= size_ratio;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001918 info->buffer_size = 1 << le16_to_cpu(qry.max_buf_write_size);
1919 tmp = 1 << qry.block_erase_timeout_typ;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001920 info->erase_blk_tout = tmp *
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001921 (1 << qry.block_erase_timeout_max);
1922 tmp = (1 << qry.buf_write_timeout_typ) *
1923 (1 << qry.buf_write_timeout_max);
1924
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001925 /* round up when converting to ms */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001926 info->buffer_write_tout = (tmp + 999) / 1000;
1927 tmp = (1 << qry.word_write_timeout_typ) *
1928 (1 << qry.word_write_timeout_max);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001929 /* round up when converting to ms */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001930 info->write_tout = (tmp + 999) / 1000;
wdenk5653fc32004-02-08 22:55:38 +00001931 info->flash_id = FLASH_MAN_CFI;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001932 if ((info->interface == FLASH_CFI_X8X16) &&
1933 (info->chipwidth == FLASH_CFI_BY8)) {
1934 /* XXX - Need to test on x8/x16 in parallel. */
1935 info->portwidth >>= 1;
wdenk855a4962004-03-14 18:23:55 +00001936 }
Mike Frysinger22159872008-10-02 01:55:38 -04001937
1938 flash_write_cmd (info, 0, 0, info->cmd_reset);
wdenk5653fc32004-02-08 22:55:38 +00001939 }
1940
wdenkbf9e3b32004-02-12 00:47:09 +00001941 return (info->size);
wdenk5653fc32004-02-08 22:55:38 +00001942}
1943
wdenk5653fc32004-02-08 22:55:38 +00001944/*-----------------------------------------------------------------------
1945 */
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001946unsigned long flash_init (void)
wdenk5653fc32004-02-08 22:55:38 +00001947{
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001948 unsigned long size = 0;
1949 int i;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001950#if defined(CONFIG_SYS_FLASH_AUTOPROTECT_LIST)
Matthias Fuchsc63ad632008-04-18 16:29:40 +02001951 struct apl_s {
1952 ulong start;
1953 ulong size;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001954 } apl[] = CONFIG_SYS_FLASH_AUTOPROTECT_LIST;
Matthias Fuchsc63ad632008-04-18 16:29:40 +02001955#endif
wdenk5653fc32004-02-08 22:55:38 +00001956
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001957#ifdef CONFIG_SYS_FLASH_PROTECTION
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001958 char *s = getenv("unlock");
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001959#endif
wdenk5653fc32004-02-08 22:55:38 +00001960
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001961#define BANK_BASE(i) (((unsigned long [CFI_MAX_FLASH_BANKS])CONFIG_SYS_FLASH_BANKS_LIST)[i])
Wolfgang Denk2a112b22008-08-08 16:39:54 +02001962
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001963 /* Init: no FLASHes known */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001964 for (i = 0; i < CONFIG_SYS_MAX_FLASH_BANKS; ++i) {
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001965 flash_info[i].flash_id = FLASH_UNKNOWN;
wdenk5653fc32004-02-08 22:55:38 +00001966
Wolfgang Denk2a112b22008-08-08 16:39:54 +02001967 if (!flash_detect_legacy (BANK_BASE(i), i))
1968 flash_get_size (BANK_BASE(i), i);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001969 size += flash_info[i].size;
1970 if (flash_info[i].flash_id == FLASH_UNKNOWN) {
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001971#ifndef CONFIG_SYS_FLASH_QUIET_TEST
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001972 printf ("## Unknown FLASH on Bank %d "
1973 "- Size = 0x%08lx = %ld MB\n",
1974 i+1, flash_info[i].size,
1975 flash_info[i].size << 20);
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001976#endif /* CONFIG_SYS_FLASH_QUIET_TEST */
wdenk5653fc32004-02-08 22:55:38 +00001977 }
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001978#ifdef CONFIG_SYS_FLASH_PROTECTION
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001979 else if ((s != NULL) && (strcmp(s, "yes") == 0)) {
1980 /*
1981 * Only the U-Boot image and it's environment
1982 * is protected, all other sectors are
1983 * unprotected (unlocked) if flash hardware
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001984 * protection is used (CONFIG_SYS_FLASH_PROTECTION)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001985 * and the environment variable "unlock" is
1986 * set to "yes".
1987 */
1988 if (flash_info[i].legacy_unlock) {
1989 int k;
Stefan Roese79b4cda2006-02-28 15:29:58 +01001990
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001991 /*
1992 * Disable legacy_unlock temporarily,
1993 * since flash_real_protect would
1994 * relock all other sectors again
1995 * otherwise.
1996 */
1997 flash_info[i].legacy_unlock = 0;
Stefan Roese79b4cda2006-02-28 15:29:58 +01001998
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001999 /*
2000 * Legacy unlocking (e.g. Intel J3) ->
2001 * unlock only one sector. This will
2002 * unlock all sectors.
2003 */
2004 flash_real_protect (&flash_info[i], 0, 0);
Stefan Roese79b4cda2006-02-28 15:29:58 +01002005
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002006 flash_info[i].legacy_unlock = 1;
2007
2008 /*
2009 * Manually mark other sectors as
2010 * unlocked (unprotected)
2011 */
2012 for (k = 1; k < flash_info[i].sector_count; k++)
2013 flash_info[i].protect[k] = 0;
2014 } else {
2015 /*
2016 * No legancy unlocking -> unlock all sectors
2017 */
2018 flash_protect (FLAG_PROTECT_CLEAR,
2019 flash_info[i].start[0],
2020 flash_info[i].start[0]
2021 + flash_info[i].size - 1,
2022 &flash_info[i]);
2023 }
Stefan Roese79b4cda2006-02-28 15:29:58 +01002024 }
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02002025#endif /* CONFIG_SYS_FLASH_PROTECTION */
wdenk5653fc32004-02-08 22:55:38 +00002026 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002027
2028 /* Monitor protection ON by default */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02002029#if (CONFIG_SYS_MONITOR_BASE >= CONFIG_SYS_FLASH_BASE)
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002030 flash_protect (FLAG_PROTECT_SET,
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02002031 CONFIG_SYS_MONITOR_BASE,
2032 CONFIG_SYS_MONITOR_BASE + monitor_flash_len - 1,
2033 flash_get_info(CONFIG_SYS_MONITOR_BASE));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002034#endif
2035
2036 /* Environment protection ON by default */
Jean-Christophe PLAGNIOL-VILLARD5a1aceb2008-09-10 22:48:04 +02002037#ifdef CONFIG_ENV_IS_IN_FLASH
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002038 flash_protect (FLAG_PROTECT_SET,
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002039 CONFIG_ENV_ADDR,
2040 CONFIG_ENV_ADDR + CONFIG_ENV_SECT_SIZE - 1,
2041 flash_get_info(CONFIG_ENV_ADDR));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002042#endif
2043
2044 /* Redundant environment protection ON by default */
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002045#ifdef CONFIG_ENV_ADDR_REDUND
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002046 flash_protect (FLAG_PROTECT_SET,
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002047 CONFIG_ENV_ADDR_REDUND,
2048 CONFIG_ENV_ADDR_REDUND + CONFIG_ENV_SIZE_REDUND - 1,
2049 flash_get_info(CONFIG_ENV_ADDR_REDUND));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002050#endif
Matthias Fuchsc63ad632008-04-18 16:29:40 +02002051
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02002052#if defined(CONFIG_SYS_FLASH_AUTOPROTECT_LIST)
Matthias Fuchsc63ad632008-04-18 16:29:40 +02002053 for (i = 0; i < (sizeof(apl) / sizeof(struct apl_s)); i++) {
2054 debug("autoprotecting from %08x to %08x\n",
2055 apl[i].start, apl[i].start + apl[i].size - 1);
2056 flash_protect (FLAG_PROTECT_SET,
2057 apl[i].start,
2058 apl[i].start + apl[i].size - 1,
2059 flash_get_info(apl[i].start));
2060 }
2061#endif
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002062 return (size);
wdenk5653fc32004-02-08 22:55:38 +00002063}