Blame view

drivers/mtd/chips/cfi_cmdset_0002.c 55.1 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
  /*
   * Common Flash Interface support:
   *   AMD & Fujitsu Standard Vendor Command Set (ID 0x0002)
   *
   * Copyright (C) 2000 Crossnet Co. <info@crossnet.co.jp>
   * Copyright (C) 2004 Arcom Control Systems Ltd <linux@arcom.com>
02b15e343   Todd Poynor   [MTD] XIP for AMD...
7
   * Copyright (C) 2005 MontaVista Software Inc. <source@mvista.com>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
8
9
10
11
12
   *
   * 2_by_8 routines added by Simon Munton
   *
   * 4_by_16 work by Carolyn J. Smith
   *
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
13
   * XIP support hooks by Vitaly Wool (based on code for Intel flash
02b15e343   Todd Poynor   [MTD] XIP for AMD...
14
   * by Nicolas Pitre)
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
15
   *
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
16
17
   * 25/09/2008 Christopher Moore: TopBottom fixup for many Macronix with CFI V1.0
   *
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18
19
20
   * Occasionally maintained by Thayne Harbaugh tharbaugh at lnxi dot com
   *
   * This code is GPL
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
21
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
23
24
25
26
27
28
29
30
31
32
33
  #include <linux/module.h>
  #include <linux/types.h>
  #include <linux/kernel.h>
  #include <linux/sched.h>
  #include <linux/init.h>
  #include <asm/io.h>
  #include <asm/byteorder.h>
  
  #include <linux/errno.h>
  #include <linux/slab.h>
  #include <linux/delay.h>
  #include <linux/interrupt.h>
eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
34
  #include <linux/reboot.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
36
37
  #include <linux/mtd/map.h>
  #include <linux/mtd/mtd.h>
  #include <linux/mtd/cfi.h>
02b15e343   Todd Poynor   [MTD] XIP for AMD...
38
  #include <linux/mtd/xip.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
39
40
41
42
43
  
  #define AMD_BOOTLOC_BUG
  #define FORCE_WORD_WRITE 0
  
  #define MAX_WORD_RETRIES 3
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
44
  #define SST49LF004B	        0x0060
89072ef99   Ryan Jackson   [MTD] CHIPS: Supp...
45
  #define SST49LF040B	        0x0050
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
46
  #define SST49LF008A		0x005a
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
47
  #define AT49BV6416		0x00d6
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
48
49
50
51
52
53
54
55
56
  
  static int cfi_amdstd_read (struct mtd_info *, loff_t, size_t, size_t *, u_char *);
  static int cfi_amdstd_write_words(struct mtd_info *, loff_t, size_t, size_t *, const u_char *);
  static int cfi_amdstd_write_buffers(struct mtd_info *, loff_t, size_t, size_t *, const u_char *);
  static int cfi_amdstd_erase_chip(struct mtd_info *, struct erase_info *);
  static int cfi_amdstd_erase_varsize(struct mtd_info *, struct erase_info *);
  static void cfi_amdstd_sync (struct mtd_info *);
  static int cfi_amdstd_suspend (struct mtd_info *);
  static void cfi_amdstd_resume (struct mtd_info *);
eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
57
  static int cfi_amdstd_reboot(struct notifier_block *, unsigned long, void *);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
58
59
60
61
62
63
64
65
66
67
  static int cfi_amdstd_secsi_read (struct mtd_info *, loff_t, size_t, size_t *, u_char *);
  
  static void cfi_amdstd_destroy(struct mtd_info *);
  
  struct mtd_info *cfi_cmdset_0002(struct map_info *, int);
  static struct mtd_info *cfi_amdstd_setup (struct mtd_info *);
  
  static int get_chip(struct map_info *map, struct flchip *chip, unsigned long adr, int mode);
  static void put_chip(struct map_info *map, struct flchip *chip, unsigned long adr);
  #include "fwh_lock.h"
69423d99f   Adrian Hunter   [MTD] update inte...
68
69
  static int cfi_atmel_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len);
  static int cfi_atmel_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
70

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
  static struct mtd_chip_driver cfi_amdstd_chipdrv = {
  	.probe		= NULL, /* Not usable directly */
  	.destroy	= cfi_amdstd_destroy,
  	.name		= "cfi_cmdset_0002",
  	.module		= THIS_MODULE
  };
  
  
  /* #define DEBUG_CFI_FEATURES */
  
  
  #ifdef DEBUG_CFI_FEATURES
  static void cfi_tell_features(struct cfi_pri_amdstd *extp)
  {
  	const char* erase_suspend[3] = {
  		"Not supported", "Read only", "Read/write"
  	};
  	const char* top_bottom[6] = {
  		"No WP", "8x8KiB sectors at top & bottom, no WP",
  		"Bottom boot", "Top boot",
  		"Uniform, Bottom WP", "Uniform, Top WP"
  	};
  
  	printk("  Silicon revision: %d
  ", extp->SiliconRevision >> 1);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
96
97
  	printk("  Address sensitive unlock: %s
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
  	       (extp->SiliconRevision & 1) ? "Not required" : "Required");
  
  	if (extp->EraseSuspend < ARRAY_SIZE(erase_suspend))
  		printk("  Erase Suspend: %s
  ", erase_suspend[extp->EraseSuspend]);
  	else
  		printk("  Erase Suspend: Unknown value %d
  ", extp->EraseSuspend);
  
  	if (extp->BlkProt == 0)
  		printk("  Block protection: Not supported
  ");
  	else
  		printk("  Block protection: %d sectors per group
  ", extp->BlkProt);
  
  
  	printk("  Temporary block unprotect: %s
  ",
  	       extp->TmpBlkUnprotect ? "Supported" : "Not supported");
  	printk("  Block protect/unprotect scheme: %d
  ", extp->BlkProtUnprot);
  	printk("  Number of simultaneous operations: %d
  ", extp->SimultaneousOps);
  	printk("  Burst mode: %s
  ",
  	       extp->BurstMode ? "Supported" : "Not supported");
  	if (extp->PageMode == 0)
  		printk("  Page mode: Not supported
  ");
  	else
  		printk("  Page mode: %d word page
  ", extp->PageMode << 2);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
131
132
  	printk("  Vpp Supply Minimum Program/Erase Voltage: %d.%d V
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
133
  	       extp->VppMin >> 4, extp->VppMin & 0xf);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
134
135
  	printk("  Vpp Supply Maximum Program/Erase Voltage: %d.%d V
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
136
137
138
139
140
141
142
143
144
145
146
147
148
  	       extp->VppMax >> 4, extp->VppMax & 0xf);
  
  	if (extp->TopBottom < ARRAY_SIZE(top_bottom))
  		printk("  Top/Bottom Boot Block: %s
  ", top_bottom[extp->TopBottom]);
  	else
  		printk("  Top/Bottom Boot Block: Unknown value %d
  ", extp->TopBottom);
  }
  #endif
  
  #ifdef AMD_BOOTLOC_BUG
  /* Wheee. Bring me the head of someone at AMD. */
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
149
  static void fixup_amd_bootblock(struct mtd_info *mtd)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
150
151
152
153
154
155
156
157
158
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	struct cfi_pri_amdstd *extp = cfi->cmdset_priv;
  	__u8 major = extp->MajorVersion;
  	__u8 minor = extp->MinorVersion;
  
  	if (((major << 8) | minor) < 0x3131) {
  		/* CFI version 1.0 => don't trust bootloc */
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
159

289c05222   Brian Norris   mtd: replace DEBU...
160
161
  		pr_debug("%s: JEDEC Vendor ID is 0x%02X Device ID is 0x%02X
  ",
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
  			map->name, cfi->mfr, cfi->id);
  
  		/* AFAICS all 29LV400 with a bottom boot block have a device ID
  		 * of 0x22BA in 16-bit mode and 0xBA in 8-bit mode.
  		 * These were badly detected as they have the 0x80 bit set
  		 * so treat them as a special case.
  		 */
  		if (((cfi->id == 0xBA) || (cfi->id == 0x22BA)) &&
  
  			/* Macronix added CFI to their 2nd generation
  			 * MX29LV400C B/T but AFAICS no other 29LV400 (AMD,
  			 * Fujitsu, Spansion, EON, ESI and older Macronix)
  			 * has CFI.
  			 *
  			 * Therefore also check the manufacturer.
  			 * This reduces the risk of false detection due to
  			 * the 8-bit device ID.
  			 */
f3e69c658   Guillaume LECERF   mtd: move more ma...
180
  			(cfi->mfr == CFI_MFR_MACRONIX)) {
289c05222   Brian Norris   mtd: replace DEBU...
181
  			pr_debug("%s: Macronix MX29LV400C with bottom boot block"
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
182
183
184
185
  				" detected
  ", map->name);
  			extp->TopBottom = 2;	/* bottom boot */
  		} else
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
186
187
188
189
190
191
192
  		if (cfi->id & 0x80) {
  			printk(KERN_WARNING "%s: JEDEC Device ID is 0x%02X. Assuming broken CFI table.
  ", map->name, cfi->id);
  			extp->TopBottom = 3;	/* top boot */
  		} else {
  			extp->TopBottom = 2;	/* bottom boot */
  		}
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
193

289c05222   Brian Norris   mtd: replace DEBU...
194
  		pr_debug("%s: AMD CFI PRI V%c.%c has no boot block field;"
87e92c062   Christopher Moore   [MTD] cfi_cmdset_...
195
196
197
  			" deduced %s from Device ID
  ", map->name, major, minor,
  			extp->TopBottom == 2 ? "bottom" : "top");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198
199
200
  	}
  }
  #endif
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
201
  static void fixup_use_write_buffers(struct mtd_info *mtd)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
202
203
204
205
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	if (cfi->cfiq->BufWriteTimeoutTyp) {
289c05222   Brian Norris   mtd: replace DEBU...
206
207
  		pr_debug("Using buffer write method
  " );
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
208
209
210
  		mtd->write = cfi_amdstd_write_buffers;
  	}
  }
5b0c5c2c0   Haavard Skinnemoen   MTD: Convert Atme...
211
  /* Atmel chips don't use the same PRI format as AMD chips */
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
212
  static void fixup_convert_atmel_pri(struct mtd_info *mtd)
5b0c5c2c0   Haavard Skinnemoen   MTD: Convert Atme...
213
214
215
216
217
218
219
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	struct cfi_pri_amdstd *extp = cfi->cmdset_priv;
  	struct cfi_pri_atmel atmel_pri;
  
  	memcpy(&atmel_pri, extp, sizeof(atmel_pri));
de591dacf   HÃ¥vard Skinnemoen   MTD: Fix bug in f...
220
  	memset((char *)extp + 5, 0, sizeof(*extp) - 5);
5b0c5c2c0   Haavard Skinnemoen   MTD: Convert Atme...
221
222
223
  
  	if (atmel_pri.Features & 0x02)
  		extp->EraseSuspend = 2;
be8f78b8e   Haavard Skinnemoen   [MTD] [NOR] AT49B...
224
225
226
227
228
229
230
231
232
233
234
235
  	/* Some chips got it backwards... */
  	if (cfi->id == AT49BV6416) {
  		if (atmel_pri.BottomBoot)
  			extp->TopBottom = 3;
  		else
  			extp->TopBottom = 2;
  	} else {
  		if (atmel_pri.BottomBoot)
  			extp->TopBottom = 2;
  		else
  			extp->TopBottom = 3;
  	}
d10a39d1a   Hans-Christian Egtvedt   [MTD] [NOR] More ...
236
237
238
239
  
  	/* burst write mode not supported */
  	cfi->cfiq->BufWriteTimeoutTyp = 0;
  	cfi->cfiq->BufWriteTimeoutMax = 0;
5b0c5c2c0   Haavard Skinnemoen   MTD: Convert Atme...
240
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
241
  static void fixup_use_secsi(struct mtd_info *mtd)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
242
243
244
245
246
  {
  	/* Setup for chips with a secsi area */
  	mtd->read_user_prot_reg = cfi_amdstd_secsi_read;
  	mtd->read_fact_prot_reg = cfi_amdstd_secsi_read;
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
247
  static void fixup_use_erase_chip(struct mtd_info *mtd)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248
249
250
251
252
253
254
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	if ((cfi->cfiq->NumEraseRegions == 1) &&
  		((cfi->cfiq->EraseRegionInfo[0] & 0xffff) == 0)) {
  		mtd->erase = cfi_amdstd_erase_chip;
  	}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
255

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
256
  }
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
257
258
259
260
  /*
   * Some Atmel chips (e.g. the AT49BV6416) power-up with all sectors
   * locked by default.
   */
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
261
  static void fixup_use_atmel_lock(struct mtd_info *mtd)
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
262
263
264
  {
  	mtd->lock = cfi_atmel_lock;
  	mtd->unlock = cfi_atmel_unlock;
e619a75ff   Justin Treon   [MTD] Unlocking a...
265
  	mtd->flags |= MTD_POWERUP_LOCK;
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
266
  }
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
267
268
269
270
271
272
  static void fixup_old_sst_eraseregion(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	/*
25985edce   Lucas De Marchi   Fix common misspe...
273
  	 * These flashes report two separate eraseblock regions based on the
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
274
275
276
277
278
279
  	 * sector_erase-size and block_erase-size, although they both operate on the
  	 * same memory. This is not allowed according to CFI, so we just pick the
  	 * sector_erase-size.
  	 */
  	cfi->cfiq->NumEraseRegions = 1;
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
280
  static void fixup_sst39vf(struct mtd_info *mtd)
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
281
282
283
284
285
286
287
288
289
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	fixup_old_sst_eraseregion(mtd);
  
  	cfi->addr_unlock1 = 0x5555;
  	cfi->addr_unlock2 = 0x2AAA;
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
290
  static void fixup_sst39vf_rev_b(struct mtd_info *mtd)
5a0563f0a   Guillaume LECERF   mtd: cfi_cmdset_0...
291
292
293
294
295
296
297
298
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	fixup_old_sst_eraseregion(mtd);
  
  	cfi->addr_unlock1 = 0x555;
  	cfi->addr_unlock2 = 0x2AA;
08968041b   Guillaume LECERF   mtd: cfi_cmdset_0...
299
300
  
  	cfi->sector_erase_cmd = CMD(0x50);
5a0563f0a   Guillaume LECERF   mtd: cfi_cmdset_0...
301
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
302
  static void fixup_sst38vf640x_sectorsize(struct mtd_info *mtd)
9fc05fcad   Guillaume LECERF   mtd: cfi_cmdset_0...
303
304
305
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
306
  	fixup_sst39vf_rev_b(mtd);
9fc05fcad   Guillaume LECERF   mtd: cfi_cmdset_0...
307
308
309
310
311
312
313
314
315
  
  	/*
  	 * CFI reports 1024 sectors (0x03ff+1) of 64KBytes (0x0100*256) where
  	 * it should report a size of 8KBytes (0x0020*256).
  	 */
  	cfi->cfiq->EraseRegionInfo[0] = 0x002003ff;
  	pr_warning("%s: Bad 38VF640x CFI data; adjusting sector size from 64 to 8KiB
  ", mtd->name);
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
316
  static void fixup_s29gl064n_sectors(struct mtd_info *mtd)
70b072550   Trent Piepho   [MTD] [NOR] Fixup...
317
318
319
320
321
322
323
324
325
326
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	if ((cfi->cfiq->EraseRegionInfo[0] & 0xffff) == 0x003f) {
  		cfi->cfiq->EraseRegionInfo[0] |= 0x0040;
  		pr_warning("%s: Bad S29GL064N CFI data, adjust from 64 to 128 sectors
  ", mtd->name);
  	}
  }
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
327
  static void fixup_s29gl032n_sectors(struct mtd_info *mtd)
70b072550   Trent Piepho   [MTD] [NOR] Fixup...
328
329
330
331
332
333
334
335
336
337
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	if ((cfi->cfiq->EraseRegionInfo[1] & 0xffff) == 0x007e) {
  		cfi->cfiq->EraseRegionInfo[1] &= ~0x0040;
  		pr_warning("%s: Bad S29GL032N CFI data, adjust from 127 to 63 sectors
  ", mtd->name);
  	}
  }
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
338
339
  /* Used to fix CFI-Tables of chips without Extended Query Tables */
  static struct cfi_fixup cfi_nopri_fixup_table[] = {
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
340
341
342
343
344
345
346
347
348
  	{ CFI_MFR_SST, 0x234a, fixup_sst39vf }, /* SST39VF1602 */
  	{ CFI_MFR_SST, 0x234b, fixup_sst39vf }, /* SST39VF1601 */
  	{ CFI_MFR_SST, 0x235a, fixup_sst39vf }, /* SST39VF3202 */
  	{ CFI_MFR_SST, 0x235b, fixup_sst39vf }, /* SST39VF3201 */
  	{ CFI_MFR_SST, 0x235c, fixup_sst39vf_rev_b }, /* SST39VF3202B */
  	{ CFI_MFR_SST, 0x235d, fixup_sst39vf_rev_b }, /* SST39VF3201B */
  	{ CFI_MFR_SST, 0x236c, fixup_sst39vf_rev_b }, /* SST39VF6402B */
  	{ CFI_MFR_SST, 0x236d, fixup_sst39vf_rev_b }, /* SST39VF6401B */
  	{ 0, 0, NULL }
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
349
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
350
  static struct cfi_fixup cfi_fixup_table[] = {
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
351
  	{ CFI_MFR_ATMEL, CFI_ID_ANY, fixup_convert_atmel_pri },
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
352
  #ifdef AMD_BOOTLOC_BUG
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
353
  	{ CFI_MFR_AMD, CFI_ID_ANY, fixup_amd_bootblock },
1065cda8a   Steffen Sledz   mtd: cfi: add sup...
354
  	{ CFI_MFR_AMIC, CFI_ID_ANY, fixup_amd_bootblock },
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
355
  	{ CFI_MFR_MACRONIX, CFI_ID_ANY, fixup_amd_bootblock },
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
356
  #endif
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
357
358
359
360
361
362
363
364
365
366
367
368
369
370
  	{ CFI_MFR_AMD, 0x0050, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x0053, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x0055, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x0056, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x005C, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x005F, fixup_use_secsi },
  	{ CFI_MFR_AMD, 0x0c01, fixup_s29gl064n_sectors },
  	{ CFI_MFR_AMD, 0x1301, fixup_s29gl064n_sectors },
  	{ CFI_MFR_AMD, 0x1a00, fixup_s29gl032n_sectors },
  	{ CFI_MFR_AMD, 0x1a01, fixup_s29gl032n_sectors },
  	{ CFI_MFR_SST, 0x536a, fixup_sst38vf640x_sectorsize }, /* SST38VF6402 */
  	{ CFI_MFR_SST, 0x536b, fixup_sst38vf640x_sectorsize }, /* SST38VF6401 */
  	{ CFI_MFR_SST, 0x536c, fixup_sst38vf640x_sectorsize }, /* SST38VF6404 */
  	{ CFI_MFR_SST, 0x536d, fixup_sst38vf640x_sectorsize }, /* SST38VF6403 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
371
  #if !FORCE_WORD_WRITE
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
372
  	{ CFI_MFR_ANY, CFI_ID_ANY, fixup_use_write_buffers },
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
373
  #endif
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
374
  	{ 0, 0, NULL }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
375
376
  };
  static struct cfi_fixup jedec_fixup_table[] = {
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
377
378
379
380
  	{ CFI_MFR_SST, SST49LF004B, fixup_use_fwh_lock },
  	{ CFI_MFR_SST, SST49LF040B, fixup_use_fwh_lock },
  	{ CFI_MFR_SST, SST49LF008A, fixup_use_fwh_lock },
  	{ 0, 0, NULL }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
381
382
383
384
385
386
387
388
  };
  
  static struct cfi_fixup fixup_table[] = {
  	/* The CFI vendor ids and the JEDEC vendor IDs appear
  	 * to be common.  It is like the devices id's are as
  	 * well.  This table is to pick all cases where
  	 * we know that is the case.
  	 */
cc3182225   Guillaume LECERF   mtd: cfi_fixup: r...
389
390
391
  	{ CFI_MFR_ANY, CFI_ID_ANY, fixup_use_erase_chip },
  	{ CFI_MFR_ATMEL, AT49BV6416, fixup_use_atmel_lock },
  	{ 0, 0, NULL }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
392
  };
fefae48bf   Wolfgang Grandegger   [MTD] CFI: remove...
393
394
395
  static void cfi_fixup_major_minor(struct cfi_private *cfi,
  				  struct cfi_pri_amdstd *extp)
  {
e63727637   Guillaume LECERF   mtd: cfi_cmdset_0...
396
  	if (cfi->mfr == CFI_MFR_SAMSUNG) {
e8953b739   Guillaume LECERF   mtd: cfi_cmdset_0...
397
398
  		if ((extp->MajorVersion == '0' && extp->MinorVersion == '0') ||
  		    (extp->MajorVersion == '3' && extp->MinorVersion == '3')) {
e63727637   Guillaume LECERF   mtd: cfi_cmdset_0...
399
400
401
  			/*
  			 * Samsung K8P2815UQB and K8D6x16UxM chips
  			 * report major=0 / minor=0.
e8953b739   Guillaume LECERF   mtd: cfi_cmdset_0...
402
  			 * K8D3x16UxC chips report major=3 / minor=3.
e63727637   Guillaume LECERF   mtd: cfi_cmdset_0...
403
404
405
406
407
408
409
410
  			 */
  			printk(KERN_NOTICE "  Fixing Samsung's Amd/Fujitsu"
  			       " Extended Query version to 1.%c
  ",
  			       extp->MinorVersion);
  			extp->MajorVersion = '1';
  		}
  	}
9fc05fcad   Guillaume LECERF   mtd: cfi_cmdset_0...
411
412
413
414
415
416
417
  	/*
  	 * SST 38VF640x chips report major=0xFF / minor=0xFF.
  	 */
  	if (cfi->mfr == CFI_MFR_SST && (cfi->id >> 4) == 0x0536) {
  		extp->MajorVersion = '1';
  		extp->MinorVersion = '0';
  	}
fefae48bf   Wolfgang Grandegger   [MTD] CFI: remove...
418
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
419
420
421
422
423
  struct mtd_info *cfi_cmdset_0002(struct map_info *map, int primary)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	struct mtd_info *mtd;
  	int i;
95b93a0cd   Burman Yan   [MTD] replace kma...
424
  	mtd = kzalloc(sizeof(*mtd), GFP_KERNEL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
425
426
427
428
429
  	if (!mtd) {
  		printk(KERN_WARNING "Failed to allocate memory for MTD device
  ");
  		return NULL;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
430
431
432
433
434
435
436
437
438
439
440
441
  	mtd->priv = map;
  	mtd->type = MTD_NORFLASH;
  
  	/* Fill in the default mtd operations */
  	mtd->erase   = cfi_amdstd_erase_varsize;
  	mtd->write   = cfi_amdstd_write_words;
  	mtd->read    = cfi_amdstd_read;
  	mtd->sync    = cfi_amdstd_sync;
  	mtd->suspend = cfi_amdstd_suspend;
  	mtd->resume  = cfi_amdstd_resume;
  	mtd->flags   = MTD_CAP_NORFLASH;
  	mtd->name    = map->name;
783ed81ff   Artem B. Bityutskiy   [MTD] assume mtd-...
442
  	mtd->writesize = 1;
13ce77f46   Anatolij Gustschin   mtd: cfi: fix wri...
443
  	mtd->writebufsize = cfi_interleave(cfi) << cfi->cfiq->MaxBufWriteSize;
d261c72ae   Anatolij Gustschin   mtd: cfi: add wri...
444

0a32a1026   Brian Norris   mtd: cleanup styl...
445
446
447
  	pr_debug("MTD %s(): write buffer size %d
  ", __func__,
  			mtd->writebufsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
448

eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
449
  	mtd->reboot_notifier.notifier_call = cfi_amdstd_reboot;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
450
451
  	if (cfi->cfi_mode==CFI_MODE_CFI){
  		unsigned char bootloc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
452
453
454
455
  		__u16 adr = primary?cfi->cfiq->P_ADR:cfi->cfiq->A_ADR;
  		struct cfi_pri_amdstd *extp;
  
  		extp = (struct cfi_pri_amdstd*)cfi_read_pri(map, adr, sizeof(*extp), "Amd/Fujitsu");
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
456
457
458
459
460
461
  		if (extp) {
  			/*
  			 * It's a real CFI chip, not one for which the probe
  			 * routine faked a CFI structure.
  			 */
  			cfi_fixup_major_minor(cfi, extp);
e17f47a14   Guillaume LECERF   mtd: cfi_cmdset_0...
462
  			/*
c9ddab25d   Gernot Hoyler   mtd: cmdset_0002:...
463
  			 * Valid primary extension versions are: 1.0, 1.1, 1.2, 1.3, 1.4, 1.5
631dd1a88   Justin P. Mattock   Update broken web...
464
465
  			 * see: http://cs.ozerki.net/zap/pub/axim-x5/docs/cfi_r20.pdf, page 19 
  			 *      http://www.spansion.com/Support/AppNotes/cfi_100_20011201.pdf
5da195326   Guillaume LECERF   mtd: cfi_cmdset_0...
466
  			 *      http://www.spansion.com/Support/Datasheets/s29ws-p_00_a12_e.pdf
c9ddab25d   Gernot Hoyler   mtd: cmdset_0002:...
467
  			 *      http://www.spansion.com/Support/Datasheets/S29GL_128S_01GS_00_02_e.pdf
e17f47a14   Guillaume LECERF   mtd: cfi_cmdset_0...
468
  			 */
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
469
  			if (extp->MajorVersion != '1' ||
c9ddab25d   Gernot Hoyler   mtd: cmdset_0002:...
470
  			    (extp->MajorVersion == '1' && (extp->MinorVersion < '0' || extp->MinorVersion > '5'))) {
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
471
  				printk(KERN_ERR "  Unknown Amd/Fujitsu Extended Query "
e17f47a14   Guillaume LECERF   mtd: cfi_cmdset_0...
472
473
474
475
  				       "version %c.%c (%#02x/%#02x).
  ",
  				       extp->MajorVersion, extp->MinorVersion,
  				       extp->MajorVersion, extp->MinorVersion);
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
476
477
478
479
  				kfree(extp);
  				kfree(mtd);
  				return NULL;
  			}
d88f977b8   Todd Poynor   [MTD] CHIPS: Reco...
480

e17f47a14   Guillaume LECERF   mtd: cfi_cmdset_0...
481
482
483
  			printk(KERN_INFO "  Amd/Fujitsu Extended Query version %c.%c.
  ",
  			       extp->MajorVersion, extp->MinorVersion);
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
484
485
  			/* Install our own private info structure */
  			cfi->cmdset_priv = extp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
486

564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
487
488
  			/* Apply cfi device specific fixups */
  			cfi_fixup(mtd, cfi_fixup_table);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
489
490
  
  #ifdef DEBUG_CFI_FEATURES
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
491
492
  			/* Tell the user about it in lots of lovely detail */
  			cfi_tell_features(extp);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
493
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
494

564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
495
  			bootloc = extp->TopBottom;
412da2f6e   David Woodhouse   mtd: cfi_cmdset_0...
496
497
498
499
  			if ((bootloc < 2) || (bootloc > 5)) {
  				printk(KERN_WARNING "%s: CFI contains unrecognised boot "
  				       "bank location (%d). Assuming bottom.
  ",
abab7ebf8   David Woodhouse   mtd: cfi_cmdset_0...
500
  				       map->name, bootloc);
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
501
502
  				bootloc = 2;
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
503

564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
504
  			if (bootloc == 3 && cfi->cfiq->NumEraseRegions > 1) {
412da2f6e   David Woodhouse   mtd: cfi_cmdset_0...
505
506
  				printk(KERN_WARNING "%s: Swapping erase regions for top-boot CFI table.
  ", map->name);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
507

564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
508
509
510
  				for (i=0; i<cfi->cfiq->NumEraseRegions / 2; i++) {
  					int j = (cfi->cfiq->NumEraseRegions-1)-i;
  					__u32 swap;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
511

564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
512
513
514
515
  					swap = cfi->cfiq->EraseRegionInfo[i];
  					cfi->cfiq->EraseRegionInfo[i] = cfi->cfiq->EraseRegionInfo[j];
  					cfi->cfiq->EraseRegionInfo[j] = swap;
  				}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
516
  			}
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
517
518
519
520
  			/* Set the default CFI lock/unlock addresses */
  			cfi->addr_unlock1 = 0x555;
  			cfi->addr_unlock2 = 0x2aa;
  		}
83dcd3bb1   Guillaume LECERF   mtd: cfi_cmdset_0...
521
  		cfi_fixup(mtd, cfi_nopri_fixup_table);
564b84978   Guillaume LECERF   mtd: cfi_cmdset_0...
522
523
524
525
  
  		if (!cfi->addr_unlock1 || !cfi->addr_unlock2) {
  			kfree(mtd);
  			return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
526
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
528
529
530
531
532
533
534
535
536
537
538
539
  
  	} /* CFI mode */
  	else if (cfi->cfi_mode == CFI_MODE_JEDEC) {
  		/* Apply jedec specific fixups */
  		cfi_fixup(mtd, jedec_fixup_table);
  	}
  	/* Apply generic fixups */
  	cfi_fixup(mtd, fixup_table);
  
  	for (i=0; i< cfi->numchips; i++) {
  		cfi->chips[i].word_write_time = 1<<cfi->cfiq->WordWriteTimeoutTyp;
  		cfi->chips[i].buffer_write_time = 1<<cfi->cfiq->BufWriteTimeoutTyp;
  		cfi->chips[i].erase_time = 1<<cfi->cfiq->BlockEraseTimeoutTyp;
83d480917   Vijay Sampath   [MTD] [NOR] Fix o...
540
541
  		cfi->chips[i].ref_point_counter = 0;
  		init_waitqueue_head(&(cfi->chips[i].wq));
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
542
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
543
  	map->fldrv = &cfi_amdstd_chipdrv;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
544

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
545
546
  	return cfi_amdstd_setup(mtd);
  }
804611281   Guillaume LECERF   mtd: cfi_cmdset_0...
547
  struct mtd_info *cfi_cmdset_0006(struct map_info *map, int primary) __attribute__((alias("cfi_cmdset_0002")));
1e804cec7   David Woodhouse   mtd: cfi_cmdset_0...
548
  struct mtd_info *cfi_cmdset_0701(struct map_info *map, int primary) __attribute__((alias("cfi_cmdset_0002")));
83ea4ef21   David Woodhouse   Export cfi_cmdset...
549
  EXPORT_SYMBOL_GPL(cfi_cmdset_0002);
804611281   Guillaume LECERF   mtd: cfi_cmdset_0...
550
  EXPORT_SYMBOL_GPL(cfi_cmdset_0006);
1e804cec7   David Woodhouse   mtd: cfi_cmdset_0...
551
  EXPORT_SYMBOL_GPL(cfi_cmdset_0701);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
552
553
554
555
556
557
558
559
  
  static struct mtd_info *cfi_amdstd_setup(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long devsize = (1<<cfi->cfiq->DevSize) * cfi->interleave;
  	unsigned long offset = 0;
  	int i,j;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
560
561
  	printk(KERN_NOTICE "number of %s chips: %d
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
562
  	       (cfi->cfi_mode == CFI_MODE_CFI)?"CFI":"JEDEC",cfi->numchips);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
563
  	/* Select the correct geometry setup */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
564
565
566
567
568
  	mtd->size = devsize * cfi->numchips;
  
  	mtd->numeraseregions = cfi->cfiq->NumEraseRegions * cfi->numchips;
  	mtd->eraseregions = kmalloc(sizeof(struct mtd_erase_region_info)
  				    * mtd->numeraseregions, GFP_KERNEL);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
569
  	if (!mtd->eraseregions) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
570
571
572
573
  		printk(KERN_WARNING "Failed to allocate memory for MTD erase region info
  ");
  		goto setup_err;
  	}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
574

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
575
576
577
578
  	for (i=0; i<cfi->cfiq->NumEraseRegions; i++) {
  		unsigned long ernum, ersize;
  		ersize = ((cfi->cfiq->EraseRegionInfo[i] >> 8) & ~0xff) * cfi->interleave;
  		ernum = (cfi->cfiq->EraseRegionInfo[i] & 0xffff) + 1;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
579

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
  		if (mtd->erasesize < ersize) {
  			mtd->erasesize = ersize;
  		}
  		for (j=0; j<cfi->numchips; j++) {
  			mtd->eraseregions[(j*cfi->cfiq->NumEraseRegions)+i].offset = (j*devsize)+offset;
  			mtd->eraseregions[(j*cfi->cfiq->NumEraseRegions)+i].erasesize = ersize;
  			mtd->eraseregions[(j*cfi->cfiq->NumEraseRegions)+i].numblocks = ernum;
  		}
  		offset += (ersize * ernum);
  	}
  	if (offset != devsize) {
  		/* Argh */
  		printk(KERN_WARNING "Sum of regions (%lx) != total size of set of interleaved chips (%lx)
  ", offset, devsize);
  		goto setup_err;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
596

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
597
  	__module_get(THIS_MODULE);
eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
598
  	register_reboot_notifier(&mtd->reboot_notifier);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
599
600
601
  	return mtd;
  
   setup_err:
17fabf156   Jiri Slaby   mtd: cfi: remove ...
602
603
  	kfree(mtd->eraseregions);
  	kfree(mtd);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
604
605
606
607
608
609
610
611
612
613
614
615
616
  	kfree(cfi->cmdset_priv);
  	kfree(cfi->cfiq);
  	return NULL;
  }
  
  /*
   * Return true if the chip is ready.
   *
   * Ready is one of: read mode, query mode, erase-suspend-read mode (in any
   * non-suspended sector) and is indicated by no toggle bits toggling.
   *
   * Note that anything more complicated than checking if no bits are toggling
   * (including checking DQ5 for an error status) is tricky to get working
25985edce   Lucas De Marchi   Fix common misspe...
617
618
   * correctly and is therefore not done	(particularly with interleaved chips
   * as each chip must be checked independently of the others).
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
619
   */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
620
  static int __xipram chip_ready(struct map_info *map, unsigned long addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
621
622
623
624
625
626
627
628
  {
  	map_word d, t;
  
  	d = map_read(map, addr);
  	t = map_read(map, addr);
  
  	return map_word_equal(map, d, t);
  }
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
629
630
631
632
633
634
635
636
637
638
639
  /*
   * Return true if the chip is ready and has the correct value.
   *
   * Ready is one of: read mode, query mode, erase-suspend-read mode (in any
   * non-suspended sector) and it is indicated by no bits toggling.
   *
   * Error are indicated by toggling bits or bits held with the wrong value,
   * or with bits toggling.
   *
   * Note that anything more complicated than checking if no bits are toggling
   * (including checking DQ5 for an error status) is tricky to get working
25985edce   Lucas De Marchi   Fix common misspe...
640
641
   * correctly and is therefore not done	(particularly with interleaved chips
   * as each chip must be checked independently of the others).
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
642
643
   *
   */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
644
  static int __xipram chip_good(struct map_info *map, unsigned long addr, map_word expected)
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
645
646
647
648
649
  {
  	map_word oldd, curd;
  
  	oldd = map_read(map, addr);
  	curd = map_read(map, addr);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
650
  	return	map_word_equal(map, oldd, curd) &&
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
651
652
  		map_word_equal(map, curd, expected);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
  static int get_chip(struct map_info *map, struct flchip *chip, unsigned long adr, int mode)
  {
  	DECLARE_WAITQUEUE(wait, current);
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long timeo;
  	struct cfi_pri_amdstd *cfip = (struct cfi_pri_amdstd *)cfi->cmdset_priv;
  
   resettime:
  	timeo = jiffies + HZ;
   retry:
  	switch (chip->state) {
  
  	case FL_STATUS:
  		for (;;) {
  			if (chip_ready(map, adr))
  				break;
  
  			if (time_after(jiffies, timeo)) {
  				printk(KERN_ERR "Waiting for chip to be ready timed out.
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
673
674
  				return -EIO;
  			}
c4e773764   Stefani Seibold   mtd: fix a huge l...
675
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
676
  			cfi_udelay(1);
c4e773764   Stefani Seibold   mtd: fix a huge l...
677
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
678
679
680
  			/* Someone else might have been playing with it. */
  			goto retry;
  		}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
681

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
682
683
684
685
686
687
  	case FL_READY:
  	case FL_CFI_QUERY:
  	case FL_JEDEC_QUERY:
  		return 0;
  
  	case FL_ERASING:
2695eab96   Joakim Tjernlund   mtd: CFI cmdset_0...
688
689
690
  		if (!cfip || !(cfip->EraseSuspend & (0x1|0x2)) ||
  		    !(mode == FL_READY || mode == FL_POINT ||
  		    (mode == FL_WRITING && (cfip->EraseSuspend & 0x2))))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
  			goto sleep;
  
  		/* We could check to see if we're trying to access the sector
  		 * that is currently being erased. However, no user will try
  		 * anything like that so we just wait for the timeout. */
  
  		/* Erase suspend */
  		/* It's harmless to issue the Erase-Suspend and Erase-Resume
  		 * commands when the erase algorithm isn't in progress. */
  		map_write(map, CMD(0xB0), chip->in_progress_block_addr);
  		chip->oldstate = FL_ERASING;
  		chip->state = FL_ERASE_SUSPENDING;
  		chip->erase_suspended = 1;
  		for (;;) {
  			if (chip_ready(map, adr))
  				break;
  
  			if (time_after(jiffies, timeo)) {
  				/* Should have suspended the erase by now.
  				 * Send an Erase-Resume command as either
  				 * there was an error (so leave the erase
  				 * routine to recover from it) or we trying to
  				 * use the erase-in-progress sector. */
100f2341e   Tadashi Abe   mtd: fix hang-up ...
714
  				put_chip(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
715
716
717
718
  				printk(KERN_ERR "MTD %s(): chip not ready after erase suspend
  ", __func__);
  				return -EIO;
  			}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
719

c4e773764   Stefani Seibold   mtd: fix a huge l...
720
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
721
  			cfi_udelay(1);
c4e773764   Stefani Seibold   mtd: fix a huge l...
722
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
723
724
725
726
727
  			/* Nobody will touch it while it's in state FL_ERASE_SUSPENDING.
  			   So we can just loop here. */
  		}
  		chip->state = FL_READY;
  		return 0;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
728
729
730
731
732
733
734
  	case FL_XIP_WHILE_ERASING:
  		if (mode != FL_READY && mode != FL_POINT &&
  		    (!cfip || !(cfip->EraseSuspend&2)))
  			goto sleep;
  		chip->oldstate = chip->state;
  		chip->state = FL_READY;
  		return 0;
eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
735
736
737
  	case FL_SHUTDOWN:
  		/* The machine is rebooting */
  		return -EIO;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
738
739
740
741
742
743
744
745
746
  	case FL_POINT:
  		/* Only if there's no operation suspended... */
  		if (mode == FL_READY && chip->oldstate == FL_READY)
  			return 0;
  
  	default:
  	sleep:
  		set_current_state(TASK_UNINTERRUPTIBLE);
  		add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
747
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
748
749
  		schedule();
  		remove_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
750
  		mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
751
752
753
754
755
756
757
758
759
760
761
  		goto resettime;
  	}
  }
  
  
  static void put_chip(struct map_info *map, struct flchip *chip, unsigned long adr)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	switch(chip->oldstate) {
  	case FL_ERASING:
08968041b   Guillaume LECERF   mtd: cfi_cmdset_0...
762
  		map_write(map, cfi->sector_erase_cmd, chip->in_progress_block_addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
763
764
765
  		chip->oldstate = FL_READY;
  		chip->state = FL_ERASING;
  		break;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
766
767
768
769
  	case FL_XIP_WHILE_ERASING:
  		chip->state = chip->oldstate;
  		chip->oldstate = FL_READY;
  		break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
770
771
772
773
774
775
776
777
778
779
780
  	case FL_READY:
  	case FL_STATUS:
  		/* We should really make set_vpp() count, rather than doing this */
  		DISABLE_VPP(map);
  		break;
  	default:
  		printk(KERN_ERR "MTD: put_chip() called with oldstate %d!!
  ", chip->oldstate);
  	}
  	wake_up(&chip->wq);
  }
02b15e343   Todd Poynor   [MTD] XIP for AMD...
781
782
783
784
785
786
787
788
789
790
791
792
  #ifdef CONFIG_MTD_XIP
  
  /*
   * No interrupt what so ever can be serviced while the flash isn't in array
   * mode.  This is ensured by the xip_disable() and xip_enable() functions
   * enclosing any code path where the flash is known not to be in array mode.
   * And within a XIP disabled code path, only functions marked with __xipram
   * may be called and nothing else (it's a good thing to inspect generated
   * assembly to make sure inline functions were actually inlined and that gcc
   * didn't emit calls to its own support functions). Also configuring MTD CFI
   * support to a single buswidth and a single interleave is also recommended.
   */
f8eb321be   Thomas Gleixner   [MTD] cfi_cmdset_...
793

02b15e343   Todd Poynor   [MTD] XIP for AMD...
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
  static void xip_disable(struct map_info *map, struct flchip *chip,
  			unsigned long adr)
  {
  	/* TODO: chips with no XIP use should ignore and return */
  	(void) map_read(map, adr); /* ensure mmu mapping is up to date */
  	local_irq_disable();
  }
  
  static void __xipram xip_enable(struct map_info *map, struct flchip *chip,
  				unsigned long adr)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  
  	if (chip->state != FL_POINT && chip->state != FL_READY) {
  		map_write(map, CMD(0xf0), adr);
  		chip->state = FL_READY;
  	}
  	(void) map_read(map, adr);
97f927a4d   Thomas Gleixner   [MTD] XIP cleanup
812
  	xip_iprefetch();
02b15e343   Todd Poynor   [MTD] XIP for AMD...
813
814
815
816
817
818
819
  	local_irq_enable();
  }
  
  /*
   * When a delay is required for the flash operation to complete, the
   * xip_udelay() function is polling for both the given timeout and pending
   * (but still masked) hardware interrupts.  Whenever there is an interrupt
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
820
   * pending then the flash erase operation is suspended, array mode restored
02b15e343   Todd Poynor   [MTD] XIP for AMD...
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
   * and interrupts unmasked.  Task scheduling might also happen at that
   * point.  The CPU eventually returns from the interrupt or the call to
   * schedule() and the suspended flash operation is resumed for the remaining
   * of the delay period.
   *
   * Warning: this function _will_ fool interrupt latency tracing tools.
   */
  
  static void __xipram xip_udelay(struct map_info *map, struct flchip *chip,
  				unsigned long adr, int usec)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	struct cfi_pri_amdstd *extp = cfi->cmdset_priv;
  	map_word status, OK = CMD(0x80);
  	unsigned long suspended, start = xip_currtime();
  	flstate_t oldstate;
  
  	do {
  		cpu_relax();
  		if (xip_irqpending() && extp &&
  		    ((chip->state == FL_ERASING && (extp->EraseSuspend & 2))) &&
  		    (cfi_interleave_is_1(cfi) || chip->oldstate == FL_READY)) {
  			/*
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
844
845
846
  			 * Let's suspend the erase operation when supported.
  			 * Note that we currently don't try to suspend
  			 * interleaved chips if there is already another
02b15e343   Todd Poynor   [MTD] XIP for AMD...
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
  			 * operation suspended (imagine what happens
  			 * when one chip was already done with the current
  			 * operation while another chip suspended it, then
  			 * we resume the whole thing at once).  Yes, it
  			 * can happen!
  			 */
  			map_write(map, CMD(0xb0), adr);
  			usec -= xip_elapsed_since(start);
  			suspended = xip_currtime();
  			do {
  				if (xip_elapsed_since(suspended) > 100000) {
  					/*
  					 * The chip doesn't want to suspend
  					 * after waiting for 100 msecs.
  					 * This is a critical error but there
  					 * is not much we can do here.
  					 */
  					return;
  				}
  				status = map_read(map, adr);
  			} while (!map_word_andequal(map, status, OK, OK));
  
  			/* Suspend succeeded */
  			oldstate = chip->state;
  			if (!map_word_bitsset(map, status, CMD(0x40)))
  				break;
  			chip->state = FL_XIP_WHILE_ERASING;
  			chip->erase_suspended = 1;
  			map_write(map, CMD(0xf0), adr);
  			(void) map_read(map, adr);
ca5c23c3b   Paulius Zaleckas   [MTD] XIP: Use ge...
877
  			xip_iprefetch();
02b15e343   Todd Poynor   [MTD] XIP for AMD...
878
  			local_irq_enable();
c4e773764   Stefani Seibold   mtd: fix a huge l...
879
  			mutex_unlock(&chip->mutex);
ca5c23c3b   Paulius Zaleckas   [MTD] XIP: Use ge...
880
  			xip_iprefetch();
02b15e343   Todd Poynor   [MTD] XIP for AMD...
881
882
883
884
885
886
887
888
  			cond_resched();
  
  			/*
  			 * We're back.  However someone else might have
  			 * decided to go write to the chip if we are in
  			 * a suspended erase state.  If so let's wait
  			 * until it's done.
  			 */
c4e773764   Stefani Seibold   mtd: fix a huge l...
889
  			mutex_lock(&chip->mutex);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
890
891
892
893
  			while (chip->state != FL_XIP_WHILE_ERASING) {
  				DECLARE_WAITQUEUE(wait, current);
  				set_current_state(TASK_UNINTERRUPTIBLE);
  				add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
894
  				mutex_unlock(&chip->mutex);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
895
896
  				schedule();
  				remove_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
897
  				mutex_lock(&chip->mutex);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
898
899
900
901
902
  			}
  			/* Disallow XIP again */
  			local_irq_disable();
  
  			/* Resume the write or erase operation */
08968041b   Guillaume LECERF   mtd: cfi_cmdset_0...
903
  			map_write(map, cfi->sector_erase_cmd, adr);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
  			chip->state = oldstate;
  			start = xip_currtime();
  		} else if (usec >= 1000000/HZ) {
  			/*
  			 * Try to save on CPU power when waiting delay
  			 * is at least a system timer tick period.
  			 * No need to be extremely accurate here.
  			 */
  			xip_cpu_idle();
  		}
  		status = map_read(map, adr);
  	} while (!map_word_andequal(map, status, OK, OK)
  		 && xip_elapsed_since(start) < usec);
  }
  
  #define UDELAY(map, chip, adr, usec)  xip_udelay(map, chip, adr, usec)
  
  /*
   * The INVALIDATE_CACHED_RANGE() macro is normally used in parallel while
   * the flash is actively programming or erasing since we have to poll for
   * the operation to complete anyway.  We can't do that in a generic way with
   * a XIP setup so do it before the actual flash operation in this case
   * and stub it out from INVALIDATE_CACHE_UDELAY.
   */
  #define XIP_INVAL_CACHED_RANGE(map, from, size)  \
  	INVALIDATE_CACHED_RANGE(map, from, size)
  
  #define INVALIDATE_CACHE_UDELAY(map, chip, adr, len, usec)  \
  	UDELAY(map, chip, adr, usec)
  
  /*
   * Extra notes:
   *
   * Activating this XIP support changes the way the code works a bit.  For
   * example the code to suspend the current process when concurrent access
   * happens is never executed because xip_udelay() will always return with the
   * same chip state as it was entered with.  This is why there is no care for
   * the presence of add_wait_queue() or schedule() calls from within a couple
   * xip_disable()'d  areas of code, like in do_erase_oneblock for example.
   * The queueing and scheduling are always happening within xip_udelay().
   *
   * Similarly, get_chip() and put_chip() just happen to always be executed
   * with chip->state set to FL_READY (or FL_XIP_WHILE_*) where flash state
   * is in array mode, therefore never executing many cases therein and not
   * causing any problem with XIP.
   */
  
  #else
  
  #define xip_disable(map, chip, adr)
  #define xip_enable(map, chip, adr)
  #define XIP_INVAL_CACHED_RANGE(x...)
  
  #define UDELAY(map, chip, adr, usec)  \
  do {  \
c4e773764   Stefani Seibold   mtd: fix a huge l...
959
  	mutex_unlock(&chip->mutex);  \
02b15e343   Todd Poynor   [MTD] XIP for AMD...
960
  	cfi_udelay(usec);  \
c4e773764   Stefani Seibold   mtd: fix a huge l...
961
  	mutex_lock(&chip->mutex);  \
02b15e343   Todd Poynor   [MTD] XIP for AMD...
962
963
964
965
  } while (0)
  
  #define INVALIDATE_CACHE_UDELAY(map, chip, adr, len, usec)  \
  do {  \
c4e773764   Stefani Seibold   mtd: fix a huge l...
966
  	mutex_unlock(&chip->mutex);  \
02b15e343   Todd Poynor   [MTD] XIP for AMD...
967
968
  	INVALIDATE_CACHED_RANGE(map, adr, len);  \
  	cfi_udelay(usec);  \
c4e773764   Stefani Seibold   mtd: fix a huge l...
969
  	mutex_lock(&chip->mutex);  \
02b15e343   Todd Poynor   [MTD] XIP for AMD...
970
971
972
  } while (0)
  
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
973
974
975
976
977
978
979
980
  
  static inline int do_read_onechip(struct map_info *map, struct flchip *chip, loff_t adr, size_t len, u_char *buf)
  {
  	unsigned long cmd_addr;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int ret;
  
  	adr += chip->start;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
981
982
  	/* Ensure cmd read/writes are aligned. */
  	cmd_addr = adr & ~(map_bankwidth(map)-1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
983

c4e773764   Stefani Seibold   mtd: fix a huge l...
984
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
985
986
  	ret = get_chip(map, chip, cmd_addr, FL_READY);
  	if (ret) {
c4e773764   Stefani Seibold   mtd: fix a huge l...
987
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
988
989
990
991
992
993
994
995
996
997
998
  		return ret;
  	}
  
  	if (chip->state != FL_POINT && chip->state != FL_READY) {
  		map_write(map, CMD(0xf0), cmd_addr);
  		chip->state = FL_READY;
  	}
  
  	map_copy_from(map, buf, adr, len);
  
  	put_chip(map, chip, cmd_addr);
c4e773764   Stefani Seibold   mtd: fix a huge l...
999
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
  	return 0;
  }
  
  
  static int cfi_amdstd_read (struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, u_char *buf)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long ofs;
  	int chipnum;
  	int ret = 0;
  
  	/* ofs: offset within the first chip that the first read should start */
  
  	chipnum = (from >> cfi->chipshift);
  	ofs = from - (chipnum <<  cfi->chipshift);
  
  
  	*retlen = 0;
  
  	while (len) {
  		unsigned long thislen;
  
  		if (chipnum >= cfi->numchips)
  			break;
  
  		if ((len + ofs -1) >> cfi->chipshift)
  			thislen = (1<<cfi->chipshift) - ofs;
  		else
  			thislen = len;
  
  		ret = do_read_onechip(map, &cfi->chips[chipnum], ofs, thislen, buf);
  		if (ret)
  			break;
  
  		*retlen += thislen;
  		len -= thislen;
  		buf += thislen;
  
  		ofs = 0;
  		chipnum++;
  	}
  	return ret;
  }
  
  
  static inline int do_read_secsi_onechip(struct map_info *map, struct flchip *chip, loff_t adr, size_t len, u_char *buf)
  {
  	DECLARE_WAITQUEUE(wait, current);
  	unsigned long timeo = jiffies + HZ;
  	struct cfi_private *cfi = map->fldrv_priv;
  
   retry:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1053
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1054
1055
  
  	if (chip->state != FL_READY){
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1056
1057
  		set_current_state(TASK_UNINTERRUPTIBLE);
  		add_wait_queue(&chip->wq, &wait);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1058

c4e773764   Stefani Seibold   mtd: fix a huge l...
1059
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1060
1061
1062
  
  		schedule();
  		remove_wait_queue(&chip->wq, &wait);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1063
1064
1065
  		timeo = jiffies + HZ;
  
  		goto retry;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1066
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1067
1068
1069
1070
1071
1072
1073
1074
  
  	adr += chip->start;
  
  	chip->state = FL_READY;
  
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x88, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1075

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1076
1077
1078
1079
1080
1081
  	map_copy_from(map, buf, adr, len);
  
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x90, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x00, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1082

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1083
  	wake_up(&chip->wq);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1084
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
  
  	return 0;
  }
  
  static int cfi_amdstd_secsi_read (struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, u_char *buf)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long ofs;
  	int chipnum;
  	int ret = 0;
  
  
  	/* ofs: offset within the first chip that the first read should start */
  
  	/* 8 secsi bytes per chip */
  	chipnum=from>>3;
  	ofs=from & 7;
  
  
  	*retlen = 0;
  
  	while (len) {
  		unsigned long thislen;
  
  		if (chipnum >= cfi->numchips)
  			break;
  
  		if ((len + ofs -1) >> 3)
  			thislen = (1<<3) - ofs;
  		else
  			thislen = len;
  
  		ret = do_read_secsi_onechip(map, &cfi->chips[chipnum], ofs, thislen, buf);
  		if (ret)
  			break;
  
  		*retlen += thislen;
  		len -= thislen;
  		buf += thislen;
  
  		ofs = 0;
  		chipnum++;
  	}
  	return ret;
  }
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1131
  static int __xipram do_write_oneword(struct map_info *map, struct flchip *chip, unsigned long adr, map_word datum)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long timeo = jiffies + HZ;
  	/*
  	 * We use a 1ms + 1 jiffies generic timeout for writes (most devices
  	 * have a max write time of a few hundreds usec). However, we should
  	 * use the maximum timeout value given by the chip at probe time
  	 * instead.  Unfortunately, struct flchip does have a field for
  	 * maximum timeout, only for typical which can be far too short
  	 * depending of the conditions.	 The ' + 1' is to avoid having a
  	 * timeout of 0 jiffies if HZ is smaller than 1000.
  	 */
  	unsigned long uWriteTimeout = ( HZ / 1000 ) + 1;
  	int ret = 0;
  	map_word oldd;
  	int retry_cnt = 0;
  
  	adr += chip->start;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1150
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1151
1152
  	ret = get_chip(map, chip, adr, FL_WRITING);
  	if (ret) {
c4e773764   Stefani Seibold   mtd: fix a huge l...
1153
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1154
1155
  		return ret;
  	}
289c05222   Brian Norris   mtd: replace DEBU...
1156
1157
  	pr_debug("MTD %s(): WRITE 0x%.8lx(0x%.8lx)
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
  	       __func__, adr, datum.x[0] );
  
  	/*
  	 * Check for a NOP for the case when the datum to write is already
  	 * present - it saves time and works around buggy chips that corrupt
  	 * data at other locations when 0xff is written to a location that
  	 * already contains 0xff.
  	 */
  	oldd = map_read(map, adr);
  	if (map_word_equal(map, oldd, datum)) {
289c05222   Brian Norris   mtd: replace DEBU...
1168
1169
  		pr_debug("MTD %s(): NOP
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1170
1171
1172
  		       __func__);
  		goto op_done;
  	}
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1173
  	XIP_INVAL_CACHED_RANGE(map, adr, map_bankwidth(map));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1174
  	ENABLE_VPP(map);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1175
  	xip_disable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1176
1177
1178
1179
1180
1181
   retry:
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0xA0, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	map_write(map, datum, adr);
  	chip->state = FL_WRITING;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1182
1183
1184
  	INVALIDATE_CACHE_UDELAY(map, chip,
  				adr, map_bankwidth(map),
  				chip->word_write_time);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1185
1186
  
  	/* See comment above for timeout value. */
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1187
  	timeo = jiffies + uWriteTimeout;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1188
1189
1190
1191
1192
1193
1194
  	for (;;) {
  		if (chip->state != FL_WRITING) {
  			/* Someone's suspended the write. Sleep */
  			DECLARE_WAITQUEUE(wait, current);
  
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1195
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1196
1197
1198
  			schedule();
  			remove_wait_queue(&chip->wq, &wait);
  			timeo = jiffies + (HZ / 2); /* FIXME */
c4e773764   Stefani Seibold   mtd: fix a huge l...
1199
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1200
1201
  			continue;
  		}
b95f9609c   Konstantin Baidarov   [MTD] chips cfi_c...
1202
  		if (time_after(jiffies, timeo) && !chip_ready(map, adr)){
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1203
  			xip_enable(map, chip, adr);
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1204
1205
  			printk(KERN_WARNING "MTD %s(): software timeout
  ", __func__);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1206
  			xip_disable(map, chip, adr);
b95f9609c   Konstantin Baidarov   [MTD] chips cfi_c...
1207
  			break;
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1208
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1209

b95f9609c   Konstantin Baidarov   [MTD] chips cfi_c...
1210
1211
  		if (chip_ready(map, adr))
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1212
  		/* Latency issues. Drop the lock, wait a while and retry */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1213
  		UDELAY(map, chip, adr, 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1214
  	}
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1215
1216
1217
1218
1219
  	/* Did we succeed? */
  	if (!chip_good(map, adr, datum)) {
  		/* reset on all failures. */
  		map_write( map, CMD(0xF0), chip->start );
  		/* FIXME - should have reset delay before continuing */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1220

1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1221
  		if (++retry_cnt <= MAX_WORD_RETRIES)
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1222
  			goto retry;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1223

fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1224
1225
  		ret = -EIO;
  	}
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1226
  	xip_enable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1227
1228
1229
   op_done:
  	chip->state = FL_READY;
  	put_chip(map, chip, adr);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1230
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
  
  	return ret;
  }
  
  
  static int cfi_amdstd_write_words(struct mtd_info *mtd, loff_t to, size_t len,
  				  size_t *retlen, const u_char *buf)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int ret = 0;
  	int chipnum;
  	unsigned long ofs, chipstart;
  	DECLARE_WAITQUEUE(wait, current);
  
  	*retlen = 0;
  	if (!len)
  		return 0;
  
  	chipnum = to >> cfi->chipshift;
  	ofs = to  - (chipnum << cfi->chipshift);
  	chipstart = cfi->chips[chipnum].start;
  
  	/* If it's not bus-aligned, do the first byte write */
  	if (ofs & (map_bankwidth(map)-1)) {
  		unsigned long bus_ofs = ofs & ~(map_bankwidth(map)-1);
  		int i = ofs - bus_ofs;
  		int n = 0;
  		map_word tmp_buf;
  
   retry:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1262
  		mutex_lock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1263
1264
  
  		if (cfi->chips[chipnum].state != FL_READY) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1265
1266
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&cfi->chips[chipnum].wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1267
  			mutex_unlock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1268
1269
1270
  
  			schedule();
  			remove_wait_queue(&cfi->chips[chipnum].wq, &wait);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1271
1272
1273
1274
1275
  			goto retry;
  		}
  
  		/* Load 'tmp_buf' with old contents of flash */
  		tmp_buf = map_read(map, bus_ofs+chipstart);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1276
  		mutex_unlock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1277
1278
1279
  
  		/* Number of bytes to copy from buffer */
  		n = min_t(int, len, map_bankwidth(map)-i);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1280

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1281
  		tmp_buf = map_word_load_partial(map, tmp_buf, buf, i, n);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1282
  		ret = do_write_oneword(map, &cfi->chips[chipnum],
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1283
  				       bus_ofs, tmp_buf);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1284
  		if (ret)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1285
  			return ret;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1286

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1287
1288
1289
1290
1291
1292
  		ofs += n;
  		buf += n;
  		(*retlen) += n;
  		len -= n;
  
  		if (ofs >> cfi->chipshift) {
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1293
  			chipnum ++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1294
1295
1296
1297
1298
  			ofs = 0;
  			if (chipnum == cfi->numchips)
  				return 0;
  		}
  	}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1299

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
  	/* We are now aligned, write as much as possible */
  	while(len >= map_bankwidth(map)) {
  		map_word datum;
  
  		datum = map_word_load(map, buf);
  
  		ret = do_write_oneword(map, &cfi->chips[chipnum],
  				       ofs, datum);
  		if (ret)
  			return ret;
  
  		ofs += map_bankwidth(map);
  		buf += map_bankwidth(map);
  		(*retlen) += map_bankwidth(map);
  		len -= map_bankwidth(map);
  
  		if (ofs >> cfi->chipshift) {
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1317
  			chipnum ++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
  			ofs = 0;
  			if (chipnum == cfi->numchips)
  				return 0;
  			chipstart = cfi->chips[chipnum].start;
  		}
  	}
  
  	/* Write the trailing bytes if any */
  	if (len & (map_bankwidth(map)-1)) {
  		map_word tmp_buf;
  
   retry1:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1330
  		mutex_lock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1331
1332
  
  		if (cfi->chips[chipnum].state != FL_READY) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1333
1334
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&cfi->chips[chipnum].wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1335
  			mutex_unlock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1336
1337
1338
  
  			schedule();
  			remove_wait_queue(&cfi->chips[chipnum].wq, &wait);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1339
1340
1341
1342
  			goto retry1;
  		}
  
  		tmp_buf = map_read(map, ofs + chipstart);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1343
  		mutex_unlock(&cfi->chips[chipnum].mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1344
1345
  
  		tmp_buf = map_word_load_partial(map, tmp_buf, buf, 0, len);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1346
1347
  
  		ret = do_write_oneword(map, &cfi->chips[chipnum],
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1348
  				ofs, tmp_buf);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1349
  		if (ret)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1350
  			return ret;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1351

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
  		(*retlen) += len;
  	}
  
  	return 0;
  }
  
  
  /*
   * FIXME: interleaved mode not tested, and probably not supported!
   */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1362
  static int __xipram do_write_buffer(struct map_info *map, struct flchip *chip,
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1363
  				    unsigned long adr, const u_char *buf,
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1364
  				    int len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long timeo = jiffies + HZ;
  	/* see comments in do_write_oneword() regarding uWriteTimeo. */
  	unsigned long uWriteTimeout = ( HZ / 1000 ) + 1;
  	int ret = -EIO;
  	unsigned long cmd_adr;
  	int z, words;
  	map_word datum;
  
  	adr += chip->start;
  	cmd_adr = adr;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1377
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1378
1379
  	ret = get_chip(map, chip, adr, FL_WRITING);
  	if (ret) {
c4e773764   Stefani Seibold   mtd: fix a huge l...
1380
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1381
1382
1383
1384
  		return ret;
  	}
  
  	datum = map_word_load(map, buf);
289c05222   Brian Norris   mtd: replace DEBU...
1385
1386
  	pr_debug("MTD %s(): WRITE 0x%.8lx(0x%.8lx)
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1387
  	       __func__, adr, datum.x[0] );
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1388
  	XIP_INVAL_CACHED_RANGE(map, adr, len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1389
  	ENABLE_VPP(map);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1390
  	xip_disable(map, chip, cmd_adr);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1391

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1392
1393
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
  
  	/* Write Buffer Load */
  	map_write(map, CMD(0x25), cmd_adr);
  
  	chip->state = FL_WRITING_TO_BUFFER;
  
  	/* Write length of data to come */
  	words = len / map_bankwidth(map);
  	map_write(map, CMD(words - 1), cmd_adr);
  	/* Write data */
  	z = 0;
  	while(z < words * map_bankwidth(map)) {
  		datum = map_word_load(map, buf);
  		map_write(map, datum, adr + z);
  
  		z += map_bankwidth(map);
  		buf += map_bankwidth(map);
  	}
  	z -= map_bankwidth(map);
  
  	adr += z;
  
  	/* Write Buffer Program Confirm: GO GO GO */
  	map_write(map, CMD(0x29), cmd_adr);
  	chip->state = FL_WRITING;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1419
1420
1421
  	INVALIDATE_CACHE_UDELAY(map, chip,
  				adr, map_bankwidth(map),
  				chip->word_write_time);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1422

1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1423
  	timeo = jiffies + uWriteTimeout;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1424
1425
1426
1427
1428
1429
1430
  	for (;;) {
  		if (chip->state != FL_WRITING) {
  			/* Someone's suspended the write. Sleep */
  			DECLARE_WAITQUEUE(wait, current);
  
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1431
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1432
1433
1434
  			schedule();
  			remove_wait_queue(&chip->wq, &wait);
  			timeo = jiffies + (HZ / 2); /* FIXME */
c4e773764   Stefani Seibold   mtd: fix a huge l...
1435
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1436
1437
  			continue;
  		}
b95f9609c   Konstantin Baidarov   [MTD] chips cfi_c...
1438
1439
  		if (time_after(jiffies, timeo) && !chip_ready(map, adr))
  			break;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1440
1441
  		if (chip_ready(map, adr)) {
  			xip_enable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1442
  			goto op_done;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1443
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1444
1445
  
  		/* Latency issues. Drop the lock, wait a while and retry */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1446
  		UDELAY(map, chip, adr, 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1447
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1448
1449
  	/* reset on all failures. */
  	map_write( map, CMD(0xF0), chip->start );
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1450
  	xip_enable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1451
  	/* FIXME - should have reset delay before continuing */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1452
1453
1454
  	printk(KERN_WARNING "MTD %s(): software timeout
  ",
  	       __func__ );
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1455
1456
1457
1458
  	ret = -EIO;
   op_done:
  	chip->state = FL_READY;
  	put_chip(map, chip, adr);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1459
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
  
  	return ret;
  }
  
  
  static int cfi_amdstd_write_buffers(struct mtd_info *mtd, loff_t to, size_t len,
  				    size_t *retlen, const u_char *buf)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int wbufsize = cfi_interleave(cfi) << cfi->cfiq->MaxBufWriteSize;
  	int ret = 0;
  	int chipnum;
  	unsigned long ofs;
  
  	*retlen = 0;
  	if (!len)
  		return 0;
  
  	chipnum = to >> cfi->chipshift;
  	ofs = to  - (chipnum << cfi->chipshift);
  
  	/* If it's not bus-aligned, do the first word write */
  	if (ofs & (map_bankwidth(map)-1)) {
  		size_t local_len = (-ofs)&(map_bankwidth(map)-1);
  		if (local_len > len)
  			local_len = len;
  		ret = cfi_amdstd_write_words(mtd, ofs + (chipnum<<cfi->chipshift),
  					     local_len, retlen, buf);
  		if (ret)
  			return ret;
  		ofs += local_len;
  		buf += local_len;
  		len -= local_len;
  
  		if (ofs >> cfi->chipshift) {
  			chipnum ++;
  			ofs = 0;
  			if (chipnum == cfi->numchips)
  				return 0;
  		}
  	}
  
  	/* Write buffer is worth it only if more than one word to write... */
  	while (len >= map_bankwidth(map) * 2) {
  		/* We must not cross write block boundaries */
  		int size = wbufsize - (ofs & (wbufsize-1));
  
  		if (size > len)
  			size = len;
  		if (size % map_bankwidth(map))
  			size -= size % map_bankwidth(map);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1512
  		ret = do_write_buffer(map, &cfi->chips[chipnum],
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
  				      ofs, buf, size);
  		if (ret)
  			return ret;
  
  		ofs += size;
  		buf += size;
  		(*retlen) += size;
  		len -= size;
  
  		if (ofs >> cfi->chipshift) {
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1523
  			chipnum ++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
  			ofs = 0;
  			if (chipnum == cfi->numchips)
  				return 0;
  		}
  	}
  
  	if (len) {
  		size_t retlen_dregs = 0;
  
  		ret = cfi_amdstd_write_words(mtd, ofs + (chipnum<<cfi->chipshift),
  					     len, &retlen_dregs, buf);
  
  		*retlen += retlen_dregs;
  		return ret;
  	}
  
  	return 0;
  }
  
  
  /*
   * Handle devices with one erase region, that only implement
   * the chip erase command.
   */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1548
  static int __xipram do_erase_chip(struct map_info *map, struct flchip *chip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1549
1550
1551
1552
1553
1554
1555
1556
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long timeo = jiffies + HZ;
  	unsigned long int adr;
  	DECLARE_WAITQUEUE(wait, current);
  	int ret = 0;
  
  	adr = cfi->addr_unlock1;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1557
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1558
1559
  	ret = get_chip(map, chip, adr, FL_WRITING);
  	if (ret) {
c4e773764   Stefani Seibold   mtd: fix a huge l...
1560
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1561
1562
  		return ret;
  	}
289c05222   Brian Norris   mtd: replace DEBU...
1563
1564
  	pr_debug("MTD %s(): ERASE 0x%.8lx
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1565
  	       __func__, chip->start );
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1566
  	XIP_INVAL_CACHED_RANGE(map, adr, map->size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1567
  	ENABLE_VPP(map);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1568
  	xip_disable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x80, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x10, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  
  	chip->state = FL_ERASING;
  	chip->erase_suspended = 0;
  	chip->in_progress_block_addr = adr;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1579
1580
1581
  	INVALIDATE_CACHE_UDELAY(map, chip,
  				adr, map->size,
  				chip->erase_time*500);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1582
1583
1584
1585
1586
1587
1588
1589
  
  	timeo = jiffies + (HZ*20);
  
  	for (;;) {
  		if (chip->state != FL_ERASING) {
  			/* Someone's suspended the erase. Sleep */
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1590
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1591
1592
  			schedule();
  			remove_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1593
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
  			continue;
  		}
  		if (chip->erase_suspended) {
  			/* This erase was suspended and resumed.
  			   Adjust the timeout */
  			timeo = jiffies + (HZ*20); /* FIXME */
  			chip->erase_suspended = 0;
  		}
  
  		if (chip_ready(map, adr))
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1604
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1605

fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1606
1607
1608
1609
  		if (time_after(jiffies, timeo)) {
  			printk(KERN_WARNING "MTD %s(): software timeout
  ",
  				__func__ );
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1610
  			break;
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1611
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1612
1613
  
  		/* Latency issues. Drop the lock, wait a while and retry */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1614
  		UDELAY(map, chip, adr, 1000000/HZ);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1615
  	}
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1616
1617
1618
1619
1620
  	/* Did we succeed? */
  	if (!chip_good(map, adr, map_word_ff(map))) {
  		/* reset on all failures. */
  		map_write( map, CMD(0xF0), chip->start );
  		/* FIXME - should have reset delay before continuing */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1621

fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1622
1623
  		ret = -EIO;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1624

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1625
  	chip->state = FL_READY;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1626
  	xip_enable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1627
  	put_chip(map, chip, adr);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1628
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1629
1630
1631
  
  	return ret;
  }
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1632
  static int __xipram do_erase_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr, int len, void *thunk)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1633
1634
1635
1636
1637
1638
1639
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	unsigned long timeo = jiffies + HZ;
  	DECLARE_WAITQUEUE(wait, current);
  	int ret = 0;
  
  	adr += chip->start;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1640
  	mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1641
1642
  	ret = get_chip(map, chip, adr, FL_ERASING);
  	if (ret) {
c4e773764   Stefani Seibold   mtd: fix a huge l...
1643
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1644
1645
  		return ret;
  	}
289c05222   Brian Norris   mtd: replace DEBU...
1646
1647
  	pr_debug("MTD %s(): ERASE 0x%.8lx
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1648
  	       __func__, adr );
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1649
  	XIP_INVAL_CACHED_RANGE(map, adr, len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1650
  	ENABLE_VPP(map);
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1651
  	xip_disable(map, chip, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1652
1653
1654
1655
1656
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x80, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, cfi->device_type, NULL);
08968041b   Guillaume LECERF   mtd: cfi_cmdset_0...
1657
  	map_write(map, cfi->sector_erase_cmd, adr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1658
1659
1660
1661
  
  	chip->state = FL_ERASING;
  	chip->erase_suspended = 0;
  	chip->in_progress_block_addr = adr;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1662
1663
1664
1665
  
  	INVALIDATE_CACHE_UDELAY(map, chip,
  				adr, len,
  				chip->erase_time*500);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1666
1667
1668
1669
1670
1671
1672
1673
  
  	timeo = jiffies + (HZ*20);
  
  	for (;;) {
  		if (chip->state != FL_ERASING) {
  			/* Someone's suspended the erase. Sleep */
  			set_current_state(TASK_UNINTERRUPTIBLE);
  			add_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1674
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1675
1676
  			schedule();
  			remove_wait_queue(&chip->wq, &wait);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1677
  			mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1678
1679
1680
1681
1682
1683
1684
1685
  			continue;
  		}
  		if (chip->erase_suspended) {
  			/* This erase was suspended and resumed.
  			   Adjust the timeout */
  			timeo = jiffies + (HZ*20); /* FIXME */
  			chip->erase_suspended = 0;
  		}
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1686
1687
  		if (chip_ready(map, adr)) {
  			xip_enable(map, chip, adr);
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1688
  			break;
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1689
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1690

fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1691
  		if (time_after(jiffies, timeo)) {
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1692
  			xip_enable(map, chip, adr);
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1693
1694
1695
  			printk(KERN_WARNING "MTD %s(): software timeout
  ",
  				__func__ );
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1696
  			break;
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1697
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1698
1699
  
  		/* Latency issues. Drop the lock, wait a while and retry */
02b15e343   Todd Poynor   [MTD] XIP for AMD...
1700
  		UDELAY(map, chip, adr, 1000000/HZ);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1701
  	}
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1702
  	/* Did we succeed? */
22fd9a875   Thomas Gleixner   [MTD] cfi_cmdset_...
1703
  	if (!chip_good(map, adr, map_word_ff(map))) {
fb4a90bfc   Eric W. Biedermann   [MTD] CFI-0002 - ...
1704
1705
1706
1707
1708
1709
  		/* reset on all failures. */
  		map_write( map, CMD(0xF0), chip->start );
  		/* FIXME - should have reset delay before continuing */
  
  		ret = -EIO;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1710

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1711
1712
  	chip->state = FL_READY;
  	put_chip(map, chip, adr);
c4e773764   Stefani Seibold   mtd: fix a huge l...
1713
  	mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1714
1715
  	return ret;
  }
ce0f33ade   Ben Dooks   [MTD] [NOR] cfi_c...
1716
  static int cfi_amdstd_erase_varsize(struct mtd_info *mtd, struct erase_info *instr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1717
1718
1719
1720
1721
1722
1723
1724
1725
1726
1727
1728
1729
  {
  	unsigned long ofs, len;
  	int ret;
  
  	ofs = instr->addr;
  	len = instr->len;
  
  	ret = cfi_varsize_frob(mtd, do_erase_oneblock, ofs, len, NULL);
  	if (ret)
  		return ret;
  
  	instr->state = MTD_ERASE_DONE;
  	mtd_erase_callback(instr);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1730

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
  	return 0;
  }
  
  
  static int cfi_amdstd_erase_chip(struct mtd_info *mtd, struct erase_info *instr)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int ret = 0;
  
  	if (instr->addr != 0)
  		return -EINVAL;
  
  	if (instr->len != mtd->size)
  		return -EINVAL;
  
  	ret = do_erase_chip(map, &cfi->chips[0]);
  	if (ret)
  		return ret;
  
  	instr->state = MTD_ERASE_DONE;
  	mtd_erase_callback(instr);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1753

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1754
1755
  	return 0;
  }
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1756
1757
1758
1759
1760
  static int do_atmel_lock(struct map_info *map, struct flchip *chip,
  			 unsigned long adr, int len, void *thunk)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	int ret;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1761
  	mutex_lock(&chip->mutex);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1762
1763
1764
1765
  	ret = get_chip(map, chip, adr + chip->start, FL_LOCKING);
  	if (ret)
  		goto out_unlock;
  	chip->state = FL_LOCKING;
0a32a1026   Brian Norris   mtd: cleanup styl...
1766
1767
  	pr_debug("MTD %s(): LOCK 0x%08lx len %d
  ", __func__, adr, len);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
  
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x80, cfi->addr_unlock1, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	map_write(map, CMD(0x40), chip->start + adr);
  
  	chip->state = FL_READY;
  	put_chip(map, chip, adr + chip->start);
  	ret = 0;
  
  out_unlock:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1786
  	mutex_unlock(&chip->mutex);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1787
1788
1789
1790
1791
1792
1793
1794
  	return ret;
  }
  
  static int do_atmel_unlock(struct map_info *map, struct flchip *chip,
  			   unsigned long adr, int len, void *thunk)
  {
  	struct cfi_private *cfi = map->fldrv_priv;
  	int ret;
c4e773764   Stefani Seibold   mtd: fix a huge l...
1795
  	mutex_lock(&chip->mutex);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1796
1797
1798
1799
  	ret = get_chip(map, chip, adr + chip->start, FL_UNLOCKING);
  	if (ret)
  		goto out_unlock;
  	chip->state = FL_UNLOCKING;
0a32a1026   Brian Norris   mtd: cleanup styl...
1800
1801
  	pr_debug("MTD %s(): LOCK 0x%08lx len %d
  ", __func__, adr, len);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1802
1803
1804
1805
1806
1807
1808
1809
1810
1811
  
  	cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi,
  			 cfi->device_type, NULL);
  	map_write(map, CMD(0x70), adr);
  
  	chip->state = FL_READY;
  	put_chip(map, chip, adr + chip->start);
  	ret = 0;
  
  out_unlock:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1812
  	mutex_unlock(&chip->mutex);
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1813
1814
  	return ret;
  }
69423d99f   Adrian Hunter   [MTD] update inte...
1815
  static int cfi_atmel_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1816
1817
1818
  {
  	return cfi_varsize_frob(mtd, do_atmel_lock, ofs, len, NULL);
  }
69423d99f   Adrian Hunter   [MTD] update inte...
1819
  static int cfi_atmel_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
0165508c8   Haavard Skinnemoen   MTD: Add lock/unl...
1820
1821
1822
  {
  	return cfi_varsize_frob(mtd, do_atmel_unlock, ofs, len, NULL);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1823
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
  
  static void cfi_amdstd_sync (struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int i;
  	struct flchip *chip;
  	int ret = 0;
  	DECLARE_WAITQUEUE(wait, current);
  
  	for (i=0; !ret && i<cfi->numchips; i++) {
  		chip = &cfi->chips[i];
  
  	retry:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1837
  		mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1838
1839
1840
1841
1842
1843
1844
1845
  
  		switch(chip->state) {
  		case FL_READY:
  		case FL_STATUS:
  		case FL_CFI_QUERY:
  		case FL_JEDEC_QUERY:
  			chip->oldstate = chip->state;
  			chip->state = FL_SYNCING;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1846
  			/* No need to wake_up() on this state change -
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1847
1848
1849
1850
  			 * as the whole point is that nobody can do anything
  			 * with the chip now anyway.
  			 */
  		case FL_SYNCING:
c4e773764   Stefani Seibold   mtd: fix a huge l...
1851
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1852
1853
1854
1855
  			break;
  
  		default:
  			/* Not an idle state */
f8e30e447   Dmitry Adamushko   mtd/chips: add mi...
1856
  			set_current_state(TASK_UNINTERRUPTIBLE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1857
  			add_wait_queue(&chip->wq, &wait);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1858

c4e773764   Stefani Seibold   mtd: fix a huge l...
1859
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1860
1861
1862
1863
  
  			schedule();
  
  			remove_wait_queue(&chip->wq, &wait);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1864

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1865
1866
1867
1868
1869
1870
1871
1872
  			goto retry;
  		}
  	}
  
  	/* Unlock the chips again */
  
  	for (i--; i >=0; i--) {
  		chip = &cfi->chips[i];
c4e773764   Stefani Seibold   mtd: fix a huge l...
1873
  		mutex_lock(&chip->mutex);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1874

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1875
1876
1877
1878
  		if (chip->state == FL_SYNCING) {
  			chip->state = chip->oldstate;
  			wake_up(&chip->wq);
  		}
c4e773764   Stefani Seibold   mtd: fix a huge l...
1879
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1880
1881
1882
1883
1884
1885
1886
1887
1888
1889
1890
1891
1892
1893
  	}
  }
  
  
  static int cfi_amdstd_suspend(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int i;
  	struct flchip *chip;
  	int ret = 0;
  
  	for (i=0; !ret && i<cfi->numchips; i++) {
  		chip = &cfi->chips[i];
c4e773764   Stefani Seibold   mtd: fix a huge l...
1894
  		mutex_lock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1895
1896
1897
1898
1899
1900
1901
1902
  
  		switch(chip->state) {
  		case FL_READY:
  		case FL_STATUS:
  		case FL_CFI_QUERY:
  		case FL_JEDEC_QUERY:
  			chip->oldstate = chip->state;
  			chip->state = FL_PM_SUSPENDED;
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1903
  			/* No need to wake_up() on this state change -
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
  			 * as the whole point is that nobody can do anything
  			 * with the chip now anyway.
  			 */
  		case FL_PM_SUSPENDED:
  			break;
  
  		default:
  			ret = -EAGAIN;
  			break;
  		}
c4e773764   Stefani Seibold   mtd: fix a huge l...
1914
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1915
1916
1917
1918
1919
1920
1921
  	}
  
  	/* Unlock the chips again */
  
  	if (ret) {
  		for (i--; i >=0; i--) {
  			chip = &cfi->chips[i];
c4e773764   Stefani Seibold   mtd: fix a huge l...
1922
  			mutex_lock(&chip->mutex);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1923

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1924
1925
1926
1927
  			if (chip->state == FL_PM_SUSPENDED) {
  				chip->state = chip->oldstate;
  				wake_up(&chip->wq);
  			}
c4e773764   Stefani Seibold   mtd: fix a huge l...
1928
  			mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1929
1930
  		}
  	}
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1931

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
1942
1943
  	return ret;
  }
  
  
  static void cfi_amdstd_resume(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int i;
  	struct flchip *chip;
  
  	for (i=0; i<cfi->numchips; i++) {
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1944

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1945
  		chip = &cfi->chips[i];
c4e773764   Stefani Seibold   mtd: fix a huge l...
1946
  		mutex_lock(&chip->mutex);
1f948b43f   Thomas Gleixner   [MTD] chips: Clea...
1947

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1948
1949
1950
1951
1952
1953
1954
1955
  		if (chip->state == FL_PM_SUSPENDED) {
  			chip->state = FL_READY;
  			map_write(map, CMD(0xF0), chip->start);
  			wake_up(&chip->wq);
  		}
  		else
  			printk(KERN_ERR "Argh. Chip not in PM_SUSPENDED state upon resume()
  ");
c4e773764   Stefani Seibold   mtd: fix a huge l...
1956
  		mutex_unlock(&chip->mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1957
1958
  	}
  }
eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
  
  /*
   * Ensure that the flash device is put back into read array mode before
   * unloading the driver or rebooting.  On some systems, rebooting while
   * the flash is in query/program/erase mode will prevent the CPU from
   * fetching the bootloader code, requiring a hard reset or power cycle.
   */
  static int cfi_amdstd_reset(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
  	int i, ret;
  	struct flchip *chip;
  
  	for (i = 0; i < cfi->numchips; i++) {
  
  		chip = &cfi->chips[i];
  
  		mutex_lock(&chip->mutex);
  
  		ret = get_chip(map, chip, chip->start, FL_SHUTDOWN);
  		if (!ret) {
  			map_write(map, CMD(0xF0), chip->start);
  			chip->state = FL_SHUTDOWN;
  			put_chip(map, chip, chip->start);
  		}
  
  		mutex_unlock(&chip->mutex);
  	}
  
  	return 0;
  }
  
  
  static int cfi_amdstd_reboot(struct notifier_block *nb, unsigned long val,
  			       void *v)
  {
  	struct mtd_info *mtd;
  
  	mtd = container_of(nb, struct mtd_info, reboot_notifier);
  	cfi_amdstd_reset(mtd);
  	return NOTIFY_DONE;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2002
2003
2004
2005
  static void cfi_amdstd_destroy(struct mtd_info *mtd)
  {
  	struct map_info *map = mtd->priv;
  	struct cfi_private *cfi = map->fldrv_priv;
fa671646f   Jesper Juhl   [PATCH] kfree cle...
2006

eafe1311a   Kevin Cernekee   mtd: cfi_cmdset_0...
2007
2008
  	cfi_amdstd_reset(mtd);
  	unregister_reboot_notifier(&mtd->reboot_notifier);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2009
2010
2011
2012
2013
  	kfree(cfi->cmdset_priv);
  	kfree(cfi->cfiq);
  	kfree(cfi);
  	kfree(mtd->eraseregions);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2014
2015
2016
  MODULE_LICENSE("GPL");
  MODULE_AUTHOR("Crossnet Co. <info@crossnet.co.jp> et al.");
  MODULE_DESCRIPTION("MTD chip driver for AMD/Fujitsu flash chips");
804611281   Guillaume LECERF   mtd: cfi_cmdset_0...
2017
  MODULE_ALIAS("cfi_cmdset_0006");
1e804cec7   David Woodhouse   mtd: cfi_cmdset_0...
2018
  MODULE_ALIAS("cfi_cmdset_0701");