Blame view

board/jse/flash.c 11.7 KB
db01a2ea9   wdenk   * Patch by Stephe...
1
2
3
4
  /*
   * (C) Copyright 2000-2004
   * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
   *
1a4596601   Wolfgang Denk   Add GPL-2.0+ SPDX...
5
   * SPDX-License-Identifier:	GPL-2.0+
db01a2ea9   wdenk   * Patch by Stephe...
6
7
8
9
10
11
12
13
14
15
   */
  
  /*
   * Modified 4/5/2001
   * Wait for completion of each sector erase command issued
   * 4/5/2001
   * Chris Hallinan - DS4.COM, Inc. - clh@net1plus.com
   */
  
  #include <common.h>
b36df5611   Stefan Roese   ppc4xx: Move ppc4...
16
  #include <asm/ppc4xx.h>
db01a2ea9   wdenk   * Patch by Stephe...
17
  #include <asm/processor.h>
6d0f6bcf3   Jean-Christophe PLAGNIOL-VILLARD   rename CFG_ macro...
18
19
  #if CONFIG_SYS_MAX_FLASH_BANKS != 1
  #error "CONFIG_SYS_MAX_FLASH_BANKS must be 1"
db01a2ea9   wdenk   * Patch by Stephe...
20
  #endif
6d0f6bcf3   Jean-Christophe PLAGNIOL-VILLARD   rename CFG_ macro...
21
  flash_info_t flash_info[CONFIG_SYS_MAX_FLASH_BANKS];	/* info for FLASH chips	*/
db01a2ea9   wdenk   * Patch by Stephe...
22
23
24
25
26
27
28
  
  /*-----------------------------------------------------------------------
   * Functions
   */
  static ulong flash_get_size (vu_long * addr, flash_info_t * info);
  static int write_word (flash_info_t * info, ulong dest, ulong data);
  static void flash_get_offsets (ulong base, flash_info_t * info);
6bdd1377a   wdenk   Patch by Stephen ...
29
30
  #define ADDR0		0x5555
  #define ADDR1		0x2aaa
db01a2ea9   wdenk   * Patch by Stephe...
31
32
33
34
35
36
37
38
  #define FLASH_WORD_SIZE unsigned char
  
  /*-----------------------------------------------------------------------
   */
  
  unsigned long flash_init (void)
  {
  	unsigned long size_b0;
db01a2ea9   wdenk   * Patch by Stephe...
39
40
41
42
43
  
  	/* Init: no FLASHes known */
  	flash_info[0].flash_id = FLASH_UNKNOWN;
  
  	/* Static FLASH Bank configuration here - FIXME XXX */
08f1080c9   wdenk   Make compile clean.
44
  	size_b0 = flash_get_size ((vu_long *) FLASH_BASE0_PRELIM, &flash_info[0]);
db01a2ea9   wdenk   * Patch by Stephe...
45
46
  
  	if (flash_info[0].flash_id == FLASH_UNKNOWN) {
08f1080c9   wdenk   Make compile clean.
47
48
49
  		printf ("## Unknown FLASH on Bank 0 - Size = 0x%08lx = %ld MB
  ",
  			size_b0, size_b0 << 20);
db01a2ea9   wdenk   * Patch by Stephe...
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
  	}
  
  	/* Only one bank */
  	/* Setup offsets */
  	flash_get_offsets (FLASH_BASE0_PRELIM, &flash_info[0]);
  
  	/* Monitor protection ON by default */
  	(void) flash_protect (FLAG_PROTECT_SET,
  			      FLASH_BASE0_PRELIM,
  			      FLASH_BASE0_PRELIM + monitor_flash_len - 1,
  			      &flash_info[0]);
  	flash_info[0].size = size_b0;
  
  	return size_b0;
  }
  
  
  /*-----------------------------------------------------------------------
   */
6bdd1377a   wdenk   Patch by Stephen ...
69
70
71
72
  /*
   * This implementation assumes that the flash chips are uniform sector
   * devices. This is true for all likely JSE devices.
   */
