blob: 0d1ee8a459d0cd01d5880ae4a92ce1c8e64fded7 [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 *
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +010060 * Define CFG_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
wdenkbf9e3b32004-02-12 00:47:09 +000064#ifndef CFG_FLASH_BANKS_LIST
65#define CFG_FLASH_BANKS_LIST { CFG_FLASH_BASE }
66#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
wdenkf7d15722004-12-18 22:35:43 +0000146#ifdef CFG_FLASH_CFI_AMD_RESET /* needed for STM_ID_29W320DB on UC100 */
147# 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
Marian Balakowicze6f2e902005-10-11 19:09:42 +0200162/* use CFG_MAX_FLASH_BANKS_DETECT if defined */
163#ifdef CFG_MAX_FLASH_BANKS_DETECT
Wolfgang Denk2a112b22008-08-08 16:39:54 +0200164# define CFI_MAX_FLASH_BANKS CFG_MAX_FLASH_BANKS_DETECT
Marian Balakowicze6f2e902005-10-11 19:09:42 +0200165#else
Wolfgang Denk2a112b22008-08-08 16:39:54 +0200166# define CFI_MAX_FLASH_BANKS CFG_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 */
174#ifndef CFG_FLASH_CFI_WIDTH
175#define CFG_FLASH_CFI_WIDTH FLASH_CFI_8BIT
176#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-VILLARD0e8d1582008-09-10 22:48:06 +0200258#if defined(CONFIG_ENV_IS_IN_FLASH) || defined(CONFIG_ENV_ADDR_REDUND) || (CFG_MONITOR_BASE >= CFG_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
264 for (i = 0; i < CFG_MAX_FLASH_BANKS; i++) {
265 info = & flash_info[i];
266 if (info->size && info->start[0] <= base &&
267 base <= info->start[0] + info->size - 1)
268 break;
269 }
270
271 return i == CFG_MAX_FLASH_BANKS ? 0 : info;
272}
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;
Sebastian Siewior340ccb22008-07-16 20:04:49 +0200312#if defined(__LITTLE_ENDIAN) || defined(CFG_WRITE_SWAPPED_DATA)
313 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;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200320#if defined(__LITTLE_ENDIAN) || defined(CFG_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);
Heiko Schocherd0b6e142007-01-19 18:05:26 +0100377#if defined(__LITTLE_ENDIAN) || defined(CFG_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
Heiko Schocherd0b6e142007-01-19 18:05:26 +0100422#if defined(__LITTLE_ENDIAN) || defined(CFG_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
649#if CFG_HZ != 1000
650 tout *= CFG_HZ/1000;
651#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:
684 if ((retcode == ERR_OK)
685 && !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{
719#if defined(__LITTLE_ENDIAN) && !defined(CFG_WRITE_SWAPPED_DATA)
720 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:
730#if defined(__LITTLE_ENDIAN) && !defined(CFG_WRITE_SWAPPED_DATA)
731 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:
739#if defined(__LITTLE_ENDIAN) && !defined(CFG_WRITE_SWAPPED_DATA)
740 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:
748#if defined(__LITTLE_ENDIAN) && !defined(CFG_WRITE_SWAPPED_DATA)
749 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;
780
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100781 dstaddr = map_physmem(dest, info->portwidth, MAP_NOCACHE);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200782
783 /* Check if Flash is (sufficiently) erased */
784 switch (info->portwidth) {
785 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100786 flag = ((flash_read8(dstaddr) & cword.c) == cword.c);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200787 break;
788 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100789 flag = ((flash_read16(dstaddr) & cword.w) == cword.w);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200790 break;
791 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100792 flag = ((flash_read32(dstaddr) & cword.l) == cword.l);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200793 break;
794 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100795 flag = ((flash_read64(dstaddr) & cword.ll) == cword.ll);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200796 break;
797 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100798 flag = 0;
799 break;
800 }
801 if (!flag) {
802 unmap_physmem(dstaddr, info->portwidth);
Stefan Roese0dc80e22007-12-27 07:50:54 +0100803 return ERR_NOT_ERASED;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200804 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200805
806 /* Disable interrupts which might cause a timeout here */
807 flag = disable_interrupts ();
808
809 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400810 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200811 case CFI_CMDSET_INTEL_EXTENDED:
812 case CFI_CMDSET_INTEL_STANDARD:
813 flash_write_cmd (info, 0, 0, FLASH_CMD_CLEAR_STATUS);
814 flash_write_cmd (info, 0, 0, FLASH_CMD_WRITE);
815 break;
816 case CFI_CMDSET_AMD_EXTENDED:
817 case CFI_CMDSET_AMD_STANDARD:
818#ifdef CONFIG_FLASH_CFI_LEGACY
819 case CFI_CMDSET_AMD_LEGACY:
Wolfgang Denk080bdb72005-10-05 01:51:29 +0200820#endif
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200821 flash_unlock_seq (info, 0);
822 flash_write_cmd (info, 0, info->addr_unlock1, AMD_CMD_WRITE);
823 break;
824 }
825
826 switch (info->portwidth) {
827 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100828 flash_write8(cword.c, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200829 break;
830 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100831 flash_write16(cword.w, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200832 break;
833 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100834 flash_write32(cword.l, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200835 break;
836 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100837 flash_write64(cword.ll, dstaddr);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200838 break;
839 }
840
841 /* re-enable interrupts if necessary */
842 if (flag)
843 enable_interrupts ();
844
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100845 unmap_physmem(dstaddr, info->portwidth);
846
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200847 return flash_full_status_check (info, find_sector (info, dest),
848 info->write_tout, "write");
849}
850
851#ifdef CFG_FLASH_USE_BUFFER_WRITE
852
853static int flash_write_cfibuffer (flash_info_t * info, ulong dest, uchar * cp,
854 int len)
855{
856 flash_sect_t sector;
857 int cnt;
858 int retcode;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100859 void *src = cp;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100860 void *dst = map_physmem(dest, len, MAP_NOCACHE);
Stefan Roese0dc80e22007-12-27 07:50:54 +0100861 void *dst2 = dst;
862 int flag = 0;
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200863 uint offset = 0;
864 unsigned int shift;
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400865 uchar write_cmd;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100866
Stefan Roese0dc80e22007-12-27 07:50:54 +0100867 switch (info->portwidth) {
868 case FLASH_CFI_8BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200869 shift = 0;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100870 break;
871 case FLASH_CFI_16BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200872 shift = 1;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100873 break;
874 case FLASH_CFI_32BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200875 shift = 2;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100876 break;
877 case FLASH_CFI_64BIT:
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200878 shift = 3;
Stefan Roese0dc80e22007-12-27 07:50:54 +0100879 break;
880 default:
881 retcode = ERR_INVAL;
882 goto out_unmap;
883 }
884
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200885 cnt = len >> shift;
886
Stefan Roese0dc80e22007-12-27 07:50:54 +0100887 while ((cnt-- > 0) && (flag == 0)) {
888 switch (info->portwidth) {
889 case FLASH_CFI_8BIT:
890 flag = ((flash_read8(dst2) & flash_read8(src)) ==
891 flash_read8(src));
892 src += 1, dst2 += 1;
893 break;
894 case FLASH_CFI_16BIT:
895 flag = ((flash_read16(dst2) & flash_read16(src)) ==
896 flash_read16(src));
897 src += 2, dst2 += 2;
898 break;
899 case FLASH_CFI_32BIT:
900 flag = ((flash_read32(dst2) & flash_read32(src)) ==
901 flash_read32(src));
902 src += 4, dst2 += 4;
903 break;
904 case FLASH_CFI_64BIT:
905 flag = ((flash_read64(dst2) & flash_read64(src)) ==
906 flash_read64(src));
907 src += 8, dst2 += 8;
908 break;
909 }
910 }
911 if (!flag) {
912 retcode = ERR_NOT_ERASED;
913 goto out_unmap;
914 }
915
916 src = cp;
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100917 sector = find_sector (info, dest);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200918
919 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400920 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200921 case CFI_CMDSET_INTEL_STANDARD:
922 case CFI_CMDSET_INTEL_EXTENDED:
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400923 write_cmd = (info->vendor == CFI_CMDSET_INTEL_PROG_REGIONS) ?
924 FLASH_CMD_WRITE_BUFFER_PROG : FLASH_CMD_WRITE_TO_BUFFER;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200925 flash_write_cmd (info, sector, 0, FLASH_CMD_CLEAR_STATUS);
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +0400926 flash_write_cmd (info, sector, 0, FLASH_CMD_READ_STATUS);
927 flash_write_cmd (info, sector, 0, write_cmd);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200928 retcode = flash_status_check (info, sector,
929 info->buffer_write_tout,
930 "write to buffer");
931 if (retcode == ERR_OK) {
932 /* reduce the number of loops by the width of
933 * the port */
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200934 cnt = len >> shift;
Vasiliy Leoenenko93c56f22008-05-07 21:24:44 +0400935 flash_write_cmd (info, sector, 0, cnt - 1);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200936 while (cnt-- > 0) {
937 switch (info->portwidth) {
938 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100939 flash_write8(flash_read8(src), dst);
940 src += 1, dst += 1;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200941 break;
942 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100943 flash_write16(flash_read16(src), dst);
944 src += 2, dst += 2;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200945 break;
946 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100947 flash_write32(flash_read32(src), dst);
948 src += 4, dst += 4;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200949 break;
950 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100951 flash_write64(flash_read64(src), dst);
952 src += 8, dst += 8;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200953 break;
954 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100955 retcode = ERR_INVAL;
956 goto out_unmap;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200957 }
958 }
959 flash_write_cmd (info, sector, 0,
960 FLASH_CMD_WRITE_BUFFER_CONFIRM);
961 retcode = flash_full_status_check (
962 info, sector, info->buffer_write_tout,
963 "buffer write");
964 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +0100965
966 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200967
968 case CFI_CMDSET_AMD_STANDARD:
969 case CFI_CMDSET_AMD_EXTENDED:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200970 flash_unlock_seq(info,0);
Guennadi Liakhovetski96ef8312008-04-03 13:36:02 +0200971
972#ifdef CONFIG_FLASH_SPANSION_S29WS_N
973 offset = ((unsigned long)dst - info->start[sector]) >> shift;
974#endif
975 flash_write_cmd(info, sector, offset, AMD_CMD_WRITE_TO_BUFFER);
976 cnt = len >> shift;
977 flash_write_cmd(info, sector, offset, (uchar)cnt - 1);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200978
979 switch (info->portwidth) {
980 case FLASH_CFI_8BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100981 while (cnt-- > 0) {
982 flash_write8(flash_read8(src), dst);
983 src += 1, dst += 1;
984 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200985 break;
986 case FLASH_CFI_16BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100987 while (cnt-- > 0) {
988 flash_write16(flash_read16(src), dst);
989 src += 2, dst += 2;
990 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200991 break;
992 case FLASH_CFI_32BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100993 while (cnt-- > 0) {
994 flash_write32(flash_read32(src), dst);
995 src += 4, dst += 4;
996 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +0200997 break;
998 case FLASH_CFI_64BIT:
Haavard Skinnemoencdbaefb2007-12-13 12:56:32 +0100999 while (cnt-- > 0) {
1000 flash_write64(flash_read64(src), dst);
1001 src += 8, dst += 8;
1002 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001003 break;
1004 default:
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001005 retcode = ERR_INVAL;
1006 goto out_unmap;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001007 }
1008
1009 flash_write_cmd (info, sector, 0, AMD_CMD_WRITE_BUFFER_CONFIRM);
1010 retcode = flash_full_status_check (info, sector,
1011 info->buffer_write_tout,
1012 "buffer write");
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001013 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001014
1015 default:
1016 debug ("Unknown Command Set\n");
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001017 retcode = ERR_INVAL;
1018 break;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001019 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001020
1021out_unmap:
1022 unmap_physmem(dst, len);
1023 return retcode;
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001024}
1025#endif /* CFG_FLASH_USE_BUFFER_WRITE */
1026
wdenk7680c142005-05-16 15:23:22 +00001027
1028/*-----------------------------------------------------------------------
1029 */
wdenkbf9e3b32004-02-12 00:47:09 +00001030int flash_erase (flash_info_t * info, int s_first, int s_last)
wdenk5653fc32004-02-08 22:55:38 +00001031{
1032 int rcode = 0;
1033 int prot;
1034 flash_sect_t sect;
1035
wdenkbf9e3b32004-02-12 00:47:09 +00001036 if (info->flash_id != FLASH_MAN_CFI) {
wdenk4b9206e2004-03-23 22:14:11 +00001037 puts ("Can't erase unknown flash type - aborted\n");
wdenk5653fc32004-02-08 22:55:38 +00001038 return 1;
1039 }
1040 if ((s_first < 0) || (s_first > s_last)) {
wdenk4b9206e2004-03-23 22:14:11 +00001041 puts ("- no sectors to erase\n");
wdenk5653fc32004-02-08 22:55:38 +00001042 return 1;
1043 }
1044
1045 prot = 0;
wdenkbf9e3b32004-02-12 00:47:09 +00001046 for (sect = s_first; sect <= s_last; ++sect) {
wdenk5653fc32004-02-08 22:55:38 +00001047 if (info->protect[sect]) {
1048 prot++;
1049 }
1050 }
1051 if (prot) {
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001052 printf ("- Warning: %d protected sectors will not be erased!\n",
1053 prot);
wdenk5653fc32004-02-08 22:55:38 +00001054 } else {
wdenk4b9206e2004-03-23 22:14:11 +00001055 putc ('\n');
wdenk5653fc32004-02-08 22:55:38 +00001056 }
1057
1058
wdenkbf9e3b32004-02-12 00:47:09 +00001059 for (sect = s_first; sect <= s_last; sect++) {
wdenk5653fc32004-02-08 22:55:38 +00001060 if (info->protect[sect] == 0) { /* not protected */
wdenkbf9e3b32004-02-12 00:47:09 +00001061 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001062 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenk5653fc32004-02-08 22:55:38 +00001063 case CFI_CMDSET_INTEL_STANDARD:
1064 case CFI_CMDSET_INTEL_EXTENDED:
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001065 flash_write_cmd (info, sect, 0,
1066 FLASH_CMD_CLEAR_STATUS);
1067 flash_write_cmd (info, sect, 0,
1068 FLASH_CMD_BLOCK_ERASE);
1069 flash_write_cmd (info, sect, 0,
1070 FLASH_CMD_ERASE_CONFIRM);
wdenk5653fc32004-02-08 22:55:38 +00001071 break;
1072 case CFI_CMDSET_AMD_STANDARD:
1073 case CFI_CMDSET_AMD_EXTENDED:
wdenkbf9e3b32004-02-12 00:47:09 +00001074 flash_unlock_seq (info, sect);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001075 flash_write_cmd (info, sect,
1076 info->addr_unlock1,
1077 AMD_CMD_ERASE_START);
wdenkbf9e3b32004-02-12 00:47:09 +00001078 flash_unlock_seq (info, sect);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001079 flash_write_cmd (info, sect, 0,
1080 AMD_CMD_ERASE_SECTOR);
wdenk5653fc32004-02-08 22:55:38 +00001081 break;
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001082#ifdef CONFIG_FLASH_CFI_LEGACY
1083 case CFI_CMDSET_AMD_LEGACY:
1084 flash_unlock_seq (info, 0);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001085 flash_write_cmd (info, 0, info->addr_unlock1,
1086 AMD_CMD_ERASE_START);
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001087 flash_unlock_seq (info, 0);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001088 flash_write_cmd (info, sect, 0,
1089 AMD_CMD_ERASE_SECTOR);
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001090 break;
1091#endif
wdenk5653fc32004-02-08 22:55:38 +00001092 default:
wdenkbf9e3b32004-02-12 00:47:09 +00001093 debug ("Unkown flash vendor %d\n",
1094 info->vendor);
wdenk5653fc32004-02-08 22:55:38 +00001095 break;
1096 }
1097
wdenkbf9e3b32004-02-12 00:47:09 +00001098 if (flash_full_status_check
1099 (info, sect, info->erase_blk_tout, "erase")) {
wdenk5653fc32004-02-08 22:55:38 +00001100 rcode = 1;
1101 } else
wdenk4b9206e2004-03-23 22:14:11 +00001102 putc ('.');
wdenk5653fc32004-02-08 22:55:38 +00001103 }
1104 }
wdenk4b9206e2004-03-23 22:14:11 +00001105 puts (" done\n");
wdenk5653fc32004-02-08 22:55:38 +00001106 return rcode;
1107}
1108
1109/*-----------------------------------------------------------------------
1110 */
wdenkbf9e3b32004-02-12 00:47:09 +00001111void flash_print_info (flash_info_t * info)
wdenk5653fc32004-02-08 22:55:38 +00001112{
1113 int i;
1114
1115 if (info->flash_id != FLASH_MAN_CFI) {
wdenk4b9206e2004-03-23 22:14:11 +00001116 puts ("missing or unknown FLASH type\n");
wdenk5653fc32004-02-08 22:55:38 +00001117 return;
1118 }
1119
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001120 printf ("%s FLASH (%d x %d)",
1121 info->name,
wdenkbf9e3b32004-02-12 00:47:09 +00001122 (info->portwidth << 3), (info->chipwidth << 3));
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001123 if (info->size < 1024*1024)
1124 printf (" Size: %ld kB in %d Sectors\n",
1125 info->size >> 10, info->sector_count);
1126 else
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001127 printf (" Size: %ld MB in %d Sectors\n",
1128 info->size >> 20, info->sector_count);
Stefan Roese260421a2006-11-13 13:55:24 +01001129 printf (" ");
1130 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001131 case CFI_CMDSET_INTEL_PROG_REGIONS:
1132 printf ("Intel Prog Regions");
1133 break;
Stefan Roese260421a2006-11-13 13:55:24 +01001134 case CFI_CMDSET_INTEL_STANDARD:
1135 printf ("Intel Standard");
1136 break;
1137 case CFI_CMDSET_INTEL_EXTENDED:
1138 printf ("Intel Extended");
1139 break;
1140 case CFI_CMDSET_AMD_STANDARD:
1141 printf ("AMD Standard");
1142 break;
1143 case CFI_CMDSET_AMD_EXTENDED:
1144 printf ("AMD Extended");
1145 break;
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001146#ifdef CONFIG_FLASH_CFI_LEGACY
1147 case CFI_CMDSET_AMD_LEGACY:
1148 printf ("AMD Legacy");
1149 break;
1150#endif
Stefan Roese260421a2006-11-13 13:55:24 +01001151 default:
1152 printf ("Unknown (%d)", info->vendor);
1153 break;
1154 }
1155 printf (" command set, Manufacturer ID: 0x%02X, Device ID: 0x%02X",
1156 info->manufacturer_id, info->device_id);
1157 if (info->device_id == 0x7E) {
1158 printf("%04X", info->device_id2);
1159 }
1160 printf ("\n Erase timeout: %ld ms, write timeout: %ld ms\n",
wdenk028ab6b2004-02-23 23:54:43 +00001161 info->erase_blk_tout,
Stefan Roese260421a2006-11-13 13:55:24 +01001162 info->write_tout);
1163 if (info->buffer_size > 1) {
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001164 printf (" Buffer write timeout: %ld ms, "
1165 "buffer size: %d bytes\n",
wdenk028ab6b2004-02-23 23:54:43 +00001166 info->buffer_write_tout,
1167 info->buffer_size);
Stefan Roese260421a2006-11-13 13:55:24 +01001168 }
wdenk5653fc32004-02-08 22:55:38 +00001169
Stefan Roese260421a2006-11-13 13:55:24 +01001170 puts ("\n Sector Start Addresses:");
wdenkbf9e3b32004-02-12 00:47:09 +00001171 for (i = 0; i < info->sector_count; ++i) {
Stefan Roese260421a2006-11-13 13:55:24 +01001172 if ((i % 5) == 0)
1173 printf ("\n");
wdenk5653fc32004-02-08 22:55:38 +00001174#ifdef CFG_FLASH_EMPTY_INFO
1175 int k;
1176 int size;
1177 int erased;
1178 volatile unsigned long *flash;
1179
1180 /*
1181 * Check if whole sector is erased
1182 */
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001183 size = flash_sector_size(info, i);
wdenk5653fc32004-02-08 22:55:38 +00001184 erased = 1;
wdenkbf9e3b32004-02-12 00:47:09 +00001185 flash = (volatile unsigned long *) info->start[i];
1186 size = size >> 2; /* divide by 4 for longword access */
1187 for (k = 0; k < size; k++) {
1188 if (*flash++ != 0xffffffff) {
1189 erased = 0;
1190 break;
1191 }
1192 }
wdenk5653fc32004-02-08 22:55:38 +00001193
wdenk5653fc32004-02-08 22:55:38 +00001194 /* print empty and read-only info */
Stefan Roese260421a2006-11-13 13:55:24 +01001195 printf (" %08lX %c %s ",
wdenk5653fc32004-02-08 22:55:38 +00001196 info->start[i],
Stefan Roese260421a2006-11-13 13:55:24 +01001197 erased ? 'E' : ' ',
1198 info->protect[i] ? "RO" : " ");
Wolfgang Denkb63de2c2005-09-25 00:23:05 +02001199#else /* ! CFG_FLASH_EMPTY_INFO */
Stefan Roese260421a2006-11-13 13:55:24 +01001200 printf (" %08lX %s ",
1201 info->start[i],
1202 info->protect[i] ? "RO" : " ");
wdenk5653fc32004-02-08 22:55:38 +00001203#endif
1204 }
wdenk4b9206e2004-03-23 22:14:11 +00001205 putc ('\n');
wdenk5653fc32004-02-08 22:55:38 +00001206 return;
1207}
1208
1209/*-----------------------------------------------------------------------
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001210 * This is used in a few places in write_buf() to show programming
1211 * progress. Making it a function is nasty because it needs to do side
1212 * effect updates to digit and dots. Repeated code is nasty too, so
1213 * we define it once here.
1214 */
Stefan Roesef0105722008-03-19 07:09:26 +01001215#ifdef CONFIG_FLASH_SHOW_PROGRESS
1216#define FLASH_SHOW_PROGRESS(scale, dots, digit, dots_sub) \
1217 dots -= dots_sub; \
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001218 if ((scale > 0) && (dots <= 0)) { \
1219 if ((digit % 5) == 0) \
1220 printf ("%d", digit / 5); \
1221 else \
1222 putc ('.'); \
1223 digit--; \
1224 dots += scale; \
1225 }
Stefan Roesef0105722008-03-19 07:09:26 +01001226#else
1227#define FLASH_SHOW_PROGRESS(scale, dots, digit, dots_sub)
1228#endif
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001229
1230/*-----------------------------------------------------------------------
wdenk5653fc32004-02-08 22:55:38 +00001231 * Copy memory to flash, returns:
1232 * 0 - OK
1233 * 1 - write timeout
1234 * 2 - Flash not erased
1235 */
wdenkbf9e3b32004-02-12 00:47:09 +00001236int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt)
wdenk5653fc32004-02-08 22:55:38 +00001237{
1238 ulong wp;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001239 uchar *p;
wdenk5653fc32004-02-08 22:55:38 +00001240 int aln;
1241 cfiword_t cword;
1242 int i, rc;
wdenkbf9e3b32004-02-12 00:47:09 +00001243#ifdef CFG_FLASH_USE_BUFFER_WRITE
1244 int buffered_size;
1245#endif
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001246#ifdef CONFIG_FLASH_SHOW_PROGRESS
1247 int digit = CONFIG_FLASH_SHOW_PROGRESS;
1248 int scale = 0;
1249 int dots = 0;
1250
1251 /*
1252 * Suppress if there are fewer than CONFIG_FLASH_SHOW_PROGRESS writes.
1253 */
1254 if (cnt >= CONFIG_FLASH_SHOW_PROGRESS) {
1255 scale = (int)((cnt + CONFIG_FLASH_SHOW_PROGRESS - 1) /
1256 CONFIG_FLASH_SHOW_PROGRESS);
1257 }
1258#endif
1259
wdenkbf9e3b32004-02-12 00:47:09 +00001260 /* get lower aligned address */
wdenk5653fc32004-02-08 22:55:38 +00001261 wp = (addr & ~(info->portwidth - 1));
1262
1263 /* handle unaligned start */
wdenkbf9e3b32004-02-12 00:47:09 +00001264 if ((aln = addr - wp) != 0) {
wdenk5653fc32004-02-08 22:55:38 +00001265 cword.l = 0;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001266 p = map_physmem(wp, info->portwidth, MAP_NOCACHE);
1267 for (i = 0; i < aln; ++i)
1268 flash_add_byte (info, &cword, flash_read8(p + i));
wdenk5653fc32004-02-08 22:55:38 +00001269
wdenkbf9e3b32004-02-12 00:47:09 +00001270 for (; (i < info->portwidth) && (cnt > 0); i++) {
1271 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001272 cnt--;
wdenk5653fc32004-02-08 22:55:38 +00001273 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001274 for (; (cnt == 0) && (i < info->portwidth); ++i)
1275 flash_add_byte (info, &cword, flash_read8(p + i));
1276
1277 rc = flash_write_cfiword (info, wp, cword);
1278 unmap_physmem(p, info->portwidth);
1279 if (rc != 0)
wdenk5653fc32004-02-08 22:55:38 +00001280 return rc;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001281
1282 wp += i;
Stefan Roesef0105722008-03-19 07:09:26 +01001283 FLASH_SHOW_PROGRESS(scale, dots, digit, i);
wdenk5653fc32004-02-08 22:55:38 +00001284 }
1285
wdenkbf9e3b32004-02-12 00:47:09 +00001286 /* handle the aligned part */
wdenk5653fc32004-02-08 22:55:38 +00001287#ifdef CFG_FLASH_USE_BUFFER_WRITE
wdenkbf9e3b32004-02-12 00:47:09 +00001288 buffered_size = (info->portwidth / info->chipwidth);
1289 buffered_size *= info->buffer_size;
1290 while (cnt >= info->portwidth) {
Stefan Roese79b4cda2006-02-28 15:29:58 +01001291 /* prohibit buffer write when buffer_size is 1 */
1292 if (info->buffer_size == 1) {
1293 cword.l = 0;
1294 for (i = 0; i < info->portwidth; i++)
1295 flash_add_byte (info, &cword, *src++);
1296 if ((rc = flash_write_cfiword (info, wp, cword)) != 0)
1297 return rc;
1298 wp += info->portwidth;
1299 cnt -= info->portwidth;
1300 continue;
1301 }
1302
1303 /* write buffer until next buffered_size aligned boundary */
1304 i = buffered_size - (wp % buffered_size);
1305 if (i > cnt)
1306 i = cnt;
wdenkbf9e3b32004-02-12 00:47:09 +00001307 if ((rc = flash_write_cfibuffer (info, wp, src, i)) != ERR_OK)
wdenk5653fc32004-02-08 22:55:38 +00001308 return rc;
Wolfgang Denk8d4ba3d2005-08-12 22:35:59 +02001309 i -= i & (info->portwidth - 1);
wdenk5653fc32004-02-08 22:55:38 +00001310 wp += i;
1311 src += i;
wdenkbf9e3b32004-02-12 00:47:09 +00001312 cnt -= i;
Stefan Roesef0105722008-03-19 07:09:26 +01001313 FLASH_SHOW_PROGRESS(scale, dots, digit, i);
wdenk5653fc32004-02-08 22:55:38 +00001314 }
1315#else
wdenkbf9e3b32004-02-12 00:47:09 +00001316 while (cnt >= info->portwidth) {
wdenk5653fc32004-02-08 22:55:38 +00001317 cword.l = 0;
wdenkbf9e3b32004-02-12 00:47:09 +00001318 for (i = 0; i < info->portwidth; i++) {
1319 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001320 }
wdenkbf9e3b32004-02-12 00:47:09 +00001321 if ((rc = flash_write_cfiword (info, wp, cword)) != 0)
wdenk5653fc32004-02-08 22:55:38 +00001322 return rc;
1323 wp += info->portwidth;
1324 cnt -= info->portwidth;
Stefan Roesef0105722008-03-19 07:09:26 +01001325 FLASH_SHOW_PROGRESS(scale, dots, digit, info->portwidth);
wdenk5653fc32004-02-08 22:55:38 +00001326 }
1327#endif /* CFG_FLASH_USE_BUFFER_WRITE */
Jerry Van Baren9a042e92008-03-08 13:48:01 -05001328
wdenk5653fc32004-02-08 22:55:38 +00001329 if (cnt == 0) {
1330 return (0);
1331 }
1332
1333 /*
1334 * handle unaligned tail bytes
1335 */
1336 cword.l = 0;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001337 p = map_physmem(wp, info->portwidth, MAP_NOCACHE);
1338 for (i = 0; (i < info->portwidth) && (cnt > 0); ++i) {
wdenkbf9e3b32004-02-12 00:47:09 +00001339 flash_add_byte (info, &cword, *src++);
wdenk5653fc32004-02-08 22:55:38 +00001340 --cnt;
1341 }
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001342 for (; i < info->portwidth; ++i)
1343 flash_add_byte (info, &cword, flash_read8(p + i));
1344 unmap_physmem(p, info->portwidth);
wdenk5653fc32004-02-08 22:55:38 +00001345
wdenkbf9e3b32004-02-12 00:47:09 +00001346 return flash_write_cfiword (info, wp, cword);
wdenk5653fc32004-02-08 22:55:38 +00001347}
1348
1349/*-----------------------------------------------------------------------
1350 */
1351#ifdef CFG_FLASH_PROTECTION
1352
wdenkbf9e3b32004-02-12 00:47:09 +00001353int flash_real_protect (flash_info_t * info, long sector, int prot)
wdenk5653fc32004-02-08 22:55:38 +00001354{
1355 int retcode = 0;
1356
Rafael Camposbc9019e2008-07-31 10:22:20 +02001357 switch (info->vendor) {
1358 case CFI_CMDSET_INTEL_PROG_REGIONS:
1359 case CFI_CMDSET_INTEL_STANDARD:
Nick Spence9e8e63c2008-08-19 22:21:16 -07001360 case CFI_CMDSET_INTEL_EXTENDED:
Rafael Camposbc9019e2008-07-31 10:22:20 +02001361 flash_write_cmd (info, sector, 0,
1362 FLASH_CMD_CLEAR_STATUS);
1363 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT);
1364 if (prot)
1365 flash_write_cmd (info, sector, 0,
1366 FLASH_CMD_PROTECT_SET);
1367 else
1368 flash_write_cmd (info, sector, 0,
1369 FLASH_CMD_PROTECT_CLEAR);
1370 break;
1371 case CFI_CMDSET_AMD_EXTENDED:
1372 case CFI_CMDSET_AMD_STANDARD:
Rafael Camposbc9019e2008-07-31 10:22:20 +02001373 /* U-Boot only checks the first byte */
1374 if (info->manufacturer_id == (uchar)ATM_MANUFACT) {
1375 if (prot) {
1376 flash_unlock_seq (info, 0);
1377 flash_write_cmd (info, 0,
1378 info->addr_unlock1,
1379 ATM_CMD_SOFTLOCK_START);
1380 flash_unlock_seq (info, 0);
1381 flash_write_cmd (info, sector, 0,
1382 ATM_CMD_LOCK_SECT);
1383 } else {
1384 flash_write_cmd (info, 0,
1385 info->addr_unlock1,
1386 AMD_CMD_UNLOCK_START);
1387 if (info->device_id == ATM_ID_BV6416)
1388 flash_write_cmd (info, sector,
1389 0, ATM_CMD_UNLOCK_SECT);
1390 }
1391 }
1392 break;
TsiChung Liew4e00acd2008-08-19 16:53:39 +00001393#ifdef CONFIG_FLASH_CFI_LEGACY
1394 case CFI_CMDSET_AMD_LEGACY:
1395 flash_write_cmd (info, sector, 0, FLASH_CMD_CLEAR_STATUS);
1396 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT);
1397 if (prot)
1398 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT_SET);
1399 else
1400 flash_write_cmd (info, sector, 0, FLASH_CMD_PROTECT_CLEAR);
1401#endif
Rafael Camposbc9019e2008-07-31 10:22:20 +02001402 };
wdenk5653fc32004-02-08 22:55:38 +00001403
wdenkbf9e3b32004-02-12 00:47:09 +00001404 if ((retcode =
1405 flash_full_status_check (info, sector, info->erase_blk_tout,
1406 prot ? "protect" : "unprotect")) == 0) {
wdenk5653fc32004-02-08 22:55:38 +00001407
1408 info->protect[sector] = prot;
Stefan Roese2662b402006-04-01 13:41:03 +02001409
1410 /*
1411 * On some of Intel's flash chips (marked via legacy_unlock)
1412 * unprotect unprotects all locking.
1413 */
1414 if ((prot == 0) && (info->legacy_unlock)) {
wdenk5653fc32004-02-08 22:55:38 +00001415 flash_sect_t i;
wdenkbf9e3b32004-02-12 00:47:09 +00001416
1417 for (i = 0; i < info->sector_count; i++) {
1418 if (info->protect[i])
1419 flash_real_protect (info, i, 1);
wdenk5653fc32004-02-08 22:55:38 +00001420 }
1421 }
1422 }
wdenk5653fc32004-02-08 22:55:38 +00001423 return retcode;
wdenkbf9e3b32004-02-12 00:47:09 +00001424}
1425
wdenk5653fc32004-02-08 22:55:38 +00001426/*-----------------------------------------------------------------------
1427 * flash_read_user_serial - read the OneTimeProgramming cells
1428 */
wdenkbf9e3b32004-02-12 00:47:09 +00001429void flash_read_user_serial (flash_info_t * info, void *buffer, int offset,
1430 int len)
wdenk5653fc32004-02-08 22:55:38 +00001431{
wdenkbf9e3b32004-02-12 00:47:09 +00001432 uchar *src;
1433 uchar *dst;
wdenk5653fc32004-02-08 22:55:38 +00001434
1435 dst = buffer;
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001436 src = flash_map (info, 0, FLASH_OFFSET_USER_PROTECTION);
wdenkbf9e3b32004-02-12 00:47:09 +00001437 flash_write_cmd (info, 0, 0, FLASH_CMD_READ_ID);
1438 memcpy (dst, src + offset, len);
Wolfgang Denkdb421e62005-09-25 16:41:22 +02001439 flash_write_cmd (info, 0, 0, info->cmd_reset);
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001440 flash_unmap(info, 0, FLASH_OFFSET_USER_PROTECTION, src);
wdenk5653fc32004-02-08 22:55:38 +00001441}
wdenkbf9e3b32004-02-12 00:47:09 +00001442
wdenk5653fc32004-02-08 22:55:38 +00001443/*
1444 * flash_read_factory_serial - read the device Id from the protection area
1445 */
wdenkbf9e3b32004-02-12 00:47:09 +00001446void flash_read_factory_serial (flash_info_t * info, void *buffer, int offset,
1447 int len)
wdenk5653fc32004-02-08 22:55:38 +00001448{
wdenkbf9e3b32004-02-12 00:47:09 +00001449 uchar *src;
wdenkcd37d9e2004-02-10 00:03:41 +00001450
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001451 src = flash_map (info, 0, FLASH_OFFSET_INTEL_PROTECTION);
wdenkbf9e3b32004-02-12 00:47:09 +00001452 flash_write_cmd (info, 0, 0, FLASH_CMD_READ_ID);
1453 memcpy (buffer, src + offset, len);
Wolfgang Denkdb421e62005-09-25 16:41:22 +02001454 flash_write_cmd (info, 0, 0, info->cmd_reset);
Haavard Skinnemoen12d30aa2007-12-13 12:56:34 +01001455 flash_unmap(info, 0, FLASH_OFFSET_INTEL_PROTECTION, src);
wdenk5653fc32004-02-08 22:55:38 +00001456}
1457
1458#endif /* CFG_FLASH_PROTECTION */
1459
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001460/*-----------------------------------------------------------------------
1461 * Reverse the order of the erase regions in the CFI QRY structure.
1462 * This is needed for chips that are either a) correctly detected as
1463 * top-boot, or b) buggy.
1464 */
1465static void cfi_reverse_geometry(struct cfi_qry *qry)
1466{
1467 unsigned int i, j;
1468 u32 tmp;
1469
1470 for (i = 0, j = qry->num_erase_regions - 1; i < j; i++, j--) {
1471 tmp = qry->erase_region_info[i];
1472 qry->erase_region_info[i] = qry->erase_region_info[j];
1473 qry->erase_region_info[j] = tmp;
1474 }
1475}
wdenk5653fc32004-02-08 22:55:38 +00001476
1477/*-----------------------------------------------------------------------
Stefan Roese260421a2006-11-13 13:55:24 +01001478 * read jedec ids from device and set corresponding fields in info struct
1479 *
1480 * Note: assume cfi->vendor, cfi->portwidth and cfi->chipwidth are correct
1481 *
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001482 */
1483static void cmdset_intel_read_jedec_ids(flash_info_t *info)
1484{
1485 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1486 flash_write_cmd(info, 0, 0, FLASH_CMD_READ_ID);
1487 udelay(1000); /* some flash are slow to respond */
1488 info->manufacturer_id = flash_read_uchar (info,
1489 FLASH_OFFSET_MANUFACTURER_ID);
1490 info->device_id = flash_read_uchar (info,
1491 FLASH_OFFSET_DEVICE_ID);
1492 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1493}
1494
1495static int cmdset_intel_init(flash_info_t *info, struct cfi_qry *qry)
1496{
1497 info->cmd_reset = FLASH_CMD_RESET;
1498
1499 cmdset_intel_read_jedec_ids(info);
1500 flash_write_cmd(info, 0, info->cfi_offset, FLASH_CMD_CFI);
1501
1502#ifdef CFG_FLASH_PROTECTION
1503 /* read legacy lock/unlock bit from intel flash */
1504 if (info->ext_addr) {
1505 info->legacy_unlock = flash_read_uchar (info,
1506 info->ext_addr + 5) & 0x08;
1507 }
1508#endif
1509
1510 return 0;
1511}
1512
1513static void cmdset_amd_read_jedec_ids(flash_info_t *info)
1514{
1515 flash_write_cmd(info, 0, 0, AMD_CMD_RESET);
1516 flash_unlock_seq(info, 0);
1517 flash_write_cmd(info, 0, info->addr_unlock1, FLASH_CMD_READ_ID);
1518 udelay(1000); /* some flash are slow to respond */
Tor Krill90447ec2008-03-28 11:29:10 +01001519
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001520 info->manufacturer_id = flash_read_uchar (info,
1521 FLASH_OFFSET_MANUFACTURER_ID);
Tor Krill90447ec2008-03-28 11:29:10 +01001522
1523 switch (info->chipwidth){
1524 case FLASH_CFI_8BIT:
1525 info->device_id = flash_read_uchar (info,
1526 FLASH_OFFSET_DEVICE_ID);
1527 if (info->device_id == 0x7E) {
1528 /* AMD 3-byte (expanded) device ids */
1529 info->device_id2 = flash_read_uchar (info,
1530 FLASH_OFFSET_DEVICE_ID2);
1531 info->device_id2 <<= 8;
1532 info->device_id2 |= flash_read_uchar (info,
1533 FLASH_OFFSET_DEVICE_ID3);
1534 }
1535 break;
1536 case FLASH_CFI_16BIT:
1537 info->device_id = flash_read_word (info,
1538 FLASH_OFFSET_DEVICE_ID);
1539 break;
1540 default:
1541 break;
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001542 }
1543 flash_write_cmd(info, 0, 0, AMD_CMD_RESET);
1544}
1545
1546static int cmdset_amd_init(flash_info_t *info, struct cfi_qry *qry)
1547{
1548 info->cmd_reset = AMD_CMD_RESET;
1549
1550 cmdset_amd_read_jedec_ids(info);
1551 flash_write_cmd(info, 0, info->cfi_offset, FLASH_CMD_CFI);
1552
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001553 return 0;
1554}
1555
1556#ifdef CONFIG_FLASH_CFI_LEGACY
Stefan Roese260421a2006-11-13 13:55:24 +01001557static void flash_read_jedec_ids (flash_info_t * info)
1558{
1559 info->manufacturer_id = 0;
1560 info->device_id = 0;
1561 info->device_id2 = 0;
1562
1563 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001564 case CFI_CMDSET_INTEL_PROG_REGIONS:
Stefan Roese260421a2006-11-13 13:55:24 +01001565 case CFI_CMDSET_INTEL_STANDARD:
1566 case CFI_CMDSET_INTEL_EXTENDED:
Michael Schwingen8225d1e2008-01-12 20:29:47 +01001567 cmdset_intel_read_jedec_ids(info);
Stefan Roese260421a2006-11-13 13:55:24 +01001568 break;
1569 case CFI_CMDSET_AMD_STANDARD:
1570 case CFI_CMDSET_AMD_EXTENDED:
Michael Schwingen8225d1e2008-01-12 20:29:47 +01001571 cmdset_amd_read_jedec_ids(info);
Stefan Roese260421a2006-11-13 13:55:24 +01001572 break;
1573 default:
1574 break;
1575 }
1576}
1577
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001578/*-----------------------------------------------------------------------
1579 * Call board code to request info about non-CFI flash.
1580 * board_flash_get_legacy needs to fill in at least:
1581 * info->portwidth, info->chipwidth and info->interface for Jedec probing.
1582 */
1583static int flash_detect_legacy(ulong base, int banknum)
1584{
1585 flash_info_t *info = &flash_info[banknum];
1586
1587 if (board_flash_get_legacy(base, banknum, info)) {
1588 /* board code may have filled info completely. If not, we
1589 use JEDEC ID probing. */
1590 if (!info->vendor) {
1591 int modes[] = {
1592 CFI_CMDSET_AMD_STANDARD,
1593 CFI_CMDSET_INTEL_STANDARD
1594 };
1595 int i;
1596
1597 for (i = 0; i < sizeof(modes) / sizeof(modes[0]); i++) {
1598 info->vendor = modes[i];
1599 info->start[0] = base;
1600 if (info->portwidth == FLASH_CFI_8BIT
1601 && info->interface == FLASH_CFI_X8X16) {
1602 info->addr_unlock1 = 0x2AAA;
1603 info->addr_unlock2 = 0x5555;
1604 } else {
1605 info->addr_unlock1 = 0x5555;
1606 info->addr_unlock2 = 0x2AAA;
1607 }
1608 flash_read_jedec_ids(info);
1609 debug("JEDEC PROBE: ID %x %x %x\n",
1610 info->manufacturer_id,
1611 info->device_id,
1612 info->device_id2);
1613 if (jedec_flash_match(info, base))
1614 break;
1615 }
1616 }
1617
1618 switch(info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001619 case CFI_CMDSET_INTEL_PROG_REGIONS:
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001620 case CFI_CMDSET_INTEL_STANDARD:
1621 case CFI_CMDSET_INTEL_EXTENDED:
1622 info->cmd_reset = FLASH_CMD_RESET;
1623 break;
1624 case CFI_CMDSET_AMD_STANDARD:
1625 case CFI_CMDSET_AMD_EXTENDED:
1626 case CFI_CMDSET_AMD_LEGACY:
1627 info->cmd_reset = AMD_CMD_RESET;
1628 break;
1629 }
1630 info->flash_id = FLASH_MAN_CFI;
1631 return 1;
1632 }
1633 return 0; /* use CFI */
1634}
1635#else
1636static inline int flash_detect_legacy(ulong base, int banknum)
1637{
1638 return 0; /* use CFI */
1639}
1640#endif
1641
Stefan Roese260421a2006-11-13 13:55:24 +01001642/*-----------------------------------------------------------------------
wdenk5653fc32004-02-08 22:55:38 +00001643 * detect if flash is compatible with the Common Flash Interface (CFI)
1644 * http://www.jedec.org/download/search/jesd68.pdf
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001645 */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001646static void flash_read_cfi (flash_info_t *info, void *buf,
1647 unsigned int start, size_t len)
1648{
1649 u8 *p = buf;
1650 unsigned int i;
1651
1652 for (i = 0; i < len; i++)
1653 p[i] = flash_read_uchar(info, start + i);
1654}
1655
1656static int __flash_detect_cfi (flash_info_t * info, struct cfi_qry *qry)
wdenk5653fc32004-02-08 22:55:38 +00001657{
Wolfgang Denk92eb7292006-12-27 01:26:13 +01001658 int cfi_offset;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001659
Michael Schwingen1ba639d2008-02-18 23:16:35 +01001660 /* We do not yet know what kind of commandset to use, so we issue
1661 the reset command in both Intel and AMD variants, in the hope
1662 that AMD flash roms ignore the Intel command. */
1663 flash_write_cmd (info, 0, 0, AMD_CMD_RESET);
1664 flash_write_cmd (info, 0, 0, FLASH_CMD_RESET);
1665
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001666 for (cfi_offset=0;
1667 cfi_offset < sizeof(flash_offset_cfi) / sizeof(uint);
1668 cfi_offset++) {
1669 flash_write_cmd (info, 0, flash_offset_cfi[cfi_offset],
1670 FLASH_CMD_CFI);
1671 if (flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP, 'Q')
1672 && flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP + 1, 'R')
1673 && flash_isequal (info, 0, FLASH_OFFSET_CFI_RESP + 2, 'Y')) {
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001674 flash_read_cfi(info, qry, FLASH_OFFSET_CFI_RESP,
1675 sizeof(struct cfi_qry));
1676 info->interface = le16_to_cpu(qry->interface_desc);
1677
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001678 info->cfi_offset = flash_offset_cfi[cfi_offset];
1679 debug ("device interface is %d\n",
1680 info->interface);
1681 debug ("found port %d chip %d ",
1682 info->portwidth, info->chipwidth);
1683 debug ("port %d bits chip %d bits\n",
1684 info->portwidth << CFI_FLASH_SHIFT_WIDTH,
1685 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
1686
1687 /* calculate command offsets as in the Linux driver */
1688 info->addr_unlock1 = 0x555;
1689 info->addr_unlock2 = 0x2aa;
1690
1691 /*
1692 * modify the unlock address if we are
1693 * in compatibility mode
1694 */
1695 if ( /* x8/x16 in x8 mode */
1696 ((info->chipwidth == FLASH_CFI_BY8) &&
1697 (info->interface == FLASH_CFI_X8X16)) ||
1698 /* x16/x32 in x16 mode */
1699 ((info->chipwidth == FLASH_CFI_BY16) &&
1700 (info->interface == FLASH_CFI_X16X32)))
1701 {
1702 info->addr_unlock1 = 0xaaa;
1703 info->addr_unlock2 = 0x555;
1704 }
1705
1706 info->name = "CFI conformant";
1707 return 1;
1708 }
1709 }
1710
1711 return 0;
1712}
1713
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001714static int flash_detect_cfi (flash_info_t * info, struct cfi_qry *qry)
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001715{
wdenkbf9e3b32004-02-12 00:47:09 +00001716 debug ("flash detect cfi\n");
wdenk5653fc32004-02-08 22:55:38 +00001717
Stefan Roese79b4cda2006-02-28 15:29:58 +01001718 for (info->portwidth = CFG_FLASH_CFI_WIDTH;
wdenkbf9e3b32004-02-12 00:47:09 +00001719 info->portwidth <= FLASH_CFI_64BIT; info->portwidth <<= 1) {
1720 for (info->chipwidth = FLASH_CFI_BY8;
1721 info->chipwidth <= info->portwidth;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001722 info->chipwidth <<= 1)
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001723 if (__flash_detect_cfi(info, qry))
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001724 return 1;
wdenk5653fc32004-02-08 22:55:38 +00001725 }
wdenkbf9e3b32004-02-12 00:47:09 +00001726 debug ("not found\n");
wdenk5653fc32004-02-08 22:55:38 +00001727 return 0;
1728}
wdenkbf9e3b32004-02-12 00:47:09 +00001729
wdenk5653fc32004-02-08 22:55:38 +00001730/*
Haavard Skinnemoen467bcee2007-12-14 15:36:18 +01001731 * Manufacturer-specific quirks. Add workarounds for geometry
1732 * reversal, etc. here.
1733 */
1734static void flash_fixup_amd(flash_info_t *info, struct cfi_qry *qry)
1735{
1736 /* check if flash geometry needs reversal */
1737 if (qry->num_erase_regions > 1) {
1738 /* reverse geometry if top boot part */
1739 if (info->cfi_version < 0x3131) {
1740 /* CFI < 1.1, try to guess from device id */
1741 if ((info->device_id & 0x80) != 0)
1742 cfi_reverse_geometry(qry);
1743 } else if (flash_read_uchar(info, info->ext_addr + 0xf) == 3) {
1744 /* CFI >= 1.1, deduct from top/bottom flag */
1745 /* note: ext_addr is valid since cfi_version > 0 */
1746 cfi_reverse_geometry(qry);
1747 }
1748 }
1749}
1750
1751static void flash_fixup_atmel(flash_info_t *info, struct cfi_qry *qry)
1752{
1753 int reverse_geometry = 0;
1754
1755 /* Check the "top boot" bit in the PRI */
1756 if (info->ext_addr && !(flash_read_uchar(info, info->ext_addr + 6) & 1))
1757 reverse_geometry = 1;
1758
1759 /* AT49BV6416(T) list the erase regions in the wrong order.
1760 * However, the device ID is identical with the non-broken
1761 * AT49BV642D since u-boot only reads the low byte (they
1762 * differ in the high byte.) So leave out this fixup for now.
1763 */
1764#if 0
1765 if (info->device_id == 0xd6 || info->device_id == 0xd2)
1766 reverse_geometry = !reverse_geometry;
1767#endif
1768
1769 if (reverse_geometry)
1770 cfi_reverse_geometry(qry);
1771}
1772
1773/*
wdenk5653fc32004-02-08 22:55:38 +00001774 * The following code cannot be run from FLASH!
1775 *
1776 */
Marian Balakowicze6f2e902005-10-11 19:09:42 +02001777ulong flash_get_size (ulong base, int banknum)
wdenk5653fc32004-02-08 22:55:38 +00001778{
wdenkbf9e3b32004-02-12 00:47:09 +00001779 flash_info_t *info = &flash_info[banknum];
wdenk5653fc32004-02-08 22:55:38 +00001780 int i, j;
1781 flash_sect_t sect_cnt;
1782 unsigned long sector;
1783 unsigned long tmp;
1784 int size_ratio;
1785 uchar num_erase_regions;
wdenkbf9e3b32004-02-12 00:47:09 +00001786 int erase_region_size;
1787 int erase_region_count;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001788 struct cfi_qry qry;
Stefan Roese260421a2006-11-13 13:55:24 +01001789
Kumar Galaf9796902008-05-15 15:13:08 -05001790 memset(&qry, 0, sizeof(qry));
1791
Stefan Roese260421a2006-11-13 13:55:24 +01001792 info->ext_addr = 0;
1793 info->cfi_version = 0;
Stefan Roese2662b402006-04-01 13:41:03 +02001794#ifdef CFG_FLASH_PROTECTION
Stefan Roese2662b402006-04-01 13:41:03 +02001795 info->legacy_unlock = 0;
1796#endif
wdenk5653fc32004-02-08 22:55:38 +00001797
1798 info->start[0] = base;
1799
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001800 if (flash_detect_cfi (info, &qry)) {
1801 info->vendor = le16_to_cpu(qry.p_id);
1802 info->ext_addr = le16_to_cpu(qry.p_adr);
1803 num_erase_regions = qry.num_erase_regions;
1804
Stefan Roese260421a2006-11-13 13:55:24 +01001805 if (info->ext_addr) {
1806 info->cfi_version = (ushort) flash_read_uchar (info,
1807 info->ext_addr + 3) << 8;
1808 info->cfi_version |= (ushort) flash_read_uchar (info,
1809 info->ext_addr + 4);
1810 }
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001811
wdenkbf9e3b32004-02-12 00:47:09 +00001812#ifdef DEBUG
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001813 flash_printqry (&qry);
wdenkbf9e3b32004-02-12 00:47:09 +00001814#endif
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001815
wdenkbf9e3b32004-02-12 00:47:09 +00001816 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001817 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenk5653fc32004-02-08 22:55:38 +00001818 case CFI_CMDSET_INTEL_STANDARD:
1819 case CFI_CMDSET_INTEL_EXTENDED:
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001820 cmdset_intel_init(info, &qry);
wdenk5653fc32004-02-08 22:55:38 +00001821 break;
1822 case CFI_CMDSET_AMD_STANDARD:
1823 case CFI_CMDSET_AMD_EXTENDED:
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001824 cmdset_amd_init(info, &qry);
wdenk5653fc32004-02-08 22:55:38 +00001825 break;
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001826 default:
1827 printf("CFI: Unknown command set 0x%x\n",
1828 info->vendor);
1829 /*
1830 * Unfortunately, this means we don't know how
1831 * to get the chip back to Read mode. Might
1832 * as well try an Intel-style reset...
1833 */
1834 flash_write_cmd(info, 0, 0, FLASH_CMD_RESET);
1835 return 0;
wdenk5653fc32004-02-08 22:55:38 +00001836 }
wdenkcd37d9e2004-02-10 00:03:41 +00001837
Haavard Skinnemoen467bcee2007-12-14 15:36:18 +01001838 /* Do manufacturer-specific fixups */
1839 switch (info->manufacturer_id) {
1840 case 0x0001:
1841 flash_fixup_amd(info, &qry);
1842 break;
1843 case 0x001f:
1844 flash_fixup_atmel(info, &qry);
1845 break;
1846 }
1847
wdenkbf9e3b32004-02-12 00:47:09 +00001848 debug ("manufacturer is %d\n", info->vendor);
Stefan Roese260421a2006-11-13 13:55:24 +01001849 debug ("manufacturer id is 0x%x\n", info->manufacturer_id);
1850 debug ("device id is 0x%x\n", info->device_id);
1851 debug ("device id2 is 0x%x\n", info->device_id2);
1852 debug ("cfi version is 0x%04x\n", info->cfi_version);
1853
wdenk5653fc32004-02-08 22:55:38 +00001854 size_ratio = info->portwidth / info->chipwidth;
wdenkbf9e3b32004-02-12 00:47:09 +00001855 /* if the chip is x8/x16 reduce the ratio by half */
1856 if ((info->interface == FLASH_CFI_X8X16)
1857 && (info->chipwidth == FLASH_CFI_BY8)) {
1858 size_ratio >>= 1;
1859 }
wdenkbf9e3b32004-02-12 00:47:09 +00001860 debug ("size_ratio %d port %d bits chip %d bits\n",
1861 size_ratio, info->portwidth << CFI_FLASH_SHIFT_WIDTH,
1862 info->chipwidth << CFI_FLASH_SHIFT_WIDTH);
1863 debug ("found %d erase regions\n", num_erase_regions);
wdenk5653fc32004-02-08 22:55:38 +00001864 sect_cnt = 0;
1865 sector = base;
wdenkbf9e3b32004-02-12 00:47:09 +00001866 for (i = 0; i < num_erase_regions; i++) {
1867 if (i > NUM_ERASE_REGIONS) {
wdenk028ab6b2004-02-23 23:54:43 +00001868 printf ("%d erase regions found, only %d used\n",
1869 num_erase_regions, NUM_ERASE_REGIONS);
wdenk5653fc32004-02-08 22:55:38 +00001870 break;
1871 }
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001872
Haavard Skinnemoen0ddf06d2007-12-14 15:36:17 +01001873 tmp = le32_to_cpu(qry.erase_region_info[i]);
1874 debug("erase region %u: 0x%08lx\n", i, tmp);
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001875
1876 erase_region_count = (tmp & 0xffff) + 1;
1877 tmp >>= 16;
wdenkbf9e3b32004-02-12 00:47:09 +00001878 erase_region_size =
1879 (tmp & 0xffff) ? ((tmp & 0xffff) * 256) : 128;
wdenk4c0d4c32004-06-09 17:34:58 +00001880 debug ("erase_region_count = %d erase_region_size = %d\n",
wdenk028ab6b2004-02-23 23:54:43 +00001881 erase_region_count, erase_region_size);
wdenkbf9e3b32004-02-12 00:47:09 +00001882 for (j = 0; j < erase_region_count; j++) {
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001883 if (sect_cnt >= CFG_MAX_FLASH_SECT) {
1884 printf("ERROR: too many flash sectors\n");
1885 break;
1886 }
wdenk5653fc32004-02-08 22:55:38 +00001887 info->start[sect_cnt] = sector;
1888 sector += (erase_region_size * size_ratio);
wdenka1191902005-01-09 17:12:27 +00001889
1890 /*
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001891 * Only read protection status from
1892 * supported devices (intel...)
wdenka1191902005-01-09 17:12:27 +00001893 */
1894 switch (info->vendor) {
Vasiliy Leoenenko9c048b52008-05-07 21:25:33 +04001895 case CFI_CMDSET_INTEL_PROG_REGIONS:
wdenka1191902005-01-09 17:12:27 +00001896 case CFI_CMDSET_INTEL_EXTENDED:
1897 case CFI_CMDSET_INTEL_STANDARD:
1898 info->protect[sect_cnt] =
1899 flash_isset (info, sect_cnt,
1900 FLASH_OFFSET_PROTECT,
1901 FLASH_STATUS_PROTECT);
1902 break;
1903 default:
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001904 /* default: not protected */
1905 info->protect[sect_cnt] = 0;
wdenka1191902005-01-09 17:12:27 +00001906 }
1907
wdenk5653fc32004-02-08 22:55:38 +00001908 sect_cnt++;
1909 }
1910 }
1911
1912 info->sector_count = sect_cnt;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001913 info->size = 1 << qry.dev_size;
wdenk5653fc32004-02-08 22:55:38 +00001914 /* multiply the size by the number of chips */
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001915 info->size *= size_ratio;
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001916 info->buffer_size = 1 << le16_to_cpu(qry.max_buf_write_size);
1917 tmp = 1 << qry.block_erase_timeout_typ;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001918 info->erase_blk_tout = tmp *
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001919 (1 << qry.block_erase_timeout_max);
1920 tmp = (1 << qry.buf_write_timeout_typ) *
1921 (1 << qry.buf_write_timeout_max);
1922
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001923 /* round up when converting to ms */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001924 info->buffer_write_tout = (tmp + 999) / 1000;
1925 tmp = (1 << qry.word_write_timeout_typ) *
1926 (1 << qry.word_write_timeout_max);
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001927 /* round up when converting to ms */
Haavard Skinnemoene23741f2007-12-14 15:36:16 +01001928 info->write_tout = (tmp + 999) / 1000;
wdenk5653fc32004-02-08 22:55:38 +00001929 info->flash_id = FLASH_MAN_CFI;
Haavard Skinnemoen7e5b9b42007-12-13 12:56:28 +01001930 if ((info->interface == FLASH_CFI_X8X16) &&
1931 (info->chipwidth == FLASH_CFI_BY8)) {
1932 /* XXX - Need to test on x8/x16 in parallel. */
1933 info->portwidth >>= 1;
wdenk855a4962004-03-14 18:23:55 +00001934 }
wdenk5653fc32004-02-08 22:55:38 +00001935 }
1936
Wolfgang Denkdb421e62005-09-25 16:41:22 +02001937 flash_write_cmd (info, 0, 0, info->cmd_reset);
wdenkbf9e3b32004-02-12 00:47:09 +00001938 return (info->size);
wdenk5653fc32004-02-08 22:55:38 +00001939}
1940
wdenk5653fc32004-02-08 22:55:38 +00001941/*-----------------------------------------------------------------------
1942 */
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001943unsigned long flash_init (void)
wdenk5653fc32004-02-08 22:55:38 +00001944{
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001945 unsigned long size = 0;
1946 int i;
Matthias Fuchsc63ad632008-04-18 16:29:40 +02001947#if defined(CFG_FLASH_AUTOPROTECT_LIST)
1948 struct apl_s {
1949 ulong start;
1950 ulong size;
1951 } apl[] = CFG_FLASH_AUTOPROTECT_LIST;
1952#endif
wdenk5653fc32004-02-08 22:55:38 +00001953
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001954#ifdef CFG_FLASH_PROTECTION
1955 char *s = getenv("unlock");
Michael Schwingen81b20cc2007-12-07 23:35:02 +01001956#endif
wdenk5653fc32004-02-08 22:55:38 +00001957
Wolfgang Denk2a112b22008-08-08 16:39:54 +02001958#define BANK_BASE(i) (((unsigned long [CFI_MAX_FLASH_BANKS])CFG_FLASH_BANKS_LIST)[i])
1959
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001960 /* Init: no FLASHes known */
1961 for (i = 0; i < CFG_MAX_FLASH_BANKS; ++i) {
1962 flash_info[i].flash_id = FLASH_UNKNOWN;
wdenk5653fc32004-02-08 22:55:38 +00001963
Wolfgang Denk2a112b22008-08-08 16:39:54 +02001964 if (!flash_detect_legacy (BANK_BASE(i), i))
1965 flash_get_size (BANK_BASE(i), i);
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001966 size += flash_info[i].size;
1967 if (flash_info[i].flash_id == FLASH_UNKNOWN) {
1968#ifndef CFG_FLASH_QUIET_TEST
1969 printf ("## Unknown FLASH on Bank %d "
1970 "- Size = 0x%08lx = %ld MB\n",
1971 i+1, flash_info[i].size,
1972 flash_info[i].size << 20);
1973#endif /* CFG_FLASH_QUIET_TEST */
wdenk5653fc32004-02-08 22:55:38 +00001974 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001975#ifdef CFG_FLASH_PROTECTION
1976 else if ((s != NULL) && (strcmp(s, "yes") == 0)) {
1977 /*
1978 * Only the U-Boot image and it's environment
1979 * is protected, all other sectors are
1980 * unprotected (unlocked) if flash hardware
1981 * protection is used (CFG_FLASH_PROTECTION)
1982 * and the environment variable "unlock" is
1983 * set to "yes".
1984 */
1985 if (flash_info[i].legacy_unlock) {
1986 int k;
Stefan Roese79b4cda2006-02-28 15:29:58 +01001987
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001988 /*
1989 * Disable legacy_unlock temporarily,
1990 * since flash_real_protect would
1991 * relock all other sectors again
1992 * otherwise.
1993 */
1994 flash_info[i].legacy_unlock = 0;
Stefan Roese79b4cda2006-02-28 15:29:58 +01001995
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02001996 /*
1997 * Legacy unlocking (e.g. Intel J3) ->
1998 * unlock only one sector. This will
1999 * unlock all sectors.
2000 */
2001 flash_real_protect (&flash_info[i], 0, 0);
Stefan Roese79b4cda2006-02-28 15:29:58 +01002002
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002003 flash_info[i].legacy_unlock = 1;
2004
2005 /*
2006 * Manually mark other sectors as
2007 * unlocked (unprotected)
2008 */
2009 for (k = 1; k < flash_info[i].sector_count; k++)
2010 flash_info[i].protect[k] = 0;
2011 } else {
2012 /*
2013 * No legancy unlocking -> unlock all sectors
2014 */
2015 flash_protect (FLAG_PROTECT_CLEAR,
2016 flash_info[i].start[0],
2017 flash_info[i].start[0]
2018 + flash_info[i].size - 1,
2019 &flash_info[i]);
2020 }
Stefan Roese79b4cda2006-02-28 15:29:58 +01002021 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002022#endif /* CFG_FLASH_PROTECTION */
wdenk5653fc32004-02-08 22:55:38 +00002023 }
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002024
2025 /* Monitor protection ON by default */
2026#if (CFG_MONITOR_BASE >= CFG_FLASH_BASE)
2027 flash_protect (FLAG_PROTECT_SET,
2028 CFG_MONITOR_BASE,
2029 CFG_MONITOR_BASE + monitor_flash_len - 1,
2030 flash_get_info(CFG_MONITOR_BASE));
2031#endif
2032
2033 /* Environment protection ON by default */
Jean-Christophe PLAGNIOL-VILLARD5a1aceb2008-09-10 22:48:04 +02002034#ifdef CONFIG_ENV_IS_IN_FLASH
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002035 flash_protect (FLAG_PROTECT_SET,
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002036 CONFIG_ENV_ADDR,
2037 CONFIG_ENV_ADDR + CONFIG_ENV_SECT_SIZE - 1,
2038 flash_get_info(CONFIG_ENV_ADDR));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002039#endif
2040
2041 /* Redundant environment protection ON by default */
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002042#ifdef CONFIG_ENV_ADDR_REDUND
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002043 flash_protect (FLAG_PROTECT_SET,
Jean-Christophe PLAGNIOL-VILLARD0e8d1582008-09-10 22:48:06 +02002044 CONFIG_ENV_ADDR_REDUND,
2045 CONFIG_ENV_ADDR_REDUND + CONFIG_ENV_SIZE_REDUND - 1,
2046 flash_get_info(CONFIG_ENV_ADDR_REDUND));
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002047#endif
Matthias Fuchsc63ad632008-04-18 16:29:40 +02002048
2049#if defined(CFG_FLASH_AUTOPROTECT_LIST)
2050 for (i = 0; i < (sizeof(apl) / sizeof(struct apl_s)); i++) {
2051 debug("autoprotecting from %08x to %08x\n",
2052 apl[i].start, apl[i].start + apl[i].size - 1);
2053 flash_protect (FLAG_PROTECT_SET,
2054 apl[i].start,
2055 apl[i].start + apl[i].size - 1,
2056 flash_get_info(apl[i].start));
2057 }
2058#endif
Haavard Skinnemoenbe60a902007-10-06 18:55:36 +02002059 return (size);
wdenk5653fc32004-02-08 22:55:38 +00002060}