db01a2ea9   wdenk   * Patch by Stephe...
73
74
  static void flash_get_offsets (ulong base, flash_info_t * info)
  {
6bdd1377a   wdenk   Patch by Stephen ...
75
76
  	unsigned idx;
  	unsigned long sector_size = info->size / info->sector_count;
db01a2ea9   wdenk   * Patch by Stephe...
77

6bdd1377a   wdenk   Patch by Stephen ...
78
79
  	for (idx = 0; idx < info->sector_count; idx += 1) {
  		info->start[idx] = base + (idx * sector_size);
db01a2ea9   wdenk   * Patch by Stephe...
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
  	}
  }
  
  /*-----------------------------------------------------------------------
   */
  void flash_print_info (flash_info_t * info)
  {
  	int i;
  	int k;
  	int size;
  	int erased;
  	volatile unsigned long *flash;
  
  	if (info->flash_id == FLASH_UNKNOWN) {
  		printf ("missing or unknown FLASH type
  ");
  		return;
  	}
  
  	switch (info->flash_id & FLASH_VENDMASK) {
  	case FLASH_MAN_AMD:
  		printf ("AMD ");
  		break;
  	case FLASH_MAN_FUJ:
  		printf ("FUJITSU ");
  		break;
  	case FLASH_MAN_SST:
  		printf ("SST ");
  		break;
6bdd1377a   wdenk   Patch by Stephen ...
109
110
111
  	case FLASH_MAN_STM:
  		printf ("ST Micro ");
  		break;
db01a2ea9   wdenk   * Patch by Stephe...
112
113
114
115
  	default:
  		printf ("Unknown Vendor ");
  		break;
  	}
6bdd1377a   wdenk   Patch by Stephen ...
116
117
118
  	  /* (Reduced table of only parts expected in JSE boards.) */
  	switch (info->flash_id) {
  	case FLASH_MAN_AMD | FLASH_AM040:
db01a2ea9   wdenk   * Patch by Stephe...
119
120
121
  		printf ("AM29F040 (512 Kbit, uniform sector size)
  ");
  		break;
6bdd1377a   wdenk   Patch by Stephen ...
122
123
124
  	case FLASH_MAN_STM | FLASH_AM040:
  		printf ("MM29W040W (512 Kbit, uniform sector size)
  ");
db01a2ea9   wdenk   * Patch by Stephe...
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
  		break;
  	default:
  		printf ("Unknown Chip Type
  ");
  		break;
  	}
  
  	printf ("  Size: %ld KB in %d Sectors
  ",
  		info->size >> 10, info->sector_count);
  
  	printf ("  Sector Start Addresses:");
  	for (i = 0; i < info->sector_count; ++i) {
  		/*
  		 * Check if whole sector is erased
  		 */
  		if (i != (info->sector_count - 1))
  			size = info->start[i + 1] - info->start[i];
  		else
  			size = info->start[0] + info->size - info->start[i];
  		erased = 1;
  		flash = (volatile unsigned long *) info->start[i];
  		size = size >> 2;	/* divide by 4 for longword access */
  		for (k = 0; k < size; k++) {
  			if (*flash++ != 0xffffffff) {
  				erased = 0;
  				break;
  			}
  		}
  
  		if ((i % 5) == 0)
  			printf ("
     ");
db01a2ea9   wdenk   * Patch by Stephe...
158
159
160
  		printf (" %08lX%s%s",
  			info->start[i],
  			erased ? " E" : "  ", info->protect[i] ? "RO " : "   "
08f1080c9   wdenk   Make compile clean.
161
  		);
db01a2ea9   wdenk   * Patch by Stephe...
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
  	}
  	printf ("
  ");
  	return;
  }
  
  /*-----------------------------------------------------------------------
   */
  
  
  /*-----------------------------------------------------------------------
   */
  
  /*
   * The following code cannot be run from FLASH!
   */
  static ulong flash_get_size (vu_long * addr, flash_info_t * info)
  {
  	short i;
  	FLASH_WORD_SIZE value;
  	ulong base = (ulong) addr;
  	volatile FLASH_WORD_SIZE *addr2 = (FLASH_WORD_SIZE *) addr;
  
  	/* Write auto select command: read Manufacturer ID */
  	addr2[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  	addr2[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  	addr2[ADDR0] = (FLASH_WORD_SIZE) 0x00900090;
db01a2ea9   wdenk   * Patch by Stephe...
189
  	value = addr2[0];
db01a2ea9   wdenk   * Patch by Stephe...
190
191
192
193
194
195
196
197
198
199
200
  
  	switch (value) {
  	case (FLASH_WORD_SIZE) AMD_MANUFACT:
  		info->flash_id = FLASH_MAN_AMD;
  		break;
  	case (FLASH_WORD_SIZE) FUJ_MANUFACT:
  		info->flash_id = FLASH_MAN_FUJ;
  		break;
  	case (FLASH_WORD_SIZE) SST_MANUFACT:
  		info->flash_id = FLASH_MAN_SST;
  		break;
6bdd1377a   wdenk   Patch by Stephen ...
201
202
203
  	case (FLASH_WORD_SIZE)STM_MANUFACT:
  		info->flash_id = FLASH_MAN_STM;
  		break;
db01a2ea9   wdenk   * Patch by Stephe...
204
205
206
207
  	default:
  		info->flash_id = FLASH_UNKNOWN;
  		info->sector_count = 0;
  		info->size = 0;
6bdd1377a   wdenk   Patch by Stephen ...
208
209
  		printf("Unknown flash manufacturer code: 0x%x
  ", value);
db01a2ea9   wdenk   * Patch by Stephe...
210
211
  		return (0);	/* no or unknown flash  */
  	}
6bdd1377a   wdenk   Patch by Stephen ...
212
  	value = addr2[1];	/* device ID		*/
db01a2ea9   wdenk   * Patch by Stephe...
213
214
215
216
217
218
219
220
221
222
223
224
  
  	switch (value) {
  	case (FLASH_WORD_SIZE) AMD_ID_F040B:
  		info->flash_id += FLASH_AM040;
  		info->sector_count = 8;
  		info->size = 0x0080000;	/* => 512 ko */
  		break;
  	case (FLASH_WORD_SIZE) AMD_ID_LV040B:
  		info->flash_id += FLASH_AM040;
  		info->sector_count = 8;
  		info->size = 0x0080000;	/* => 512 ko */
  		break;
6bdd1377a   wdenk   Patch by Stephen ...
225
226
227
228
229
  	case (FLASH_WORD_SIZE)STM_ID_M29W040B: /* most likele JSE chip */
  		info->flash_id += FLASH_AM040;
  		info->sector_count = 8;
  		info->size = 0x0080000; /* => 512 ko */
  		break;
db01a2ea9   wdenk   * Patch by Stephe...
230
231
232
233
234
  	default:
  		info->flash_id = FLASH_UNKNOWN;
  		return (0);	/* => no or unknown flash */
  
  	}
6bdd1377a   wdenk   Patch by Stephen ...
235
236
  	  /* Calculate the sector offsets (Use JSE Optimized code). */
  	flash_get_offsets(base, info);
db01a2ea9   wdenk   * Patch by Stephe...
237
238
239
240
241
  
  	/* check for protected sectors */
  	for (i = 0; i < info->sector_count; i++) {
  		/* read sector protection at sector address, (A7 .. A0) = 0x02 */
  		/* D0 = 1 if protected */
db01a2ea9   wdenk   * Patch by Stephe...
242
243
244
245
246
  		addr2 = (volatile FLASH_WORD_SIZE *) (info->start[i]);
  		if ((info->flash_id & FLASH_VENDMASK) == FLASH_MAN_SST)
  			info->protect[i] = 0;
  		else
  			info->protect[i] = addr2[2] & 1;
db01a2ea9   wdenk   * Patch by Stephe...
247
248
249
250
251
252
  	}
  
  	/*
  	 * Prevent writes to uninitialized FLASH.
  	 */
  	if (info->flash_id != FLASH_UNKNOWN) {
db01a2ea9   wdenk   * Patch by Stephe...
253
254
  		addr2 = (FLASH_WORD_SIZE *) info->start[0];
  		*addr2 = (FLASH_WORD_SIZE) 0x00F000F0;	/* reset bank */
db01a2ea9   wdenk   * Patch by Stephe...
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
  	}
  
  	return (info->size);
  }
  
  int wait_for_DQ7 (flash_info_t * info, int sect)
  {
  	ulong start, now, last;
  	volatile FLASH_WORD_SIZE *addr =
  		(FLASH_WORD_SIZE *) (info->start[sect]);
  
  	start = get_timer (0);
  	last = start;
  	while ((addr[0] & (FLASH_WORD_SIZE) 0x00800080) !=
  	       (FLASH_WORD_SIZE) 0x00800080) {
6d0f6bcf3   Jean-Christophe PLAGNIOL-VILLARD   rename CFG_ macro...
270
  		if ((now = get_timer (start)) > CONFIG_SYS_FLASH_ERASE_TOUT) {
db01a2ea9   wdenk   * Patch by Stephe...
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
  			printf ("Timeout
  ");
  			return -1;
  		}
  		/* show that we're waiting */
  		if ((now - last) > 1000) {	/* every second */
  			putc ('.');
  			last = now;
  		}
  	}
  	return 0;
  }
  
  /*-----------------------------------------------------------------------
   */
  
  int flash_erase (flash_info_t * info, int s_first, int s_last)
  {
  	volatile FLASH_WORD_SIZE *addr = (FLASH_WORD_SIZE *) (info->start[0]);
  	volatile FLASH_WORD_SIZE *addr2;
8e119b03c   Wolfgang Denk   board/jse/flash.c...
291
  	int flag, prot, sect;
db01a2ea9   wdenk   * Patch by Stephe...
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
  	int i;
  
  	if ((s_first < 0) || (s_first > s_last)) {
  		if (info->flash_id == FLASH_UNKNOWN) {
  			printf ("- missing
  ");
  		} else {
  			printf ("- no sectors to erase
  ");
  		}
  		return 1;
  	}
  
  	if (info->flash_id == FLASH_UNKNOWN) {
  		printf ("Can't erase unknown flash type - aborted
  ");
  		return 1;
  	}
  
  	prot = 0;
  	for (sect = s_first; sect <= s_last; ++sect) {
  		if (info->protect[sect]) {
  			prot++;
  		}
  	}
  
  	if (prot) {
  		printf ("- Warning: %d protected sectors will not be erased!
  ", prot);
  	} else {
  		printf ("
  ");
  	}
db01a2ea9   wdenk   * Patch by Stephe...
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
  	/* Disable interrupts which might cause a timeout here */
  	flag = disable_interrupts ();
  
  	/* Start erase on unprotected sectors */
  	for (sect = s_first; sect <= s_last; sect++) {
  		if (info->protect[sect] == 0) {	/* not protected */
  			addr2 = (FLASH_WORD_SIZE *) (info->start[sect]);
  			printf ("Erasing sector %p
  ", addr2);	/* CLH */
  
  			if ((info->flash_id & FLASH_VENDMASK) ==
  			    FLASH_MAN_SST) {
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  				addr[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00800080;
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  				addr[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  				addr2[0] = (FLASH_WORD_SIZE) 0x00500050;	/* block erase */
  				for (i = 0; i < 50; i++)
  					udelay (1000);	/* wait 1 ms */
  			} else {
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  				addr[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00800080;
  				addr[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  				addr[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  				addr2[0] = (FLASH_WORD_SIZE) 0x00300030;	/* sector erase */
  			}
db01a2ea9   wdenk   * Patch by Stephe...
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
  			/*
  			 * Wait for each sector to complete, it's more
  			 * reliable.  According to AMD Spec, you must
  			 * issue all erase commands within a specified
  			 * timeout.  This has been seen to fail, especially
  			 * if printf()s are included (for debug)!!
  			 */
  			wait_for_DQ7 (info, sect);
  		}
  	}
  
  	/* re-enable interrupts if necessary */
  	if (flag)
  		enable_interrupts ();
  
  	/* wait at least 80us - let's wait 1 ms */
  	udelay (1000);
db01a2ea9   wdenk   * Patch by Stephe...
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
  	/* reset to read mode */
  	addr = (FLASH_WORD_SIZE *) info->start[0];
  	addr[0] = (FLASH_WORD_SIZE) 0x00F000F0;	/* reset bank */
  
  	printf (" done
  ");
  	return 0;
  }
  
  /*-----------------------------------------------------------------------
   * Copy memory to flash, returns:
   * 0 - OK
   * 1 - write timeout
   * 2 - Flash not erased
   */
  
  int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt)
  {
  	ulong cp, wp, data;
  	int i, l, rc;
  
  	wp = (addr & ~3);	/* get lower word aligned address */
  
  	/*
  	 * handle unaligned start bytes
  	 */
  	if ((l = addr - wp) != 0) {
  		data = 0;
  		for (i = 0, cp = wp; i < l; ++i, ++cp) {
  			data = (data << 8) | (*(uchar *) cp);
  		}
  		for (; i < 4 && cnt > 0; ++i) {
  			data = (data << 8) | *src++;
  			--cnt;
  			++cp;
  		}
  		for (; cnt == 0 && i < 4; ++i, ++cp) {
  			data = (data << 8) | (*(uchar *) cp);
  		}
  
  		if ((rc = write_word (info, wp, data)) != 0) {
  			return (rc);
  		}
  		wp += 4;
  	}
  
  	/*
  	 * handle word aligned part
  	 */
  	while (cnt >= 4) {
  		data = 0;
  		for (i = 0; i < 4; ++i) {
  			data = (data << 8) | *src++;
  		}
  		if ((rc = write_word (info, wp, data)) != 0) {
  			return (rc);
  		}
  		wp += 4;
  		cnt -= 4;
  	}
  
  	if (cnt == 0) {
  		return (0);
  	}
  
  	/*
  	 * handle unaligned tail bytes
  	 */
  	data = 0;
  	for (i = 0, cp = wp; i < 4 && cnt > 0; ++i, ++cp) {
  		data = (data << 8) | *src++;
  		--cnt;
  	}
  	for (; i < 4; ++i, ++cp) {
  		data = (data << 8) | (*(uchar *) cp);
  	}
  
  	return (write_word (info, wp, data));
  }
  
  /*-----------------------------------------------------------------------
   * Write a word to Flash, returns:
   * 0 - OK
   * 1 - write timeout
   * 2 - Flash not erased
   */
  static int write_word (flash_info_t * info, ulong dest, ulong data)
  {
  	volatile FLASH_WORD_SIZE *addr2 =
  		(FLASH_WORD_SIZE *) (info->start[0]);
  	volatile FLASH_WORD_SIZE *dest2 = (FLASH_WORD_SIZE *) dest;
  	volatile FLASH_WORD_SIZE *data2 = (FLASH_WORD_SIZE *) & data;
  	ulong start;
  	int i;
  
  	/* Check if Flash is (sufficiently) erased */
  	if ((*((volatile FLASH_WORD_SIZE *) dest) &
6bdd1377a   wdenk   Patch by Stephen ...
467
  	    (FLASH_WORD_SIZE) data) != (FLASH_WORD_SIZE) data) {
db01a2ea9   wdenk   * Patch by Stephe...
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
  		return (2);
  	}
  
  	for (i = 0; i < 4 / sizeof (FLASH_WORD_SIZE); i++) {
  		int flag;
  
  		/* Disable interrupts which might cause a timeout here */
  		flag = disable_interrupts ();
  
  		addr2[ADDR0] = (FLASH_WORD_SIZE) 0x00AA00AA;
  		addr2[ADDR1] = (FLASH_WORD_SIZE) 0x00550055;
  		addr2[ADDR0] = (FLASH_WORD_SIZE) 0x00A000A0;
  
  		dest2[i] = data2[i];
  
  		/* re-enable interrupts if necessary */
  		if (flag)
  			enable_interrupts ();
  
  		/* data polling for D7 */
  		start = get_timer (0);
  		while ((dest2[i] & (FLASH_WORD_SIZE) 0x00800080) !=
  		       (data2[i] & (FLASH_WORD_SIZE) 0x00800080)) {
6d0f6bcf3   Jean-Christophe PLAGNIOL-VILLARD   rename CFG_ macro...
491
  			if (get_timer (start) > CONFIG_SYS_FLASH_WRITE_TOUT) {
db01a2ea9   wdenk   * Patch by Stephe...
492
493
494
495
496
497
498
  				return (1);
  			}
  		}
  	}
  
  	return (0);
  }