Blame view

drivers/mtd/onenand/onenand_base.c 71.8 KB
916527f48   Kyungmin Park   OneNAND support (...
1
2
3
4
5
6
  /*
   *  linux/drivers/mtd/onenand/onenand_base.c
   *
   *  Copyright (C) 2005-2007 Samsung Electronics
   *  Kyungmin Park <kyungmin.park@samsung.com>
   *
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
7
8
9
10
11
   *  Credits:
   *      Adrian Hunter <ext-adrian.hunter@nokia.com>:
   *      auto-placement support, read-while load support, various fixes
   *      Copyright (C) Nokia Corporation, 2007
   *
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
12
13
14
15
16
   *      Rohit Hagargundgi <h.rohit at samsung.com>,
   *      Amul Kumar Saha <amul.saha@samsung.com>:
   *      Flex-OneNAND support
   *      Copyright (C) Samsung Electronics, 2009
   *
916527f48   Kyungmin Park   OneNAND support (...
17
18
19
20
21
22
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   */
  
  #include <common.h>
7b15e2bb9   Mike Frysinger   linux/compat.h: r...
23
  #include <linux/compat.h>
916527f48   Kyungmin Park   OneNAND support (...
24
25
26
27
28
  #include <linux/mtd/mtd.h>
  #include <linux/mtd/onenand.h>
  
  #include <asm/io.h>
  #include <asm/errno.h>
195ccfc59   Fathi BOUDRA   OneNAND: Fill in ...
29
  #include <malloc.h>
916527f48   Kyungmin Park   OneNAND support (...
30

77e475cc0   Kyungmin Park   Fix OneNAND read
31
  /* It should access 16-bit instead of 8-bit */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
32
  static void *memcpy_16(void *dst, const void *src, unsigned int len)
77e475cc0   Kyungmin Park   Fix OneNAND read
33
34
35
36
37
38
39
40
41
42
  {
  	void *ret = dst;
  	short *d = dst;
  	const short *s = src;
  
  	len >>= 1;
  	while (len-- > 0)
  		*d++ = *s++;
  	return ret;
  }
1ae398620   Stefan Roese   OneNAND: Addition...
43
  /**
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
   *  onenand_oob_128 - oob info for Flex-Onenand with 4KB page
   *  For now, we expose only 64 out of 80 ecc bytes
   */
  static struct nand_ecclayout onenand_oob_128 = {
  	.eccbytes	= 64,
  	.eccpos		= {
  		6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
  		22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
  		38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
  		54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
  		70, 71, 72, 73, 74, 75, 76, 77, 78, 79,
  		86, 87, 88, 89, 90, 91, 92, 93, 94, 95,
  		102, 103, 104, 105
  		},
  	.oobfree	= {
  		{2, 4}, {18, 4}, {34, 4}, {50, 4},
  		{66, 4}, {82, 4}, {98, 4}, {114, 4}
  	}
  };
  
  /**
1ae398620   Stefan Roese   OneNAND: Addition...
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
   * onenand_oob_64 - oob info for large (2KB) page
   */
  static struct nand_ecclayout onenand_oob_64 = {
  	.eccbytes	= 20,
  	.eccpos		= {
  		8, 9, 10, 11, 12,
  		24, 25, 26, 27, 28,
  		40, 41, 42, 43, 44,
  		56, 57, 58, 59, 60,
  		},
  	.oobfree	= {
  		{2, 3}, {14, 2}, {18, 3}, {30, 2},
  		{34, 3}, {46, 2}, {50, 3}, {62, 2}
  	}
  };
  
  /**
   * onenand_oob_32 - oob info for middle (1KB) page
   */
  static struct nand_ecclayout onenand_oob_32 = {
  	.eccbytes	= 10,
  	.eccpos		= {
  		8, 9, 10, 11, 12,
  		24, 25, 26, 27, 28,
  		},
  	.oobfree	= { {2, 3}, {14, 2}, {18, 3}, {30, 2} }
  };
916527f48   Kyungmin Park   OneNAND support (...
92
93
94
95
96
97
98
99
100
  static const unsigned char ffchars[] = {
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 16 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 32 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 48 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 64 */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
101
102
103
104
105
106
107
108
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 80 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 96 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 112 */
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
  	0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,	/* 128 */
916527f48   Kyungmin Park   OneNAND support (...
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
  };
  
  /**
   * onenand_readw - [OneNAND Interface] Read OneNAND register
   * @param addr		address to read
   *
   * Read OneNAND register
   */
  static unsigned short onenand_readw(void __iomem * addr)
  {
  	return readw(addr);
  }
  
  /**
   * onenand_writew - [OneNAND Interface] Write OneNAND register with value
   * @param value		value to write
   * @param addr		address to write
   *
   * Write OneNAND register with value
   */
  static void onenand_writew(unsigned short value, void __iomem * addr)
  {
  	writew(value, addr);
  }
  
  /**
   * onenand_block_address - [DEFAULT] Get block address
   * @param device	the device id
   * @param block		the block
   * @return		translated block address if DDP, otherwise same
   *
   * Setup Start Address 1 Register (F100h)
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
142
  static int onenand_block_address(struct onenand_chip *this, int block)
916527f48   Kyungmin Park   OneNAND support (...
143
  {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
144
145
146
  	/* Device Flash Core select, NAND Flash Block Address */
  	if (block & this->density_mask)
  		return ONENAND_DDP_CHIP1 | (block ^ this->density_mask);
916527f48   Kyungmin Park   OneNAND support (...
147
148
149
150
151
152
153
154
155
156
157
158
  
  	return block;
  }
  
  /**
   * onenand_bufferram_address - [DEFAULT] Get bufferram address
   * @param device	the device id
   * @param block		the block
   * @return		set DBS value if DDP, otherwise 0
   *
   * Setup Start Address 2 Register (F101h) for DDP
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
159
  static int onenand_bufferram_address(struct onenand_chip *this, int block)
916527f48   Kyungmin Park   OneNAND support (...
160
  {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
161
162
163
  	/* Device BufferRAM Select */
  	if (block & this->density_mask)
  		return ONENAND_DDP_CHIP1;
916527f48   Kyungmin Park   OneNAND support (...
164

ef0921d6b   Kyungmin Park   Sync with 2.6.27
165
  	return ONENAND_DDP_CHIP0;
916527f48   Kyungmin Park   OneNAND support (...
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
  }
  
  /**
   * onenand_page_address - [DEFAULT] Get page address
   * @param page		the page address
   * @param sector	the sector address
   * @return		combined page and sector address
   *
   * Setup Start Address 8 Register (F107h)
   */
  static int onenand_page_address(int page, int sector)
  {
  	/* Flash Page Address, Flash Sector Address */
  	int fpa, fsa;
  
  	fpa = page & ONENAND_FPA_MASK;
  	fsa = sector & ONENAND_FSA_MASK;
  
  	return ((fpa << ONENAND_FPA_SHIFT) | fsa);
  }
  
  /**
   * onenand_buffer_address - [DEFAULT] Get buffer address
   * @param dataram1	DataRAM index
   * @param sectors	the sector address
   * @param count		the number of sectors
   * @return		the start buffer value
   *
   * Setup Start Buffer Register (F200h)
   */
  static int onenand_buffer_address(int dataram1, int sectors, int count)
  {
  	int bsa, bsc;
  
  	/* BufferRAM Sector Address */
  	bsa = sectors & ONENAND_BSA_MASK;
  
  	if (dataram1)
  		bsa |= ONENAND_BSA_DATARAM1;	/* DataRAM1 */
  	else
  		bsa |= ONENAND_BSA_DATARAM0;	/* DataRAM0 */
  
  	/* BufferRAM Sector Count */
  	bsc = count & ONENAND_BSC_MASK;
  
  	return ((bsa << ONENAND_BSA_SHIFT) | bsc);
  }
  
  /**
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
   * flexonenand_block - Return block number for flash address
   * @param this		- OneNAND device structure
   * @param addr		- Address for which block number is needed
   */
  static unsigned int flexonenand_block(struct onenand_chip *this, loff_t addr)
  {
  	unsigned int boundary, blk, die = 0;
  
  	if (ONENAND_IS_DDP(this) && addr >= this->diesize[0]) {
  		die = 1;
  		addr -= this->diesize[0];
  	}
  
  	boundary = this->boundary[die];
  
  	blk = addr >> (this->erase_shift - 1);
  	if (blk > boundary)
  		blk = (blk + boundary + 1) >> 1;
  
  	blk += die ? this->density_mask : 0;
  	return blk;
  }
  
  unsigned int onenand_block(struct onenand_chip *this, loff_t addr)
  {
  	if (!FLEXONENAND(this))
  		return addr >> this->erase_shift;
  	return flexonenand_block(this, addr);
  }
  
  /**
   * flexonenand_addr - Return address of the block
   * @this:		OneNAND device structure
   * @block:		Block number on Flex-OneNAND
   *
   * Return address of the block
   */
  static loff_t flexonenand_addr(struct onenand_chip *this, int block)
  {
  	loff_t ofs = 0;
  	int die = 0, boundary;
  
  	if (ONENAND_IS_DDP(this) && block >= this->density_mask) {
  		block -= this->density_mask;
  		die = 1;
  		ofs = this->diesize[0];
  	}
  
  	boundary = this->boundary[die];
  	ofs += (loff_t) block << (this->erase_shift - 1);
  	if (block > (boundary + 1))
  		ofs += (loff_t) (block - boundary - 1)
  			<< (this->erase_shift - 1);
  	return ofs;
  }
  
  loff_t onenand_addr(struct onenand_chip *this, int block)
  {
  	if (!FLEXONENAND(this))
  		return (loff_t) block << this->erase_shift;
  	return flexonenand_addr(this, block);
  }
  
  /**
   * flexonenand_region - [Flex-OneNAND] Return erase region of addr
   * @param mtd		MTD device structure
   * @param addr		address whose erase region needs to be identified
   */
  int flexonenand_region(struct mtd_info *mtd, loff_t addr)
  {
  	int i;
  
  	for (i = 0; i < mtd->numeraseregions; i++)
  		if (addr < mtd->eraseregions[i].offset)
  			break;
  	return i - 1;
  }
  
  /**
ef0921d6b   Kyungmin Park   Sync with 2.6.27
294
295
296
297
298
299
300
301
302
303
304
305
   * onenand_get_density - [DEFAULT] Get OneNAND density
   * @param dev_id        OneNAND device ID
   *
   * Get OneNAND density from device ID
   */
  static inline int onenand_get_density(int dev_id)
  {
  	int density = dev_id >> ONENAND_DEVICE_DENSITY_SHIFT;
  	return (density & ONENAND_DEVICE_DENSITY_MASK);
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
306
307
308
309
310
311
312
313
314
315
316
317
318
   * onenand_command - [DEFAULT] Send command to OneNAND device
   * @param mtd		MTD device structure
   * @param cmd		the command to be sent
   * @param addr		offset to read from or write to
   * @param len		number of bytes to read or write
   *
   * Send command to OneNAND device. This function is used for middle/large page
   * devices (1KB/2KB Bytes per page)
   */
  static int onenand_command(struct mtd_info *mtd, int cmd, loff_t addr,
  			   size_t len)
  {
  	struct onenand_chip *this = mtd->priv;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
319
  	int value;
916527f48   Kyungmin Park   OneNAND support (...
320
  	int block, page;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
321

916527f48   Kyungmin Park   OneNAND support (...
322
  	/* Now we use page size operation */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
323
  	int sectors = 0, count = 0;
916527f48   Kyungmin Park   OneNAND support (...
324
325
326
327
328
329
  
  	/* Address translation */
  	switch (cmd) {
  	case ONENAND_CMD_UNLOCK:
  	case ONENAND_CMD_LOCK:
  	case ONENAND_CMD_LOCK_TIGHT:
ef0921d6b   Kyungmin Park   Sync with 2.6.27
330
  	case ONENAND_CMD_UNLOCK_ALL:
916527f48   Kyungmin Park   OneNAND support (...
331
332
333
  		block = -1;
  		page = -1;
  		break;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
334
335
336
337
338
  	case FLEXONENAND_CMD_PI_ACCESS:
  		/* addr contains die index */
  		block = addr * this->density_mask;
  		page = -1;
  		break;
916527f48   Kyungmin Park   OneNAND support (...
339
340
  	case ONENAND_CMD_ERASE:
  	case ONENAND_CMD_BUFFERRAM:
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
341
  		block = onenand_block(this, addr);
916527f48   Kyungmin Park   OneNAND support (...
342
343
  		page = -1;
  		break;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
344
345
346
347
348
  	case FLEXONENAND_CMD_READ_PI:
  		cmd = ONENAND_CMD_READ;
  		block = addr * this->density_mask;
  		page = 0;
  		break;
916527f48   Kyungmin Park   OneNAND support (...
349
  	default:
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
350
351
352
  		block = onenand_block(this, addr);
  		page = (int) (addr
  			- onenand_addr(this, block)) >> this->page_shift;
916527f48   Kyungmin Park   OneNAND support (...
353
354
355
356
357
358
359
  		page &= this->page_mask;
  		break;
  	}
  
  	/* NOTE: The setting order of the registers is very important! */
  	if (cmd == ONENAND_CMD_BUFFERRAM) {
  		/* Select DataRAM for DDP */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
360
  		value = onenand_bufferram_address(this, block);
916527f48   Kyungmin Park   OneNAND support (...
361
362
  		this->write_word(value,
  				 this->base + ONENAND_REG_START_ADDRESS2);
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
363
  		if (ONENAND_IS_4KB_PAGE(this))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
364
365
366
367
  			ONENAND_SET_BUFFERRAM0(this);
  		else
  			/* Switch to the next data buffer */
  			ONENAND_SET_NEXT_BUFFERRAM(this);
916527f48   Kyungmin Park   OneNAND support (...
368
369
370
371
372
373
  
  		return 0;
  	}
  
  	if (block != -1) {
  		/* Write 'DFS, FBA' of Flash */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
374
  		value = onenand_block_address(this, block);
916527f48   Kyungmin Park   OneNAND support (...
375
376
  		this->write_word(value,
  				 this->base + ONENAND_REG_START_ADDRESS1);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
377

cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
378
  		/* Select DataRAM for DDP */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
379
380
381
  		value = onenand_bufferram_address(this, block);
  		this->write_word(value,
  				 this->base + ONENAND_REG_START_ADDRESS2);
916527f48   Kyungmin Park   OneNAND support (...
382
383
384
385
386
387
  	}
  
  	if (page != -1) {
  		int dataram;
  
  		switch (cmd) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
388
  		case FLEXONENAND_CMD_RECOVER_LSB:
916527f48   Kyungmin Park   OneNAND support (...
389
390
  		case ONENAND_CMD_READ:
  		case ONENAND_CMD_READOOB:
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
391
  			if (ONENAND_IS_4KB_PAGE(this))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
392
393
394
  				dataram = ONENAND_SET_BUFFERRAM0(this);
  			else
  				dataram = ONENAND_SET_NEXT_BUFFERRAM(this);
916527f48   Kyungmin Park   OneNAND support (...
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
  			break;
  
  		default:
  			dataram = ONENAND_CURRENT_BUFFERRAM(this);
  			break;
  		}
  
  		/* Write 'FPA, FSA' of Flash */
  		value = onenand_page_address(page, sectors);
  		this->write_word(value,
  				 this->base + ONENAND_REG_START_ADDRESS8);
  
  		/* Write 'BSA, BSC' of DataRAM */
  		value = onenand_buffer_address(dataram, sectors, count);
  		this->write_word(value, this->base + ONENAND_REG_START_BUFFER);
916527f48   Kyungmin Park   OneNAND support (...
410
411
412
413
414
415
416
417
418
419
420
  	}
  
  	/* Interrupt clear */
  	this->write_word(ONENAND_INT_CLEAR, this->base + ONENAND_REG_INTERRUPT);
  	/* Write command */
  	this->write_word(cmd, this->base + ONENAND_REG_COMMAND);
  
  	return 0;
  }
  
  /**
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
   * onenand_read_ecc - return ecc status
   * @param this		onenand chip structure
   */
  static int onenand_read_ecc(struct onenand_chip *this)
  {
  	int ecc, i;
  
  	if (!FLEXONENAND(this))
  		return this->read_word(this->base + ONENAND_REG_ECC_STATUS);
  
  	for (i = 0; i < 4; i++) {
  		ecc = this->read_word(this->base
  				+ ((ONENAND_REG_ECC_STATUS + i) << 1));
  		if (likely(!ecc))
  			continue;
  		if (ecc & FLEXONENAND_UNCORRECTABLE_ERROR)
  			return ONENAND_ECC_2BIT_ALL;
  	}
  
  	return 0;
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
444
445
446
447
448
449
450
451
452
453
454
455
456
   * onenand_wait - [DEFAULT] wait until the command is done
   * @param mtd		MTD device structure
   * @param state		state to select the max. timeout value
   *
   * Wait for command done. This applies to all OneNAND command
   * Read can take up to 30us, erase up to 2ms and program up to 350us
   * according to general OneNAND specs
   */
  static int onenand_wait(struct mtd_info *mtd, int state)
  {
  	struct onenand_chip *this = mtd->priv;
  	unsigned int flags = ONENAND_INT_MASTER;
  	unsigned int interrupt = 0;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
457
  	unsigned int ctrl;
916527f48   Kyungmin Park   OneNAND support (...
458
459
460
461
462
463
464
465
  
  	while (1) {
  		interrupt = this->read_word(this->base + ONENAND_REG_INTERRUPT);
  		if (interrupt & flags)
  			break;
  	}
  
  	ctrl = this->read_word(this->base + ONENAND_REG_CTRL_STATUS);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
466
467
468
469
470
471
472
473
  	if (interrupt & ONENAND_INT_READ) {
  		int ecc = onenand_read_ecc(this);
  		if (ecc & ONENAND_ECC_2BIT_ALL) {
  			printk("onenand_wait: ECC error = 0x%04x
  ", ecc);
  			return -EBADMSG;
  		}
  	}
916527f48   Kyungmin Park   OneNAND support (...
474
  	if (ctrl & ONENAND_CTRL_ERROR) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
475
476
477
478
479
480
  		printk("onenand_wait: controller error = 0x%04x
  ", ctrl);
  		if (ctrl & ONENAND_CTRL_LOCK)
  			printk("onenand_wait: it's locked error = 0x%04x
  ",
  				ctrl);
916527f48   Kyungmin Park   OneNAND support (...
481

916527f48   Kyungmin Park   OneNAND support (...
482
483
  		return -EIO;
  	}
916527f48   Kyungmin Park   OneNAND support (...
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
  
  	return 0;
  }
  
  /**
   * onenand_bufferram_offset - [DEFAULT] BufferRAM offset
   * @param mtd		MTD data structure
   * @param area		BufferRAM area
   * @return		offset given area
   *
   * Return BufferRAM offset given area
   */
  static inline int onenand_bufferram_offset(struct mtd_info *mtd, int area)
  {
  	struct onenand_chip *this = mtd->priv;
  
  	if (ONENAND_CURRENT_BUFFERRAM(this)) {
  		if (area == ONENAND_DATARAM)
d438d5084   Kyungmin Park   Fix OneNAND build...
502
  			return mtd->writesize;
916527f48   Kyungmin Park   OneNAND support (...
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
  		if (area == ONENAND_SPARERAM)
  			return mtd->oobsize;
  	}
  
  	return 0;
  }
  
  /**
   * onenand_read_bufferram - [OneNAND Interface] Read the bufferram area
   * @param mtd		MTD data structure
   * @param area		BufferRAM area
   * @param buffer	the databuffer to put/get data
   * @param offset	offset to read from or write to
   * @param count		number of bytes to read/write
   *
   * Read the BufferRAM area
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
520
  static int onenand_read_bufferram(struct mtd_info *mtd, loff_t addr, int area,
916527f48   Kyungmin Park   OneNAND support (...
521
522
523
524
525
526
527
528
  				  unsigned char *buffer, int offset,
  				  size_t count)
  {
  	struct onenand_chip *this = mtd->priv;
  	void __iomem *bufferram;
  
  	bufferram = this->base + area;
  	bufferram += onenand_bufferram_offset(mtd, area);
d2c6fbec4   Wolfgang Denk   onenand: rename 1...
529
  	memcpy_16(buffer, bufferram + offset, count);
916527f48   Kyungmin Park   OneNAND support (...
530
531
532
533
534
535
536
537
538
539
540
541
542
543
  
  	return 0;
  }
  
  /**
   * onenand_sync_read_bufferram - [OneNAND Interface] Read the bufferram area with Sync. Burst mode
   * @param mtd		MTD data structure
   * @param area		BufferRAM area
   * @param buffer	the databuffer to put/get data
   * @param offset	offset to read from or write to
   * @param count		number of bytes to read/write
   *
   * Read the BufferRAM area with Sync. Burst Mode
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
544
  static int onenand_sync_read_bufferram(struct mtd_info *mtd, loff_t addr, int area,
916527f48   Kyungmin Park   OneNAND support (...
545
546
547
548
549
550
551
552
553
554
  				       unsigned char *buffer, int offset,
  				       size_t count)
  {
  	struct onenand_chip *this = mtd->priv;
  	void __iomem *bufferram;
  
  	bufferram = this->base + area;
  	bufferram += onenand_bufferram_offset(mtd, area);
  
  	this->mmcontrol(mtd, ONENAND_SYS_CFG1_SYNC_READ);
d2c6fbec4   Wolfgang Denk   onenand: rename 1...
555
  	memcpy_16(buffer, bufferram + offset, count);
916527f48   Kyungmin Park   OneNAND support (...
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
  
  	this->mmcontrol(mtd, 0);
  
  	return 0;
  }
  
  /**
   * onenand_write_bufferram - [OneNAND Interface] Write the bufferram area
   * @param mtd		MTD data structure
   * @param area		BufferRAM area
   * @param buffer	the databuffer to put/get data
   * @param offset	offset to read from or write to
   * @param count		number of bytes to read/write
   *
   * Write the BufferRAM area
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
572
  static int onenand_write_bufferram(struct mtd_info *mtd, loff_t addr, int area,
916527f48   Kyungmin Park   OneNAND support (...
573
574
575
576
577
578
579
580
  				   const unsigned char *buffer, int offset,
  				   size_t count)
  {
  	struct onenand_chip *this = mtd->priv;
  	void __iomem *bufferram;
  
  	bufferram = this->base + area;
  	bufferram += onenand_bufferram_offset(mtd, area);
d2c6fbec4   Wolfgang Denk   onenand: rename 1...
581
  	memcpy_16(bufferram + offset, buffer, count);
916527f48   Kyungmin Park   OneNAND support (...
582
583
584
585
586
  
  	return 0;
  }
  
  /**
4fca3310d   Stefan Roese   OneNAND: Fix comp...
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
   * onenand_get_2x_blockpage - [GENERIC] Get blockpage at 2x program mode
   * @param mtd		MTD data structure
   * @param addr		address to check
   * @return		blockpage address
   *
   * Get blockpage address at 2x program mode
   */
  static int onenand_get_2x_blockpage(struct mtd_info *mtd, loff_t addr)
  {
  	struct onenand_chip *this = mtd->priv;
  	int blockpage, block, page;
  
  	/* Calculate the even block number */
  	block = (int) (addr >> this->erase_shift) & ~1;
  	/* Is it the odd plane? */
  	if (addr & this->writesize)
  		block++;
  	page = (int) (addr >> (this->page_shift + 1)) & this->page_mask;
  	blockpage = (block << 7) | page;
  
  	return blockpage;
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
611
612
613
614
615
616
617
618
619
620
   * onenand_check_bufferram - [GENERIC] Check BufferRAM information
   * @param mtd		MTD data structure
   * @param addr		address to check
   * @return		1 if there are valid data, otherwise 0
   *
   * Check bufferram if there is data we required
   */
  static int onenand_check_bufferram(struct mtd_info *mtd, loff_t addr)
  {
  	struct onenand_chip *this = mtd->priv;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
621
622
  	int blockpage, found = 0;
  	unsigned int i;
916527f48   Kyungmin Park   OneNAND support (...
623

ef0921d6b   Kyungmin Park   Sync with 2.6.27
624
625
626
627
  	if (ONENAND_IS_2PLANE(this))
  		blockpage = onenand_get_2x_blockpage(mtd, addr);
  	else
  		blockpage = (int) (addr >> this->page_shift);
916527f48   Kyungmin Park   OneNAND support (...
628
629
  
  	/* Is there valid data? */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
630
631
632
633
634
635
636
637
638
639
640
  	i = ONENAND_CURRENT_BUFFERRAM(this);
  	if (this->bufferram[i].blockpage == blockpage)
  		found = 1;
  	else {
  		/* Check another BufferRAM */
  		i = ONENAND_NEXT_BUFFERRAM(this);
  		if (this->bufferram[i].blockpage == blockpage) {
  			ONENAND_SET_NEXT_BUFFERRAM(this);
  			found = 1;
  		}
  	}
916527f48   Kyungmin Park   OneNAND support (...
641

ef0921d6b   Kyungmin Park   Sync with 2.6.27
642
643
  	if (found && ONENAND_IS_DDP(this)) {
  		/* Select DataRAM for DDP */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
644
  		int block = onenand_block(this, addr);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
645
646
647
648
649
  		int value = onenand_bufferram_address(this, block);
  		this->write_word(value, this->base + ONENAND_REG_START_ADDRESS2);
  	}
  
  	return found;
916527f48   Kyungmin Park   OneNAND support (...
650
651
652
653
654
655
656
657
658
659
660
661
662
663
  }
  
  /**
   * onenand_update_bufferram - [GENERIC] Update BufferRAM information
   * @param mtd		MTD data structure
   * @param addr		address to update
   * @param valid		valid flag
   *
   * Update BufferRAM information
   */
  static int onenand_update_bufferram(struct mtd_info *mtd, loff_t addr,
  				    int valid)
  {
  	struct onenand_chip *this = mtd->priv;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
664
665
  	int blockpage;
  	unsigned int i;
916527f48   Kyungmin Park   OneNAND support (...
666

ef0921d6b   Kyungmin Park   Sync with 2.6.27
667
668
669
670
  	if (ONENAND_IS_2PLANE(this))
  		blockpage = onenand_get_2x_blockpage(mtd, addr);
  	else
  		blockpage = (int)(addr >> this->page_shift);
916527f48   Kyungmin Park   OneNAND support (...
671

ef0921d6b   Kyungmin Park   Sync with 2.6.27
672
673
674
675
  	/* Invalidate another BufferRAM */
  	i = ONENAND_NEXT_BUFFERRAM(this);
  	if (this->bufferram[i].blockpage == blockpage)
  		this->bufferram[i].blockpage = -1;
916527f48   Kyungmin Park   OneNAND support (...
676
677
678
  
  	/* Update BufferRAM */
  	i = ONENAND_CURRENT_BUFFERRAM(this);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
679
680
681
682
  	if (valid)
  		this->bufferram[i].blockpage = blockpage;
  	else
  		this->bufferram[i].blockpage = -1;
916527f48   Kyungmin Park   OneNAND support (...
683
684
685
686
687
  
  	return 0;
  }
  
  /**
d438d5084   Kyungmin Park   Fix OneNAND build...
688
689
690
691
692
693
694
695
   * onenand_invalidate_bufferram - [GENERIC] Invalidate BufferRAM information
   * @param mtd           MTD data structure
   * @param addr          start address to invalidate
   * @param len           length to invalidate
   *
   * Invalidate BufferRAM information
   */
  static void onenand_invalidate_bufferram(struct mtd_info *mtd, loff_t addr,
4b0708093   Wolfgang Denk   Coding Style clea...
696
  					 unsigned int len)
d438d5084   Kyungmin Park   Fix OneNAND build...
697
698
699
700
701
702
703
  {
  	struct onenand_chip *this = mtd->priv;
  	int i;
  	loff_t end_addr = addr + len;
  
  	/* Invalidate BufferRAM */
  	for (i = 0; i < MAX_BUFFERRAM; i++) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
704
  		loff_t buf_addr = this->bufferram[i].blockpage << this->page_shift;
d438d5084   Kyungmin Park   Fix OneNAND build...
705
706
  
  		if (buf_addr >= addr && buf_addr < end_addr)
ef0921d6b   Kyungmin Park   Sync with 2.6.27
707
  			this->bufferram[i].blockpage = -1;
d438d5084   Kyungmin Park   Fix OneNAND build...
708
709
710
711
  	}
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
   * onenand_get_device - [GENERIC] Get chip for selected access
   * @param mtd		MTD device structure
   * @param new_state	the state which is requested
   *
   * Get the device and lock it for exclusive access
   */
  static void onenand_get_device(struct mtd_info *mtd, int new_state)
  {
  	/* Do nothing */
  }
  
  /**
   * onenand_release_device - [GENERIC] release chip
   * @param mtd		MTD device structure
   *
   * Deselect, release chip lock and wake up anyone waiting on the device
   */
  static void onenand_release_device(struct mtd_info *mtd)
  {
  	/* Do nothing */
  }
  
  /**
dfe64e2c8   Sergey Lapin   mtd: resync with ...
735
   * onenand_transfer_auto_oob - [INTERN] oob auto-placement transfer
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
   * @param mtd		MTD device structure
   * @param buf		destination address
   * @param column	oob offset to read from
   * @param thislen	oob length to read
   */
  static int onenand_transfer_auto_oob(struct mtd_info *mtd, uint8_t *buf,
  					int column, int thislen)
  {
  	struct onenand_chip *this = mtd->priv;
  	struct nand_oobfree *free;
  	int readcol = column;
  	int readend = column + thislen;
  	int lastgap = 0;
  	unsigned int i;
  	uint8_t *oob_buf = this->oob_buf;
  
  	free = this->ecclayout->oobfree;
  	for (i = 0; i < MTD_MAX_OOBFREE_ENTRIES && free->length; i++, free++) {
  		if (readcol >= lastgap)
  			readcol += free->offset - lastgap;
  		if (readend >= lastgap)
  			readend += free->offset - lastgap;
  		lastgap = free->offset + free->length;
  	}
ef0921d6b   Kyungmin Park   Sync with 2.6.27
760
  	this->read_bufferram(mtd, 0, ONENAND_SPARERAM, oob_buf, 0, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
  	free = this->ecclayout->oobfree;
  	for (i = 0; i < MTD_MAX_OOBFREE_ENTRIES && free->length; i++, free++) {
  		int free_end = free->offset + free->length;
  		if (free->offset < readend && free_end > readcol) {
  			int st = max_t(int,free->offset,readcol);
  			int ed = min_t(int,free_end,readend);
  			int n = ed - st;
  			memcpy(buf, oob_buf + st, n);
  			buf += n;
  		} else if (column == 0)
  			break;
  	}
  	return 0;
  }
  
  /**
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
   * onenand_recover_lsb - [Flex-OneNAND] Recover LSB page data
   * @param mtd		MTD device structure
   * @param addr		address to recover
   * @param status	return value from onenand_wait
   *
   * MLC NAND Flash cell has paired pages - LSB page and MSB page. LSB page has
   * lower page address and MSB page has higher page address in paired pages.
   * If power off occurs during MSB page program, the paired LSB page data can
   * become corrupt. LSB page recovery read is a way to read LSB page though page
   * data are corrupted. When uncorrectable error occurs as a result of LSB page
   * read after power up, issue LSB page recovery read.
   */
  static int onenand_recover_lsb(struct mtd_info *mtd, loff_t addr, int status)
  {
  	struct onenand_chip *this = mtd->priv;
  	int i;
  
  	/* Recovery is only for Flex-OneNAND */
  	if (!FLEXONENAND(this))
  		return status;
  
  	/* check if we failed due to uncorrectable error */
dfe64e2c8   Sergey Lapin   mtd: resync with ...
799
  	if (!mtd_is_eccerr(status) && status != ONENAND_BBT_READ_ECC_ERROR)
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
  		return status;
  
  	/* check if address lies in MLC region */
  	i = flexonenand_region(mtd, addr);
  	if (mtd->eraseregions[i].erasesize < (1 << this->erase_shift))
  		return status;
  
  	printk("onenand_recover_lsb:"
  		"Attempting to recover from uncorrectable read
  ");
  
  	/* Issue the LSB page recovery command */
  	this->command(mtd, FLEXONENAND_CMD_RECOVER_LSB, addr, this->writesize);
  	return this->wait(mtd, FL_READING);
  }
  
  /**
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
817
   * onenand_read_ops_nolock - [OneNAND Interface] OneNAND read main and/or out-of-band
916527f48   Kyungmin Park   OneNAND support (...
818
819
   * @param mtd		MTD device structure
   * @param from		offset to read from
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
820
   * @param ops		oob operation description structure
916527f48   Kyungmin Park   OneNAND support (...
821
   *
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
822
   * OneNAND read main and/or out-of-band data
916527f48   Kyungmin Park   OneNAND support (...
823
   */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
824
825
  static int onenand_read_ops_nolock(struct mtd_info *mtd, loff_t from,
  		struct mtd_oob_ops *ops)
916527f48   Kyungmin Park   OneNAND support (...
826
827
  {
  	struct onenand_chip *this = mtd->priv;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
828
829
830
831
832
833
834
835
836
  	struct mtd_ecc_stats stats;
  	size_t len = ops->len;
  	size_t ooblen = ops->ooblen;
  	u_char *buf = ops->datbuf;
  	u_char *oobbuf = ops->oobbuf;
  	int read = 0, column, thislen;
  	int oobread = 0, oobcolumn, thisooblen, oobsize;
  	int ret = 0, boundary = 0;
  	int writesize = this->writesize;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
837
838
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_read_ops_nolock: from = 0x%08x, len = %i
  ", (unsigned int) from, (int) len);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
839

dfe64e2c8   Sergey Lapin   mtd: resync with ...
840
  	if (ops->mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
841
842
843
  		oobsize = this->ecclayout->oobavail;
  	else
  		oobsize = mtd->oobsize;
916527f48   Kyungmin Park   OneNAND support (...
844

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
845
  	oobcolumn = from & (mtd->oobsize - 1);
916527f48   Kyungmin Park   OneNAND support (...
846
847
848
  
  	/* Do not allow reads past end of device */
  	if ((from + len) > mtd->size) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
849
850
851
852
  		printk(KERN_ERR "onenand_read_ops_nolock: Attempt read beyond end of device
  ");
  		ops->retlen = 0;
  		ops->oobretlen = 0;
916527f48   Kyungmin Park   OneNAND support (...
853
854
  		return -EINVAL;
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
855
  	stats = mtd->ecc_stats;
916527f48   Kyungmin Park   OneNAND support (...
856

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
857
  	/* Read-while-load method */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
858
  	/* Note: We can't use this feature in MLC */
916527f48   Kyungmin Park   OneNAND support (...
859

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
860
861
  	/* Do first load to bufferRAM */
  	if (read < len) {
916527f48   Kyungmin Park   OneNAND support (...
862
  		if (!onenand_check_bufferram(mtd, from)) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
863
  			this->main_buf = buf;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
864
  			this->command(mtd, ONENAND_CMD_READ, from, writesize);
916527f48   Kyungmin Park   OneNAND support (...
865
  			ret = this->wait(mtd, FL_READING);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
866
867
  			if (unlikely(ret))
  				ret = onenand_recover_lsb(mtd, from, ret);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
868
869
870
  			onenand_update_bufferram(mtd, from, !ret);
  			if (ret == -EBADMSG)
  				ret = 0;
916527f48   Kyungmin Park   OneNAND support (...
871
  		}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
872
  	}
916527f48   Kyungmin Park   OneNAND support (...
873

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
874
875
876
877
  	thislen = min_t(int, writesize, len - read);
  	column = from & (writesize - 1);
  	if (column + thislen > writesize)
  		thislen = writesize - column;
916527f48   Kyungmin Park   OneNAND support (...
878

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
879
880
881
  	while (!ret) {
  		/* If there is more to load then start next load */
  		from += thislen;
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
882
  		if (!ONENAND_IS_4KB_PAGE(this) && read + thislen < len) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
883
  			this->main_buf = buf + thislen;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
884
885
886
887
888
889
890
891
892
893
894
895
896
897
  			this->command(mtd, ONENAND_CMD_READ, from, writesize);
  			/*
  			 * Chip boundary handling in DDP
  			 * Now we issued chip 1 read and pointed chip 1
  			 * bufferam so we have to point chip 0 bufferam.
  			 */
  			if (ONENAND_IS_DDP(this) &&
  					unlikely(from == (this->chipsize >> 1))) {
  				this->write_word(ONENAND_DDP_CHIP0, this->base + ONENAND_REG_START_ADDRESS2);
  				boundary = 1;
  			} else
  				boundary = 0;
  			ONENAND_SET_PREV_BUFFERRAM(this);
  		}
916527f48   Kyungmin Park   OneNAND support (...
898

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
899
  		/* While load is going, read from last bufferRAM */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
900
  		this->read_bufferram(mtd, from - thislen, ONENAND_DATARAM, buf, column, thislen);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
901
902
903
904
905
  
  		/* Read oob area if needed */
  		if (oobbuf) {
  			thisooblen = oobsize - oobcolumn;
  			thisooblen = min_t(int, thisooblen, ooblen - oobread);
dfe64e2c8   Sergey Lapin   mtd: resync with ...
906
  			if (ops->mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
907
908
  				onenand_transfer_auto_oob(mtd, oobbuf, oobcolumn, thisooblen);
  			else
ef0921d6b   Kyungmin Park   Sync with 2.6.27
909
  				this->read_bufferram(mtd, 0, ONENAND_SPARERAM, oobbuf, oobcolumn, thisooblen);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
910
911
912
  			oobread += thisooblen;
  			oobbuf += thisooblen;
  			oobcolumn = 0;
916527f48   Kyungmin Park   OneNAND support (...
913
  		}
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
914
  		if (ONENAND_IS_4KB_PAGE(this) && (read + thislen < len)) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
915
916
917
918
919
  			this->command(mtd, ONENAND_CMD_READ, from, writesize);
  			ret = this->wait(mtd, FL_READING);
  			if (unlikely(ret))
  				ret = onenand_recover_lsb(mtd, from, ret);
  			onenand_update_bufferram(mtd, from, !ret);
dfe64e2c8   Sergey Lapin   mtd: resync with ...
920
  			if (mtd_is_eccerr(ret))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
921
922
  				ret = 0;
  		}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
923
924
925
926
927
928
929
  		/* See if we are done */
  		read += thislen;
  		if (read == len)
  			break;
  		/* Set up for next read from bufferRAM */
  		if (unlikely(boundary))
  			this->write_word(ONENAND_DDP_CHIP1, this->base + ONENAND_REG_START_ADDRESS2);
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
930
  		if (!ONENAND_IS_4KB_PAGE(this))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
931
  			ONENAND_SET_NEXT_BUFFERRAM(this);
916527f48   Kyungmin Park   OneNAND support (...
932
  		buf += thislen;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
933
934
  		thislen = min_t(int, writesize, len - read);
  		column = 0;
916527f48   Kyungmin Park   OneNAND support (...
935

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
936
  		if (!ONENAND_IS_4KB_PAGE(this)) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
937
938
939
  			/* Now wait for load */
  			ret = this->wait(mtd, FL_READING);
  			onenand_update_bufferram(mtd, from, !ret);
dfe64e2c8   Sergey Lapin   mtd: resync with ...
940
  			if (mtd_is_eccerr(ret))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
941
942
  				ret = 0;
  		}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
943
  	}
916527f48   Kyungmin Park   OneNAND support (...
944
945
946
947
948
949
  
  	/*
  	 * Return success, if no ECC failures, else -EBADMSG
  	 * fs driver will take care of that, because
  	 * retlen == desired len and result == -EBADMSG
  	 */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
950
951
952
953
954
955
956
957
  	ops->retlen = read;
  	ops->oobretlen = oobread;
  
  	if (ret)
  		return ret;
  
  	if (mtd->ecc_stats.failed - stats.failed)
  		return -EBADMSG;
40462e541   Paul Burton   mtd: driver _read...
958
959
  	/* return max bitflips per ecc step; ONENANDs correct 1 bit only */
  	return mtd->ecc_stats.corrected != stats.corrected ? 1 : 0;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
  }
  
  /**
   * onenand_read_oob_nolock - [MTD Interface] OneNAND read out-of-band
   * @param mtd		MTD device structure
   * @param from		offset to read from
   * @param ops		oob operation description structure
   *
   * OneNAND read out-of-band data from the spare area
   */
  static int onenand_read_oob_nolock(struct mtd_info *mtd, loff_t from,
  		struct mtd_oob_ops *ops)
  {
  	struct onenand_chip *this = mtd->priv;
  	struct mtd_ecc_stats stats;
  	int read = 0, thislen, column, oobsize;
  	size_t len = ops->ooblen;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
977
  	unsigned int mode = ops->mode;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
978
  	u_char *buf = ops->oobbuf;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
979
  	int ret = 0, readcmd;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
980
981
  
  	from += ops->ooboffs;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
982
983
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_read_oob_nolock: from = 0x%08x, len = %i
  ", (unsigned int) from, (int) len);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
984
985
986
  
  	/* Initialize return length value */
  	ops->oobretlen = 0;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
987
  	if (mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
  		oobsize = this->ecclayout->oobavail;
  	else
  		oobsize = mtd->oobsize;
  
  	column = from & (mtd->oobsize - 1);
  
  	if (unlikely(column >= oobsize)) {
  		printk(KERN_ERR "onenand_read_oob_nolock: Attempted to start read outside oob
  ");
  		return -EINVAL;
  	}
  
  	/* Do not allow reads past end of device */
  	if (unlikely(from >= mtd->size ||
  		column + len > ((mtd->size >> this->page_shift) -
  				(from >> this->page_shift)) * oobsize)) {
  		printk(KERN_ERR "onenand_read_oob_nolock: Attempted to read beyond end of device
  ");
  		return -EINVAL;
  	}
  
  	stats = mtd->ecc_stats;
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
1010
1011
  	readcmd = ONENAND_IS_4KB_PAGE(this) ?
  		ONENAND_CMD_READ : ONENAND_CMD_READOOB;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1012

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1013
1014
1015
  	while (read < len) {
  		thislen = oobsize - column;
  		thislen = min_t(int, thislen, len);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1016
  		this->spare_buf = buf;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1017
  		this->command(mtd, readcmd, from, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1018
1019
1020
1021
  
  		onenand_update_bufferram(mtd, from, 0);
  
  		ret = this->wait(mtd, FL_READING);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1022
1023
  		if (unlikely(ret))
  			ret = onenand_recover_lsb(mtd, from, ret);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1024
1025
1026
1027
1028
  		if (ret && ret != -EBADMSG) {
  			printk(KERN_ERR "onenand_read_oob_nolock: read failed = 0x%x
  ", ret);
  			break;
  		}
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1029
  		if (mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1030
1031
  			onenand_transfer_auto_oob(mtd, buf, column, thislen);
  		else
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1032
  			this->read_bufferram(mtd, 0, ONENAND_SPARERAM, buf, column, thislen);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
  
  		read += thislen;
  
  		if (read == len)
  			break;
  
  		buf += thislen;
  
  		/* Read more? */
  		if (read < len) {
  			/* Page size */
  			from += mtd->writesize;
  			column = 0;
  		}
  	}
  
  	ops->oobretlen = read;
  
  	if (ret)
  		return ret;
  
  	if (mtd->ecc_stats.failed - stats.failed)
  		return -EBADMSG;
  
  	return 0;
916527f48   Kyungmin Park   OneNAND support (...
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
  }
  
  /**
   * onenand_read - [MTD Interface] MTD compability function for onenand_read_ecc
   * @param mtd		MTD device structure
   * @param from		offset to read from
   * @param len		number of bytes to read
   * @param retlen	pointer to variable to store the number of read bytes
   * @param buf		the databuffer to put data
   *
   * This function simply calls onenand_read_ecc with oob buffer and oobsel = NULL
  */
  int onenand_read(struct mtd_info *mtd, loff_t from, size_t len,
  		 size_t * retlen, u_char * buf)
  {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
  	struct mtd_oob_ops ops = {
  		.len    = len,
  		.ooblen = 0,
  		.datbuf = buf,
  		.oobbuf = NULL,
  	};
  	int ret;
  
  	onenand_get_device(mtd, FL_READING);
  	ret = onenand_read_ops_nolock(mtd, from, &ops);
  	onenand_release_device(mtd);
  
  	*retlen = ops.retlen;
  	return ret;
916527f48   Kyungmin Park   OneNAND support (...
1087
1088
1089
1090
1091
1092
  }
  
  /**
   * onenand_read_oob - [MTD Interface] OneNAND read out-of-band
   * @param mtd		MTD device structure
   * @param from		offset to read from
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1093
   * @param ops		oob operations description structure
916527f48   Kyungmin Park   OneNAND support (...
1094
   *
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1095
   * OneNAND main and/or out-of-band
916527f48   Kyungmin Park   OneNAND support (...
1096
   */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1097
1098
1099
1100
1101
1102
  int onenand_read_oob(struct mtd_info *mtd, loff_t from,
  			struct mtd_oob_ops *ops)
  {
  	int ret;
  
  	switch (ops->mode) {
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1103
1104
  	case MTD_OPS_PLACE_OOB:
  	case MTD_OPS_AUTO_OOB:
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1105
  		break;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1106
  	case MTD_OPS_RAW:
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
  		/* Not implemented yet */
  	default:
  		return -EINVAL;
  	}
  
  	onenand_get_device(mtd, FL_READING);
  	if (ops->datbuf)
  		ret = onenand_read_ops_nolock(mtd, from, ops);
  	else
  		ret = onenand_read_oob_nolock(mtd, from, ops);
  	onenand_release_device(mtd);
  
  	return ret;
  }
  
  /**
   * onenand_bbt_wait - [DEFAULT] wait until the command is done
   * @param mtd		MTD device structure
   * @param state		state to select the max. timeout value
   *
   * Wait for command done.
   */
  static int onenand_bbt_wait(struct mtd_info *mtd, int state)
  {
  	struct onenand_chip *this = mtd->priv;
  	unsigned int flags = ONENAND_INT_MASTER;
  	unsigned int interrupt;
  	unsigned int ctrl;
  
  	while (1) {
  		interrupt = this->read_word(this->base + ONENAND_REG_INTERRUPT);
  		if (interrupt & flags)
  			break;
  	}
  
  	/* To get correct interrupt status in timeout case */
  	interrupt = this->read_word(this->base + ONENAND_REG_INTERRUPT);
  	ctrl = this->read_word(this->base + ONENAND_REG_CTRL_STATUS);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1145
  	if (interrupt & ONENAND_INT_READ) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1146
1147
1148
1149
1150
  		int ecc = onenand_read_ecc(this);
  		if (ecc & ONENAND_ECC_2BIT_ALL) {
  			printk(KERN_INFO "onenand_bbt_wait: ecc error = 0x%04x"
  				", controller = 0x%04x
  ", ecc, ctrl);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1151
  			return ONENAND_BBT_READ_ERROR;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1152
  		}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1153
1154
1155
1156
1157
1158
  	} else {
  		printk(KERN_ERR "onenand_bbt_wait: read timeout!"
  				"ctrl=0x%04x intr=0x%04x
  ", ctrl, interrupt);
  		return ONENAND_BBT_READ_FATAL_ERROR;
  	}
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1159
1160
1161
1162
1163
1164
  	/* Initial bad block case: 0x2400 or 0x0400 */
  	if (ctrl & ONENAND_CTRL_ERROR) {
  		printk(KERN_DEBUG "onenand_bbt_wait: controller error = 0x%04x
  ", ctrl);
  		return ONENAND_BBT_READ_ERROR;
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
  	return 0;
  }
  
  /**
   * onenand_bbt_read_oob - [MTD Interface] OneNAND read out-of-band for bbt scan
   * @param mtd		MTD device structure
   * @param from		offset to read from
   * @param ops		oob operation description structure
   *
   * OneNAND read out-of-band data from the spare area for bbt scan
   */
  int onenand_bbt_read_oob(struct mtd_info *mtd, loff_t from,
  		struct mtd_oob_ops *ops)
916527f48   Kyungmin Park   OneNAND support (...
1178
1179
1180
  {
  	struct onenand_chip *this = mtd->priv;
  	int read = 0, thislen, column;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1181
  	int ret = 0, readcmd;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1182
1183
  	size_t len = ops->ooblen;
  	u_char *buf = ops->oobbuf;
916527f48   Kyungmin Park   OneNAND support (...
1184

ef0921d6b   Kyungmin Park   Sync with 2.6.27
1185
1186
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_bbt_read_oob: from = 0x%08x, len = %zi
  ", (unsigned int) from, len);
916527f48   Kyungmin Park   OneNAND support (...
1187

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
1188
1189
  	readcmd = ONENAND_IS_4KB_PAGE(this) ?
  		ONENAND_CMD_READ : ONENAND_CMD_READOOB;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1190

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1191
1192
  	/* Initialize return value */
  	ops->oobretlen = 0;
916527f48   Kyungmin Park   OneNAND support (...
1193
1194
1195
  
  	/* Do not allow reads past end of device */
  	if (unlikely((from + len) > mtd->size)) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1196
1197
1198
  		printk(KERN_ERR "onenand_bbt_read_oob: Attempt read beyond end of device
  ");
  		return ONENAND_BBT_READ_FATAL_ERROR;
916527f48   Kyungmin Park   OneNAND support (...
1199
1200
1201
1202
1203
1204
1205
1206
  	}
  
  	/* Grab the lock and see if the device is available */
  	onenand_get_device(mtd, FL_READING);
  
  	column = from & (mtd->oobsize - 1);
  
  	while (read < len) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1207

916527f48   Kyungmin Park   OneNAND support (...
1208
1209
  		thislen = mtd->oobsize - column;
  		thislen = min_t(int, thislen, len);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1210
  		this->spare_buf = buf;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1211
  		this->command(mtd, readcmd, from, mtd->oobsize);
916527f48   Kyungmin Park   OneNAND support (...
1212
1213
  
  		onenand_update_bufferram(mtd, from, 0);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1214
  		ret = this->bbt_wait(mtd, FL_READING);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1215
1216
  		if (unlikely(ret))
  			ret = onenand_recover_lsb(mtd, from, ret);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1217
1218
  		if (ret)
  			break;
916527f48   Kyungmin Park   OneNAND support (...
1219

ce3277a6f   Kyungmin Park   OneNAND: Remove u...
1220
  		this->read_bufferram(mtd, 0, ONENAND_SPARERAM, buf, column, thislen);
916527f48   Kyungmin Park   OneNAND support (...
1221
1222
1223
  		read += thislen;
  		if (read == len)
  			break;
916527f48   Kyungmin Park   OneNAND support (...
1224
  		buf += thislen;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1225

916527f48   Kyungmin Park   OneNAND support (...
1226
1227
  		/* Read more? */
  		if (read < len) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1228
1229
  			/* Update Page size */
  			from += this->writesize;
916527f48   Kyungmin Park   OneNAND support (...
1230
1231
1232
1233
1234
1235
  			column = 0;
  		}
  	}
  
  	/* Deselect and wake up anyone waiting on the device */
  	onenand_release_device(mtd);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1236
  	ops->oobretlen = read;
916527f48   Kyungmin Park   OneNAND support (...
1237
1238
  	return ret;
  }
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1239

916527f48   Kyungmin Park   OneNAND support (...
1240
1241
  #ifdef CONFIG_MTD_ONENAND_VERIFY_WRITE
  /**
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1242
1243
1244
1245
   * onenand_verify_oob - [GENERIC] verify the oob contents after a write
   * @param mtd           MTD device structure
   * @param buf           the databuffer to verify
   * @param to            offset to read from
916527f48   Kyungmin Park   OneNAND support (...
1246
   */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1247
  static int onenand_verify_oob(struct mtd_info *mtd, const u_char *buf, loff_t to)
916527f48   Kyungmin Park   OneNAND support (...
1248
1249
  {
  	struct onenand_chip *this = mtd->priv;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1250
  	u_char *oob_buf = this->oob_buf;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1251
  	int status, i, readcmd;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1252

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
1253
1254
  	readcmd = ONENAND_IS_4KB_PAGE(this) ?
  		ONENAND_CMD_READ : ONENAND_CMD_READOOB;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1255
1256
  
  	this->command(mtd, readcmd, to, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1257
1258
1259
1260
  	onenand_update_bufferram(mtd, to, 0);
  	status = this->wait(mtd, FL_READING);
  	if (status)
  		return status;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1261
  	this->read_bufferram(mtd, 0, ONENAND_SPARERAM, oob_buf, 0, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
  	for (i = 0; i < mtd->oobsize; i++)
  		if (buf[i] != 0xFF && buf[i] != oob_buf[i])
  			return -EBADMSG;
  
  	return 0;
  }
  
  /**
   * onenand_verify - [GENERIC] verify the chip contents after a write
   * @param mtd          MTD device structure
   * @param buf          the databuffer to verify
   * @param addr         offset to read from
   * @param len          number of bytes to read and compare
   */
  static int onenand_verify(struct mtd_info *mtd, const u_char *buf, loff_t addr, size_t len)
  {
  	struct onenand_chip *this = mtd->priv;
  	void __iomem *dataram;
916527f48   Kyungmin Park   OneNAND support (...
1280
  	int ret = 0;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1281
  	int thislen, column;
916527f48   Kyungmin Park   OneNAND support (...
1282

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1283
1284
1285
1286
1287
  	while (len != 0) {
  		thislen = min_t(int, this->writesize, len);
  		column = addr & (this->writesize - 1);
  		if (column + thislen > this->writesize)
  			thislen = this->writesize - column;
916527f48   Kyungmin Park   OneNAND support (...
1288

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1289
  		this->command(mtd, ONENAND_CMD_READ, addr, this->writesize);
916527f48   Kyungmin Park   OneNAND support (...
1290

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1291
  		onenand_update_bufferram(mtd, addr, 0);
916527f48   Kyungmin Park   OneNAND support (...
1292

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1293
1294
1295
  		ret = this->wait(mtd, FL_READING);
  		if (ret)
  			return ret;
916527f48   Kyungmin Park   OneNAND support (...
1296

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
  		onenand_update_bufferram(mtd, addr, 1);
  
  		dataram = this->base + ONENAND_DATARAM;
  		dataram += onenand_bufferram_offset(mtd, ONENAND_DATARAM);
  
  		if (memcmp(buf, dataram + column, thislen))
  			return -EBADMSG;
  
  		len -= thislen;
  		buf += thislen;
  		addr += thislen;
  	}
916527f48   Kyungmin Park   OneNAND support (...
1309
1310
1311
1312
  
  	return 0;
  }
  #else
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1313
1314
  #define onenand_verify(...)             (0)
  #define onenand_verify_oob(...)         (0)
916527f48   Kyungmin Park   OneNAND support (...
1315
  #endif
1ae398620   Stefan Roese   OneNAND: Addition...
1316
  #define NOTALIGNED(x)	((x & (this->subpagesize - 1)) != 0)
916527f48   Kyungmin Park   OneNAND support (...
1317
1318
  
  /**
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1319
   * onenand_fill_auto_oob - [INTERN] oob auto-placement transfer
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
   * @param mtd           MTD device structure
   * @param oob_buf       oob buffer
   * @param buf           source address
   * @param column        oob offset to write to
   * @param thislen       oob length to write
   */
  static int onenand_fill_auto_oob(struct mtd_info *mtd, u_char *oob_buf,
  		const u_char *buf, int column, int thislen)
  {
  	struct onenand_chip *this = mtd->priv;
  	struct nand_oobfree *free;
  	int writecol = column;
  	int writeend = column + thislen;
  	int lastgap = 0;
  	unsigned int i;
  
  	free = this->ecclayout->oobfree;
  	for (i = 0; i < MTD_MAX_OOBFREE_ENTRIES && free->length; i++, free++) {
  		if (writecol >= lastgap)
  			writecol += free->offset - lastgap;
  		if (writeend >= lastgap)
  			writeend += free->offset - lastgap;
  		lastgap = free->offset + free->length;
  	}
  	free = this->ecclayout->oobfree;
  	for (i = 0; i < MTD_MAX_OOBFREE_ENTRIES && free->length; i++, free++) {
  		int free_end = free->offset + free->length;
  		if (free->offset < writeend && free_end > writecol) {
  			int st = max_t(int,free->offset,writecol);
  			int ed = min_t(int,free_end,writeend);
  			int n = ed - st;
  			memcpy(oob_buf + st, buf, n);
  			buf += n;
  		} else if (column == 0)
  			break;
  	}
  	return 0;
  }
  
  /**
   * onenand_write_ops_nolock - [OneNAND Interface] write main and/or out-of-band
   * @param mtd           MTD device structure
   * @param to            offset to write to
   * @param ops           oob operation description structure
916527f48   Kyungmin Park   OneNAND support (...
1364
   *
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1365
   * Write main and/or oob with ECC
916527f48   Kyungmin Park   OneNAND support (...
1366
   */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1367
1368
  static int onenand_write_ops_nolock(struct mtd_info *mtd, loff_t to,
  		struct mtd_oob_ops *ops)
916527f48   Kyungmin Park   OneNAND support (...
1369
1370
  {
  	struct onenand_chip *this = mtd->priv;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1371
1372
1373
1374
1375
1376
1377
  	int written = 0, column, thislen, subpage;
  	int oobwritten = 0, oobcolumn, thisooblen, oobsize;
  	size_t len = ops->len;
  	size_t ooblen = ops->ooblen;
  	const u_char *buf = ops->datbuf;
  	const u_char *oob = ops->oobbuf;
  	u_char *oobbuf;
916527f48   Kyungmin Park   OneNAND support (...
1378
  	int ret = 0;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1379
1380
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_write_ops_nolock: to = 0x%08x, len = %i
  ", (unsigned int) to, (int) len);
916527f48   Kyungmin Park   OneNAND support (...
1381
1382
  
  	/* Initialize retlen, in case of early exit */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1383
1384
  	ops->retlen = 0;
  	ops->oobretlen = 0;
916527f48   Kyungmin Park   OneNAND support (...
1385

916527f48   Kyungmin Park   OneNAND support (...
1386
  	/* Reject writes, which are not page aligned */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1387
1388
1389
  	if (unlikely(NOTALIGNED(to) || NOTALIGNED(len))) {
  		printk(KERN_ERR "onenand_write_ops_nolock: Attempt to write not page aligned data
  ");
916527f48   Kyungmin Park   OneNAND support (...
1390
1391
  		return -EINVAL;
  	}
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1392
  	if (ops->mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1393
1394
1395
1396
1397
1398
1399
  		oobsize = this->ecclayout->oobavail;
  	else
  		oobsize = mtd->oobsize;
  
  	oobcolumn = to & (mtd->oobsize - 1);
  
  	column = to & (mtd->writesize - 1);
916527f48   Kyungmin Park   OneNAND support (...
1400
1401
1402
  
  	/* Loop until all data write */
  	while (written < len) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1403
  		u_char *wbuf = (u_char *) buf;
916527f48   Kyungmin Park   OneNAND support (...
1404

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1405
1406
  		thislen = min_t(int, mtd->writesize - column, len - written);
  		thisooblen = min_t(int, oobsize - oobcolumn, ooblen - oobwritten);
916527f48   Kyungmin Park   OneNAND support (...
1407

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1408
  		this->command(mtd, ONENAND_CMD_BUFFERRAM, to, thislen);
916527f48   Kyungmin Park   OneNAND support (...
1409

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1410
1411
1412
1413
1414
1415
1416
  		/* Partial page write */
  		subpage = thislen < mtd->writesize;
  		if (subpage) {
  			memset(this->page_buf, 0xff, mtd->writesize);
  			memcpy(this->page_buf + column, buf, thislen);
  			wbuf = this->page_buf;
  		}
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1417
  		this->write_bufferram(mtd, to, ONENAND_DATARAM, wbuf, 0, mtd->writesize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1418
1419
1420
  
  		if (oob) {
  			oobbuf = this->oob_buf;
916527f48   Kyungmin Park   OneNAND support (...
1421

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1422
1423
1424
  			/* We send data to spare ram with oobsize
  			 *                          * to prevent byte access */
  			memset(oobbuf, 0xff, mtd->oobsize);
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1425
  			if (ops->mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1426
1427
1428
1429
1430
1431
1432
1433
1434
  				onenand_fill_auto_oob(mtd, oobbuf, oob, oobcolumn, thisooblen);
  			else
  				memcpy(oobbuf + oobcolumn, oob, thisooblen);
  
  			oobwritten += thisooblen;
  			oob += thisooblen;
  			oobcolumn = 0;
  		} else
  			oobbuf = (u_char *) ffchars;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1435
  		this->write_bufferram(mtd, 0, ONENAND_SPARERAM, oobbuf, 0, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1436
1437
  
  		this->command(mtd, ONENAND_CMD_PROG, to, mtd->writesize);
916527f48   Kyungmin Park   OneNAND support (...
1438
1439
  
  		ret = this->wait(mtd, FL_WRITING);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1440
1441
1442
1443
1444
1445
1446
  
  		/* In partial page write we don't update bufferram */
  		onenand_update_bufferram(mtd, to, !ret && !subpage);
  		if (ONENAND_IS_2PLANE(this)) {
  			ONENAND_SET_BUFFERRAM1(this);
  			onenand_update_bufferram(mtd, to + this->writesize, !ret && !subpage);
  		}
916527f48   Kyungmin Park   OneNAND support (...
1447
  		if (ret) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1448
1449
  			printk(KERN_ERR "onenand_write_ops_nolock: write filaed %d
  ", ret);
916527f48   Kyungmin Park   OneNAND support (...
1450
1451
  			break;
  		}
916527f48   Kyungmin Park   OneNAND support (...
1452
  		/* Only check verify write turn on */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1453
  		ret = onenand_verify(mtd, buf, to, thislen);
916527f48   Kyungmin Park   OneNAND support (...
1454
  		if (ret) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1455
1456
  			printk(KERN_ERR "onenand_write_ops_nolock: verify failed %d
  ", ret);
916527f48   Kyungmin Park   OneNAND support (...
1457
1458
  			break;
  		}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1459
  		written += thislen;
916527f48   Kyungmin Park   OneNAND support (...
1460
1461
  		if (written == len)
  			break;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1462
  		column = 0;
916527f48   Kyungmin Park   OneNAND support (...
1463
1464
1465
  		to += thislen;
  		buf += thislen;
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1466
  	ops->retlen = written;
916527f48   Kyungmin Park   OneNAND support (...
1467
1468
1469
1470
1471
  
  	return ret;
  }
  
  /**
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1472
   * onenand_write_oob_nolock - [INTERN] OneNAND write out-of-band
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1473
1474
1475
1476
1477
1478
   * @param mtd           MTD device structure
   * @param to            offset to write to
   * @param len           number of bytes to write
   * @param retlen        pointer to variable to store the number of written bytes
   * @param buf           the data to write
   * @param mode          operation mode
916527f48   Kyungmin Park   OneNAND support (...
1479
1480
1481
   *
   * OneNAND write out-of-band
   */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1482
1483
  static int onenand_write_oob_nolock(struct mtd_info *mtd, loff_t to,
  		struct mtd_oob_ops *ops)
916527f48   Kyungmin Park   OneNAND support (...
1484
1485
  {
  	struct onenand_chip *this = mtd->priv;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1486
  	int column, ret = 0, oobsize;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1487
  	int written = 0, oobcmd;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1488
1489
1490
  	u_char *oobbuf;
  	size_t len = ops->ooblen;
  	const u_char *buf = ops->oobbuf;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1491
  	unsigned int mode = ops->mode;
916527f48   Kyungmin Park   OneNAND support (...
1492

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1493
  	to += ops->ooboffs;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1494
1495
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_write_oob_nolock: to = 0x%08x, len = %i
  ", (unsigned int) to, (int) len);
916527f48   Kyungmin Park   OneNAND support (...
1496
1497
  
  	/* Initialize retlen, in case of early exit */
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1498
  	ops->oobretlen = 0;
916527f48   Kyungmin Park   OneNAND support (...
1499

dfe64e2c8   Sergey Lapin   mtd: resync with ...
1500
  	if (mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1501
1502
1503
1504
1505
1506
1507
1508
1509
  		oobsize = this->ecclayout->oobavail;
  	else
  		oobsize = mtd->oobsize;
  
  	column = to & (mtd->oobsize - 1);
  
  	if (unlikely(column >= oobsize)) {
  		printk(KERN_ERR "onenand_write_oob_nolock: Attempted to start write outside oob
  ");
916527f48   Kyungmin Park   OneNAND support (...
1510
1511
  		return -EINVAL;
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
1529
  	/* For compatibility with NAND: Do not allow write past end of page */
  	if (unlikely(column + len > oobsize)) {
  		printk(KERN_ERR "onenand_write_oob_nolock: "
  				"Attempt to write past end of page
  ");
  		return -EINVAL;
  	}
  
  	/* Do not allow reads past end of device */
  	if (unlikely(to >= mtd->size ||
  				column + len > ((mtd->size >> this->page_shift) -
  					(to >> this->page_shift)) * oobsize)) {
  		printk(KERN_ERR "onenand_write_oob_nolock: Attempted to write past end of device
  ");
  		return -EINVAL;
  	}
  
  	oobbuf = this->oob_buf;
916527f48   Kyungmin Park   OneNAND support (...
1530

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
1531
1532
  	oobcmd = ONENAND_IS_4KB_PAGE(this) ?
  		ONENAND_CMD_PROG : ONENAND_CMD_PROGOOB;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1533

916527f48   Kyungmin Park   OneNAND support (...
1534
1535
  	/* Loop until all data write */
  	while (written < len) {
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1536
  		int thislen = min_t(int, oobsize, len - written);
916527f48   Kyungmin Park   OneNAND support (...
1537
1538
  
  		this->command(mtd, ONENAND_CMD_BUFFERRAM, to, mtd->oobsize);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1539
1540
1541
  		/* We send data to spare ram with oobsize
  		 * to prevent byte access */
  		memset(oobbuf, 0xff, mtd->oobsize);
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1542
  		if (mode == MTD_OPS_AUTO_OOB)
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1543
1544
1545
  			onenand_fill_auto_oob(mtd, oobbuf, buf, column, thislen);
  		else
  			memcpy(oobbuf + column, buf, thislen);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1546
  		this->write_bufferram(mtd, 0, ONENAND_SPARERAM, oobbuf, 0, mtd->oobsize);
916527f48   Kyungmin Park   OneNAND support (...
1547

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
1548
  		if (ONENAND_IS_4KB_PAGE(this)) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1549
1550
1551
1552
1553
1554
1555
  			/* Set main area of DataRAM to 0xff*/
  			memset(this->page_buf, 0xff, mtd->writesize);
  			this->write_bufferram(mtd, 0, ONENAND_DATARAM,
  				this->page_buf,	0, mtd->writesize);
  		}
  
  		this->command(mtd, oobcmd, to, mtd->oobsize);
916527f48   Kyungmin Park   OneNAND support (...
1556
1557
  
  		onenand_update_bufferram(mtd, to, 0);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1558
1559
1560
1561
  		if (ONENAND_IS_2PLANE(this)) {
  			ONENAND_SET_BUFFERRAM1(this);
  			onenand_update_bufferram(mtd, to + this->writesize, 0);
  		}
916527f48   Kyungmin Park   OneNAND support (...
1562

bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1563
1564
1565
1566
  		ret = this->wait(mtd, FL_WRITING);
  		if (ret) {
  			printk(KERN_ERR "onenand_write_oob_nolock: write failed %d
  ", ret);
916527f48   Kyungmin Park   OneNAND support (...
1567
  			break;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1568
1569
1570
1571
1572
1573
1574
1575
  		}
  
  		ret = onenand_verify_oob(mtd, oobbuf, to);
  		if (ret) {
  			printk(KERN_ERR "onenand_write_oob_nolock: verify failed %d
  ", ret);
  			break;
  		}
916527f48   Kyungmin Park   OneNAND support (...
1576
1577
1578
1579
  
  		written += thislen;
  		if (written == len)
  			break;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1580
  		to += mtd->writesize;
916527f48   Kyungmin Park   OneNAND support (...
1581
  		buf += thislen;
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1582
  		column = 0;
916527f48   Kyungmin Park   OneNAND support (...
1583
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
  	ops->oobretlen = written;
  
  	return ret;
  }
  
  /**
   * onenand_write - [MTD Interface] compability function for onenand_write_ecc
   * @param mtd		MTD device structure
   * @param to		offset to write to
   * @param len		number of bytes to write
   * @param retlen	pointer to variable to store the number of written bytes
   * @param buf		the data to write
   *
   * Write with ECC
   */
  int onenand_write(struct mtd_info *mtd, loff_t to, size_t len,
  		  size_t * retlen, const u_char * buf)
  {
  	struct mtd_oob_ops ops = {
  		.len    = len,
  		.ooblen = 0,
  		.datbuf = (u_char *) buf,
  		.oobbuf = NULL,
  	};
  	int ret;
  
  	onenand_get_device(mtd, FL_WRITING);
  	ret = onenand_write_ops_nolock(mtd, to, &ops);
916527f48   Kyungmin Park   OneNAND support (...
1612
  	onenand_release_device(mtd);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
  	*retlen = ops.retlen;
  	return ret;
  }
  
  /**
   * onenand_write_oob - [MTD Interface] OneNAND write out-of-band
   * @param mtd		MTD device structure
   * @param to		offset to write to
   * @param ops		oob operation description structure
   *
   * OneNAND write main and/or out-of-band
   */
  int onenand_write_oob(struct mtd_info *mtd, loff_t to,
  			struct mtd_oob_ops *ops)
  {
  	int ret;
  
  	switch (ops->mode) {
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1631
1632
  	case MTD_OPS_PLACE_OOB:
  	case MTD_OPS_AUTO_OOB:
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1633
  		break;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1634
  	case MTD_OPS_RAW:
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
  		/* Not implemented yet */
  	default:
  		return -EINVAL;
  	}
  
  	onenand_get_device(mtd, FL_WRITING);
  	if (ops->datbuf)
  		ret = onenand_write_ops_nolock(mtd, to, ops);
  	else
  		ret = onenand_write_oob_nolock(mtd, to, ops);
  	onenand_release_device(mtd);
  
  	return ret;
916527f48   Kyungmin Park   OneNAND support (...
1648

916527f48   Kyungmin Park   OneNAND support (...
1649
1650
1651
  }
  
  /**
d438d5084   Kyungmin Park   Fix OneNAND build...
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
   * onenand_block_isbad_nolock - [GENERIC] Check if a block is marked bad
   * @param mtd		MTD device structure
   * @param ofs		offset from device start
   * @param allowbbt	1, if its allowed to access the bbt area
   *
   * Check, if the block is bad, Either by reading the bad block table or
   * calling of the scan function.
   */
  static int onenand_block_isbad_nolock(struct mtd_info *mtd, loff_t ofs, int allowbbt)
  {
  	struct onenand_chip *this = mtd->priv;
  	struct bbm_info *bbm = this->bbm;
  
  	/* Return info from the table */
  	return bbm->isbad_bbt(mtd, ofs, allowbbt);
  }
  
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
   * onenand_erase - [MTD Interface] erase block(s)
   * @param mtd		MTD device structure
   * @param instr		erase instruction
   *
   * Erase one ore more blocks
   */
  int onenand_erase(struct mtd_info *mtd, struct erase_info *instr)
  {
  	struct onenand_chip *this = mtd->priv;
  	unsigned int block_size;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1681
1682
1683
1684
1685
  	loff_t addr = instr->addr;
  	unsigned int len = instr->len;
  	int ret = 0, i;
  	struct mtd_erase_region_info *region = NULL;
  	unsigned int region_end = 0;
916527f48   Kyungmin Park   OneNAND support (...
1686

cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1687
1688
1689
  	MTDDEBUG(MTD_DEBUG_LEVEL3, "onenand_erase: start = 0x%08x, len = %i
  ",
  			(unsigned int) addr, len);
916527f48   Kyungmin Park   OneNAND support (...
1690

cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
1717
1718
  	if (FLEXONENAND(this)) {
  		/* Find the eraseregion of this address */
  		i = flexonenand_region(mtd, addr);
  		region = &mtd->eraseregions[i];
  
  		block_size = region->erasesize;
  		region_end = region->offset
  			+ region->erasesize * region->numblocks;
  
  		/* Start address within region must align on block boundary.
  		 * Erase region's start offset is always block start address.
  		 */
  		if (unlikely((addr - region->offset) & (block_size - 1))) {
  			MTDDEBUG(MTD_DEBUG_LEVEL0, "onenand_erase:"
  				" Unaligned address
  ");
  			return -EINVAL;
  		}
  	} else {
  		block_size = 1 << this->erase_shift;
  
  		/* Start address must align on block boundary */
  		if (unlikely(addr & (block_size - 1))) {
  			MTDDEBUG(MTD_DEBUG_LEVEL0, "onenand_erase:"
  						"Unaligned address
  ");
  			return -EINVAL;
  		}
916527f48   Kyungmin Park   OneNAND support (...
1719
  	}
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1720
1721
  	/* Length must align on block boundary */
  	if (unlikely(len & (block_size - 1))) {
3167c5386   Scott Wood   NAND: Rename DEBU...
1722
  		MTDDEBUG (MTD_DEBUG_LEVEL0,
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1723
1724
  			 "onenand_erase: Length not block aligned
  ");
916527f48   Kyungmin Park   OneNAND support (...
1725
1726
  		return -EINVAL;
  	}
916527f48   Kyungmin Park   OneNAND support (...
1727
1728
1729
1730
  	/* Grab the lock and see if the device is available */
  	onenand_get_device(mtd, FL_ERASING);
  
  	/* Loop throught the pages */
916527f48   Kyungmin Park   OneNAND support (...
1731
1732
1733
  	instr->state = MTD_ERASING;
  
  	while (len) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1734
1735
1736
1737
1738
1739
1740
1741
1742
  		/* Check if we have a bad block, we do not erase bad blocks */
  		if (instr->priv == 0 && onenand_block_isbad_nolock(mtd, addr, 0)) {
  			printk(KERN_WARNING "onenand_erase: attempt to erase"
  				" a bad block at addr 0x%08x
  ",
  				(unsigned int) addr);
  			instr->state = MTD_ERASE_FAILED;
  			goto erase_exit;
  		}
916527f48   Kyungmin Park   OneNAND support (...
1743
1744
  
  		this->command(mtd, ONENAND_CMD_ERASE, addr, block_size);
d438d5084   Kyungmin Park   Fix OneNAND build...
1745
  		onenand_invalidate_bufferram(mtd, addr, block_size);
916527f48   Kyungmin Park   OneNAND support (...
1746
1747
1748
1749
  		ret = this->wait(mtd, FL_ERASING);
  		/* Check, if it is write protected */
  		if (ret) {
  			if (ret == -EPERM)
3167c5386   Scott Wood   NAND: Rename DEBU...
1750
  				MTDDEBUG (MTD_DEBUG_LEVEL0, "onenand_erase: "
4b0708093   Wolfgang Denk   Coding Style clea...
1751
1752
  					  "Device is write protected!!!
  ");
916527f48   Kyungmin Park   OneNAND support (...
1753
  			else
3167c5386   Scott Wood   NAND: Rename DEBU...
1754
  				MTDDEBUG (MTD_DEBUG_LEVEL0, "onenand_erase: "
4b0708093   Wolfgang Denk   Coding Style clea...
1755
1756
  					  "Failed erase, block %d
  ",
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1757
  					onenand_block(this, addr));
916527f48   Kyungmin Park   OneNAND support (...
1758
1759
  			instr->state = MTD_ERASE_FAILED;
  			instr->fail_addr = addr;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1760

916527f48   Kyungmin Park   OneNAND support (...
1761
1762
1763
1764
1765
  			goto erase_exit;
  		}
  
  		len -= block_size;
  		addr += block_size;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
  
  		if (addr == region_end) {
  			if (!len)
  				break;
  			region++;
  
  			block_size = region->erasesize;
  			region_end = region->offset
  				+ region->erasesize * region->numblocks;
  
  			if (len & (block_size - 1)) {
  				/* This has been checked at MTD
  				 * partitioning level. */
  				printk("onenand_erase: Unaligned address
  ");
  				goto erase_exit;
  			}
  		}
916527f48   Kyungmin Park   OneNAND support (...
1784
1785
1786
  	}
  
  	instr->state = MTD_ERASE_DONE;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1787
  erase_exit:
916527f48   Kyungmin Park   OneNAND support (...
1788
1789
1790
1791
1792
1793
1794
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
  
  	ret = instr->state == MTD_ERASE_DONE ? 0 : -EIO;
  	/* Do call back function */
  	if (!ret)
  		mtd_erase_callback(instr);
  
  	/* Deselect and wake up anyone waiting on the device */
  	onenand_release_device(mtd);
  
  	return ret;
  }
  
  /**
   * onenand_sync - [MTD Interface] sync
   * @param mtd		MTD device structure
   *
   * Sync is actually a wait for chip ready function
   */
  void onenand_sync(struct mtd_info *mtd)
  {
3167c5386   Scott Wood   NAND: Rename DEBU...
1808
1809
  	MTDDEBUG (MTD_DEBUG_LEVEL3, "onenand_sync: called
  ");
916527f48   Kyungmin Park   OneNAND support (...
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
  
  	/* Grab the lock and see if the device is available */
  	onenand_get_device(mtd, FL_SYNCING);
  
  	/* Release it and go back */
  	onenand_release_device(mtd);
  }
  
  /**
   * onenand_block_isbad - [MTD Interface] Check whether the block at the given offset is bad
   * @param mtd		MTD device structure
   * @param ofs		offset relative to mtd start
d438d5084   Kyungmin Park   Fix OneNAND build...
1822
1823
   *
   * Check whether the block is bad
916527f48   Kyungmin Park   OneNAND support (...
1824
1825
1826
   */
  int onenand_block_isbad(struct mtd_info *mtd, loff_t ofs)
  {
d438d5084   Kyungmin Park   Fix OneNAND build...
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
  	int ret;
  
  	/* Check for invalid offset */
  	if (ofs > mtd->size)
  		return -EINVAL;
  
  	onenand_get_device(mtd, FL_READING);
  	ret = onenand_block_isbad_nolock(mtd,ofs, 0);
  	onenand_release_device(mtd);
  	return ret;
916527f48   Kyungmin Park   OneNAND support (...
1837
1838
1839
  }
  
  /**
1714f51a2   Kyungmin Park   Add markbad function
1840
1841
1842
1843
1844
1845
1846
1847
1848
1849
1850
1851
1852
   * onenand_default_block_markbad - [DEFAULT] mark a block bad
   * @param mtd           MTD device structure
   * @param ofs           offset from device start
   *
   * This is the default implementation, which can be overridden by
   * a hardware specific driver.
   */
  static int onenand_default_block_markbad(struct mtd_info *mtd, loff_t ofs)
  {
  	struct onenand_chip *this = mtd->priv;
  	struct bbm_info *bbm = this->bbm;
  	u_char buf[2] = {0, 0};
  	struct mtd_oob_ops ops = {
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1853
  		.mode = MTD_OPS_PLACE_OOB,
1714f51a2   Kyungmin Park   Add markbad function
1854
1855
1856
1857
1858
1859
1860
  		.ooblen = 2,
  		.oobbuf = buf,
  		.ooboffs = 0,
  	};
  	int block;
  
  	/* Get block number */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1861
  	block = onenand_block(this, ofs);
1714f51a2   Kyungmin Park   Add markbad function
1862
1863
1864
1865
1866
1867
1868
1869
1870
  	if (bbm->bbt)
  		bbm->bbt[block >> 2] |= 0x01 << ((block & 0x03) << 1);
  
  	/* We write two bytes, so we dont have to mess with 16 bit access */
  	ofs += mtd->oobsize + (bbm->badblockpos & ~0x01);
  	return onenand_write_oob_nolock(mtd, ofs, &ops);
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
1871
1872
1873
   * onenand_block_markbad - [MTD Interface] Mark the block at the given offset as bad
   * @param mtd		MTD device structure
   * @param ofs		offset relative to mtd start
d438d5084   Kyungmin Park   Fix OneNAND build...
1874
1875
   *
   * Mark the block as bad
916527f48   Kyungmin Park   OneNAND support (...
1876
1877
1878
   */
  int onenand_block_markbad(struct mtd_info *mtd, loff_t ofs)
  {
d438d5084   Kyungmin Park   Fix OneNAND build...
1879
1880
1881
1882
1883
1884
1885
1886
1887
  	int ret;
  
  	ret = onenand_block_isbad(mtd, ofs);
  	if (ret) {
  		/* If it was bad already, return success and do nothing */
  		if (ret > 0)
  			return 0;
  		return ret;
  	}
dfe64e2c8   Sergey Lapin   mtd: resync with ...
1888
  	ret = mtd_block_markbad(mtd, ofs);
d438d5084   Kyungmin Park   Fix OneNAND build...
1889
  	return ret;
916527f48   Kyungmin Park   OneNAND support (...
1890
1891
1892
  }
  
  /**
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1893
1894
1895
1896
1897
   * onenand_do_lock_cmd - [OneNAND Interface] Lock or unlock block(s)
   * @param mtd           MTD device structure
   * @param ofs           offset relative to mtd start
   * @param len           number of bytes to lock or unlock
   * @param cmd           lock or unlock command
916527f48   Kyungmin Park   OneNAND support (...
1898
   *
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1899
   * Lock or unlock one or more blocks
916527f48   Kyungmin Park   OneNAND support (...
1900
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1901
  static int onenand_do_lock_cmd(struct mtd_info *mtd, loff_t ofs, size_t len, int cmd)
916527f48   Kyungmin Park   OneNAND support (...
1902
1903
1904
  {
  	struct onenand_chip *this = mtd->priv;
  	int start, end, block, value, status;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1905
1906
  	start = onenand_block(this, ofs);
  	end = onenand_block(this, ofs + len);
916527f48   Kyungmin Park   OneNAND support (...
1907
1908
  
  	/* Continuous lock scheme */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1909
  	if (this->options & ONENAND_HAS_CONT_LOCK) {
916527f48   Kyungmin Park   OneNAND support (...
1910
1911
1912
1913
1914
1915
1916
  		/* Set start block address */
  		this->write_word(start,
  				 this->base + ONENAND_REG_START_BLOCK_ADDRESS);
  		/* Set end block address */
  		this->write_word(end - 1,
  				 this->base + ONENAND_REG_END_BLOCK_ADDRESS);
  		/* Write unlock command */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1917
  		this->command(mtd, cmd, 0, 0);
916527f48   Kyungmin Park   OneNAND support (...
1918
1919
1920
1921
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
1933
1934
1935
1936
  
  		/* There's no return value */
  		this->wait(mtd, FL_UNLOCKING);
  
  		/* Sanity check */
  		while (this->read_word(this->base + ONENAND_REG_CTRL_STATUS)
  		       & ONENAND_CTRL_ONGO)
  			continue;
  
  		/* Check lock status */
  		status = this->read_word(this->base + ONENAND_REG_WP_STATUS);
  		if (!(status & ONENAND_WP_US))
  			printk(KERN_ERR "wp status = 0x%x
  ", status);
  
  		return 0;
  	}
  
  	/* Block lock scheme */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
1937
  	for (block = start; block < end; block++) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
1938
1939
1940
1941
1942
1943
  		/* Set block address */
  		value = onenand_block_address(this, block);
  		this->write_word(value, this->base + ONENAND_REG_START_ADDRESS1);
  		/* Select DataRAM for DDP */
  		value = onenand_bufferram_address(this, block);
  		this->write_word(value, this->base + ONENAND_REG_START_ADDRESS2);
916527f48   Kyungmin Park   OneNAND support (...
1944
1945
1946
1947
1948
1949
1950
1951
1952
1953
1954
1955
1956
  		/* Set start block address */
  		this->write_word(block,
  				 this->base + ONENAND_REG_START_BLOCK_ADDRESS);
  		/* Write unlock command */
  		this->command(mtd, ONENAND_CMD_UNLOCK, 0, 0);
  
  		/* There's no return value */
  		this->wait(mtd, FL_UNLOCKING);
  
  		/* Sanity check */
  		while (this->read_word(this->base + ONENAND_REG_CTRL_STATUS)
  		       & ONENAND_CTRL_ONGO)
  			continue;
916527f48   Kyungmin Park   OneNAND support (...
1957
1958
1959
1960
1961
1962
1963
1964
1965
1966
  		/* Check lock status */
  		status = this->read_word(this->base + ONENAND_REG_WP_STATUS);
  		if (!(status & ONENAND_WP_US))
  			printk(KERN_ERR "block = %d, wp status = 0x%x
  ",
  			       block, status);
  	}
  
  	return 0;
  }
4fca3310d   Stefan Roese   OneNAND: Fix comp...
1967
  #ifdef ONENAND_LINUX
916527f48   Kyungmin Park   OneNAND support (...
1968
  /**
ef0921d6b   Kyungmin Park   Sync with 2.6.27
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
2002
   * onenand_lock - [MTD Interface] Lock block(s)
   * @param mtd           MTD device structure
   * @param ofs           offset relative to mtd start
   * @param len           number of bytes to unlock
   *
   * Lock one or more blocks
   */
  static int onenand_lock(struct mtd_info *mtd, loff_t ofs, size_t len)
  {
  	int ret;
  
  	onenand_get_device(mtd, FL_LOCKING);
  	ret = onenand_do_lock_cmd(mtd, ofs, len, ONENAND_CMD_LOCK);
  	onenand_release_device(mtd);
  	return ret;
  }
  
  /**
   * onenand_unlock - [MTD Interface] Unlock block(s)
   * @param mtd           MTD device structure
   * @param ofs           offset relative to mtd start
   * @param len           number of bytes to unlock
   *
   * Unlock one or more blocks
   */
  static int onenand_unlock(struct mtd_info *mtd, loff_t ofs, size_t len)
  {
  	int ret;
  
  	onenand_get_device(mtd, FL_LOCKING);
  	ret = onenand_do_lock_cmd(mtd, ofs, len, ONENAND_CMD_UNLOCK);
  	onenand_release_device(mtd);
  	return ret;
  }
4fca3310d   Stefan Roese   OneNAND: Fix comp...
2003
  #endif
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
2028
2029
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
2044
2045
2046
2047
2048
  
  /**
   * onenand_check_lock_status - [OneNAND Interface] Check lock status
   * @param this          onenand chip data structure
   *
   * Check lock status
   */
  static int onenand_check_lock_status(struct onenand_chip *this)
  {
  	unsigned int value, block, status;
  	unsigned int end;
  
  	end = this->chipsize >> this->erase_shift;
  	for (block = 0; block < end; block++) {
  		/* Set block address */
  		value = onenand_block_address(this, block);
  		this->write_word(value, this->base + ONENAND_REG_START_ADDRESS1);
  		/* Select DataRAM for DDP */
  		value = onenand_bufferram_address(this, block);
  		this->write_word(value, this->base + ONENAND_REG_START_ADDRESS2);
  		/* Set start block address */
  		this->write_word(block, this->base + ONENAND_REG_START_BLOCK_ADDRESS);
  
  		/* Check lock status */
  		status = this->read_word(this->base + ONENAND_REG_WP_STATUS);
  		if (!(status & ONENAND_WP_US)) {
  			printk(KERN_ERR "block = %d, wp status = 0x%x
  ", block, status);
  			return 0;
  		}
  	}
  
  	return 1;
  }
  
  /**
   * onenand_unlock_all - [OneNAND Interface] unlock all blocks
   * @param mtd           MTD device structure
   *
   * Unlock all blocks
   */
  static void onenand_unlock_all(struct mtd_info *mtd)
  {
  	struct onenand_chip *this = mtd->priv;
  	loff_t ofs = 0;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2049
  	size_t len = mtd->size;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
2061
2062
2063
  
  	if (this->options & ONENAND_HAS_UNLOCK_ALL) {
  		/* Set start block address */
  		this->write_word(0, this->base + ONENAND_REG_START_BLOCK_ADDRESS);
  		/* Write unlock command */
  		this->command(mtd, ONENAND_CMD_UNLOCK_ALL, 0, 0);
  
  		/* There's no return value */
  		this->wait(mtd, FL_LOCKING);
  
  		/* Sanity check */
  		while (this->read_word(this->base + ONENAND_REG_CTRL_STATUS)
  				& ONENAND_CTRL_ONGO)
  			continue;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2064
2065
2066
2067
2068
  		/* Check lock status */
  		if (onenand_check_lock_status(this))
  			return;
  
  		/* Workaround for all block unlock in DDP */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2069
  		if (ONENAND_IS_DDP(this) && !FLEXONENAND(this)) {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
2093
2094
2095
2096
2097
2098
2099
  			/* All blocks on another chip */
  			ofs = this->chipsize >> 1;
  			len = this->chipsize >> 1;
  		}
  	}
  
  	onenand_do_lock_cmd(mtd, ofs, len, ONENAND_CMD_UNLOCK);
  }
  
  
  /**
   * onenand_check_features - Check and set OneNAND features
   * @param mtd           MTD data structure
   *
   * Check and set OneNAND features
   * - lock scheme
   * - two plane
   */
  static void onenand_check_features(struct mtd_info *mtd)
  {
  	struct onenand_chip *this = mtd->priv;
  	unsigned int density, process;
  
  	/* Lock scheme depends on density and process */
  	density = onenand_get_density(this->device_id);
  	process = this->version_id >> ONENAND_VERSION_PROCESS_SHIFT;
  
  	/* Lock scheme */
  	switch (density) {
  	case ONENAND_DEVICE_DENSITY_4Gb:
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
2100
2101
2102
2103
  		if (ONENAND_IS_DDP(this))
  			this->options |= ONENAND_HAS_2PLANE;
  		else
  			this->options |= ONENAND_HAS_4KB_PAGE;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2104
2105
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
2120
2121
2122
  
  	case ONENAND_DEVICE_DENSITY_2Gb:
  		/* 2Gb DDP don't have 2 plane */
  		if (!ONENAND_IS_DDP(this))
  			this->options |= ONENAND_HAS_2PLANE;
  		this->options |= ONENAND_HAS_UNLOCK_ALL;
  
  	case ONENAND_DEVICE_DENSITY_1Gb:
  		/* A-Die has all block unlock */
  		if (process)
  			this->options |= ONENAND_HAS_UNLOCK_ALL;
  		break;
  
  	default:
  		/* Some OneNAND has continuous lock scheme */
  		if (!process)
  			this->options |= ONENAND_HAS_CONT_LOCK;
  		break;
  	}
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2123
  	if (ONENAND_IS_MLC(this))
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
2124
2125
2126
  		this->options |= ONENAND_HAS_4KB_PAGE;
  
  	if (ONENAND_IS_4KB_PAGE(this))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2127
2128
2129
2130
2131
2132
  		this->options &= ~ONENAND_HAS_2PLANE;
  
  	if (FLEXONENAND(this)) {
  		this->options &= ~ONENAND_HAS_CONT_LOCK;
  		this->options |= ONENAND_HAS_UNLOCK_ALL;
  	}
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2133
2134
2135
2136
2137
2138
2139
2140
2141
  	if (this->options & ONENAND_HAS_CONT_LOCK)
  		printk(KERN_DEBUG "Lock scheme is Continuous Lock
  ");
  	if (this->options & ONENAND_HAS_UNLOCK_ALL)
  		printk(KERN_DEBUG "Chip support all block unlock
  ");
  	if (this->options & ONENAND_HAS_2PLANE)
  		printk(KERN_DEBUG "Chip has 2 plane
  ");
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
2142
2143
2144
  	if (this->options & ONENAND_HAS_4KB_PAGE)
  		printk(KERN_DEBUG "Chip has 4KiB pagesize
  ");
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2145
2146
2147
  }
  
  /**
916527f48   Kyungmin Park   OneNAND support (...
2148
2149
2150
2151
2152
   * onenand_print_device_info - Print device ID
   * @param device        device ID
   *
   * Print device ID
   */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2153
  char *onenand_print_device_info(int device, int version)
916527f48   Kyungmin Park   OneNAND support (...
2154
  {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2155
  	int vcc, demuxed, ddp, density, flexonenand;
195ccfc59   Fathi BOUDRA   OneNAND: Fill in ...
2156
  	char *dev_info = malloc(80);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2157
  	char *p = dev_info;
916527f48   Kyungmin Park   OneNAND support (...
2158
2159
2160
2161
  
  	vcc = device & ONENAND_DEVICE_VCC_MASK;
  	demuxed = device & ONENAND_DEVICE_IS_DEMUX;
  	ddp = device & ONENAND_DEVICE_IS_DDP;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2162
2163
2164
  	density = onenand_get_density(device);
  	flexonenand = device & DEVICE_IS_FLEXONENAND;
  	p += sprintf(dev_info, "%s%sOneNAND%s %dMB %sV 16-bit (0x%02x)",
916527f48   Kyungmin Park   OneNAND support (...
2165
  	       demuxed ? "" : "Muxed ",
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2166
  	       flexonenand ? "Flex-" : "",
916527f48   Kyungmin Park   OneNAND support (...
2167
2168
  	       ddp ? "(DDP)" : "",
  	       (16 << density), vcc ? "2.65/3.3" : "1.8", device);
195ccfc59   Fathi BOUDRA   OneNAND: Fill in ...
2169

ef0921d6b   Kyungmin Park   Sync with 2.6.27
2170
2171
2172
2173
  	sprintf(p, "
  OneNAND version = 0x%04x", version);
  	printk("%s
  ", dev_info);
195ccfc59   Fathi BOUDRA   OneNAND: Fill in ...
2174
  	return dev_info;
916527f48   Kyungmin Park   OneNAND support (...
2175
2176
2177
  }
  
  static const struct onenand_manufacturers onenand_manuf_ids[] = {
456be17db   Enric Balletbo i Serra   mtd: OneNAND: add...
2178
  	{ONENAND_MFR_NUMONYX, "Numonyx"},
916527f48   Kyungmin Park   OneNAND support (...
2179
  	{ONENAND_MFR_SAMSUNG, "Samsung"},
916527f48   Kyungmin Park   OneNAND support (...
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
  };
  
  /**
   * onenand_check_maf - Check manufacturer ID
   * @param manuf         manufacturer ID
   *
   * Check manufacturer ID
   */
  static int onenand_check_maf(int manuf)
  {
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2190
  	int size = ARRAY_SIZE(onenand_manuf_ids);
916527f48   Kyungmin Park   OneNAND support (...
2191
  	int i;
24ccca5eb   Marek Vasut   GCC4.6: Squash wa...
2192
2193
2194
  #ifdef ONENAND_DEBUG
  	char *name;
  #endif
916527f48   Kyungmin Park   OneNAND support (...
2195

cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2196
  	for (i = 0; i < size; i++)
916527f48   Kyungmin Park   OneNAND support (...
2197
2198
  		if (manuf == onenand_manuf_ids[i].id)
  			break;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2199

24ccca5eb   Marek Vasut   GCC4.6: Squash wa...
2200
  #ifdef ONENAND_DEBUG
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2201
2202
2203
2204
  	if (i < size)
  		name = onenand_manuf_ids[i].name;
  	else
  		name = "Unknown";
916527f48   Kyungmin Park   OneNAND support (...
2205

ef0921d6b   Kyungmin Park   Sync with 2.6.27
2206
2207
  	printk(KERN_DEBUG "OneNAND Manufacturer: %s (0x%0x)
  ", name, manuf);
916527f48   Kyungmin Park   OneNAND support (...
2208
  #endif
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2209
  	return i == size;
916527f48   Kyungmin Park   OneNAND support (...
2210
2211
2212
  }
  
  /**
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2213
2214
2215
2216
2217
2218
2219
2220
2221
  * flexonenand_get_boundary	- Reads the SLC boundary
  * @param onenand_info		- onenand info structure
  *
  * Fill up boundary[] field in onenand_chip
  **/
  static int flexonenand_get_boundary(struct mtd_info *mtd)
  {
  	struct onenand_chip *this = mtd->priv;
  	unsigned int die, bdry;
24ccca5eb   Marek Vasut   GCC4.6: Squash wa...
2222
  	int syscfg, locked;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2223
2224
2225
2226
2227
2228
2229
2230
2231
2232
  
  	/* Disable ECC */
  	syscfg = this->read_word(this->base + ONENAND_REG_SYS_CFG1);
  	this->write_word((syscfg | 0x0100), this->base + ONENAND_REG_SYS_CFG1);
  
  	for (die = 0; die < this->dies; die++) {
  		this->command(mtd, FLEXONENAND_CMD_PI_ACCESS, die, 0);
  		this->wait(mtd, FL_SYNCING);
  
  		this->command(mtd, FLEXONENAND_CMD_READ_PI, die, 0);
24ccca5eb   Marek Vasut   GCC4.6: Squash wa...
2233
  		this->wait(mtd, FL_READING);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2234
2235
2236
2237
2238
2239
2240
2241
2242
  
  		bdry = this->read_word(this->base + ONENAND_DATARAM);
  		if ((bdry >> FLEXONENAND_PI_UNLOCK_SHIFT) == 3)
  			locked = 0;
  		else
  			locked = 1;
  		this->boundary[die] = bdry & FLEXONENAND_PI_MASK;
  
  		this->command(mtd, ONENAND_CMD_RESET, 0, 0);
24ccca5eb   Marek Vasut   GCC4.6: Squash wa...
2243
  		this->wait(mtd, FL_RESETING);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2244
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
2257
2258
2259
2260
2261
2262
2263
2264
2265
2266
2267
2268
2269
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
2305
2306
2307
2308
2309
2310
2311
2312
2313
2314
2315
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
2331
2332
2333
2334
2335
2336
2337
2338
2339
2340
2341
2342
2343
2344
2345
2346
2347
2348
2349
2350
  
  		printk(KERN_INFO "Die %d boundary: %d%s
  ", die,
  		       this->boundary[die], locked ? "(Locked)" : "(Unlocked)");
  	}
  
  	/* Enable ECC */
  	this->write_word(syscfg, this->base + ONENAND_REG_SYS_CFG1);
  	return 0;
  }
  
  /**
   * flexonenand_get_size - Fill up fields in onenand_chip and mtd_info
   * 			  boundary[], diesize[], mtd->size, mtd->erasesize,
   * 			  mtd->eraseregions
   * @param mtd		- MTD device structure
   */
  static void flexonenand_get_size(struct mtd_info *mtd)
  {
  	struct onenand_chip *this = mtd->priv;
  	int die, i, eraseshift, density;
  	int blksperdie, maxbdry;
  	loff_t ofs;
  
  	density = onenand_get_density(this->device_id);
  	blksperdie = ((loff_t)(16 << density) << 20) >> (this->erase_shift);
  	blksperdie >>= ONENAND_IS_DDP(this) ? 1 : 0;
  	maxbdry = blksperdie - 1;
  	eraseshift = this->erase_shift - 1;
  
  	mtd->numeraseregions = this->dies << 1;
  
  	/* This fills up the device boundary */
  	flexonenand_get_boundary(mtd);
  	die = 0;
  	ofs = 0;
  	i = -1;
  	for (; die < this->dies; die++) {
  		if (!die || this->boundary[die-1] != maxbdry) {
  			i++;
  			mtd->eraseregions[i].offset = ofs;
  			mtd->eraseregions[i].erasesize = 1 << eraseshift;
  			mtd->eraseregions[i].numblocks =
  							this->boundary[die] + 1;
  			ofs += mtd->eraseregions[i].numblocks << eraseshift;
  			eraseshift++;
  		} else {
  			mtd->numeraseregions -= 1;
  			mtd->eraseregions[i].numblocks +=
  							this->boundary[die] + 1;
  			ofs += (this->boundary[die] + 1) << (eraseshift - 1);
  		}
  		if (this->boundary[die] != maxbdry) {
  			i++;
  			mtd->eraseregions[i].offset = ofs;
  			mtd->eraseregions[i].erasesize = 1 << eraseshift;
  			mtd->eraseregions[i].numblocks = maxbdry ^
  							 this->boundary[die];
  			ofs += mtd->eraseregions[i].numblocks << eraseshift;
  			eraseshift--;
  		} else
  			mtd->numeraseregions -= 1;
  	}
  
  	/* Expose MLC erase size except when all blocks are SLC */
  	mtd->erasesize = 1 << this->erase_shift;
  	if (mtd->numeraseregions == 1)
  		mtd->erasesize >>= 1;
  
  	printk(KERN_INFO "Device has %d eraseregions
  ", mtd->numeraseregions);
  	for (i = 0; i < mtd->numeraseregions; i++)
  		printk(KERN_INFO "[offset: 0x%08llx, erasesize: 0x%05x,"
  			" numblocks: %04u]
  ", mtd->eraseregions[i].offset,
  			mtd->eraseregions[i].erasesize,
  			mtd->eraseregions[i].numblocks);
  
  	for (die = 0, mtd->size = 0; die < this->dies; die++) {
  		this->diesize[die] = (loff_t) (blksperdie << this->erase_shift);
  		this->diesize[die] -= (loff_t) (this->boundary[die] + 1)
  						 << (this->erase_shift - 1);
  		mtd->size += this->diesize[die];
  	}
  }
  
  /**
   * flexonenand_check_blocks_erased - Check if blocks are erased
   * @param mtd_info	- mtd info structure
   * @param start		- first erase block to check
   * @param end		- last erase block to check
   *
   * Converting an unerased block from MLC to SLC
   * causes byte values to change. Since both data and its ECC
   * have changed, reads on the block give uncorrectable error.
   * This might lead to the block being detected as bad.
   *
   * Avoid this by ensuring that the block to be converted is
   * erased.
   */
  static int flexonenand_check_blocks_erased(struct mtd_info *mtd,
  					int start, int end)
  {
  	struct onenand_chip *this = mtd->priv;
  	int i, ret;
  	int block;
  	struct mtd_oob_ops ops = {
dfe64e2c8   Sergey Lapin   mtd: resync with ...
2351
  		.mode = MTD_OPS_PLACE_OOB,
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2352
2353
2354
2355
2356
2357
2358
2359
2360
2361
2362
2363
2364
2365
2366
2367
2368
2369
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
2386
2387
2388
2389
2390
2391
2392
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
2414
2415
2416
2417
2418
2419
2420
2421
2422
2423
2424
2425
2426
2427
2428
2429
2430
2431
2432
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
2445
2446
2447
2448
2449
2450
2451
2452
2453
2454
2455
2456
2457
2458
2459
2460
2461
2462
2463
2464
2465
2466
2467
2468
2469
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2481
2482
  		.ooboffs = 0,
  		.ooblen	= mtd->oobsize,
  		.datbuf	= NULL,
  		.oobbuf	= this->oob_buf,
  	};
  	loff_t addr;
  
  	printk(KERN_DEBUG "Check blocks from %d to %d
  ", start, end);
  
  	for (block = start; block <= end; block++) {
  		addr = flexonenand_addr(this, block);
  		if (onenand_block_isbad_nolock(mtd, addr, 0))
  			continue;
  
  		/*
  		 * Since main area write results in ECC write to spare,
  		 * it is sufficient to check only ECC bytes for change.
  		 */
  		ret = onenand_read_oob_nolock(mtd, addr, &ops);
  		if (ret)
  			return ret;
  
  		for (i = 0; i < mtd->oobsize; i++)
  			if (this->oob_buf[i] != 0xff)
  				break;
  
  		if (i != mtd->oobsize) {
  			printk(KERN_WARNING "Block %d not erased.
  ", block);
  			return 1;
  		}
  	}
  
  	return 0;
  }
  
  /**
   * flexonenand_set_boundary	- Writes the SLC boundary
   * @param mtd			- mtd info structure
   */
  int flexonenand_set_boundary(struct mtd_info *mtd, int die,
  				    int boundary, int lock)
  {
  	struct onenand_chip *this = mtd->priv;
  	int ret, density, blksperdie, old, new, thisboundary;
  	loff_t addr;
  
  	if (die >= this->dies)
  		return -EINVAL;
  
  	if (boundary == this->boundary[die])
  		return 0;
  
  	density = onenand_get_density(this->device_id);
  	blksperdie = ((16 << density) << 20) >> this->erase_shift;
  	blksperdie >>= ONENAND_IS_DDP(this) ? 1 : 0;
  
  	if (boundary >= blksperdie) {
  		printk("flexonenand_set_boundary:"
  			"Invalid boundary value. "
  			"Boundary not changed.
  ");
  		return -EINVAL;
  	}
  
  	/* Check if converting blocks are erased */
  	old = this->boundary[die] + (die * this->density_mask);
  	new = boundary + (die * this->density_mask);
  	ret = flexonenand_check_blocks_erased(mtd, min(old, new)
  						+ 1, max(old, new));
  	if (ret) {
  		printk(KERN_ERR "flexonenand_set_boundary: Please erase blocks before boundary change
  ");
  		return ret;
  	}
  
  	this->command(mtd, FLEXONENAND_CMD_PI_ACCESS, die, 0);
  	this->wait(mtd, FL_SYNCING);
  
  	/* Check is boundary is locked */
  	this->command(mtd, FLEXONENAND_CMD_READ_PI, die, 0);
  	ret = this->wait(mtd, FL_READING);
  
  	thisboundary = this->read_word(this->base + ONENAND_DATARAM);
  	if ((thisboundary >> FLEXONENAND_PI_UNLOCK_SHIFT) != 3) {
  		printk(KERN_ERR "flexonenand_set_boundary: boundary locked
  ");
  		goto out;
  	}
  
  	printk(KERN_INFO "flexonenand_set_boundary: Changing die %d boundary: %d%s
  ",
  			die, boundary, lock ? "(Locked)" : "(Unlocked)");
  
  	boundary &= FLEXONENAND_PI_MASK;
  	boundary |= lock ? 0 : (3 << FLEXONENAND_PI_UNLOCK_SHIFT);
  
  	addr = die ? this->diesize[0] : 0;
  	this->command(mtd, ONENAND_CMD_ERASE, addr, 0);
  	ret = this->wait(mtd, FL_ERASING);
  	if (ret) {
  		printk("flexonenand_set_boundary:"
  			"Failed PI erase for Die %d
  ", die);
  		goto out;
  	}
  
  	this->write_word(boundary, this->base + ONENAND_DATARAM);
  	this->command(mtd, ONENAND_CMD_PROG, addr, 0);
  	ret = this->wait(mtd, FL_WRITING);
  	if (ret) {
  		printk("flexonenand_set_boundary:"
  			"Failed PI write for Die %d
  ", die);
  		goto out;
  	}
  
  	this->command(mtd, FLEXONENAND_CMD_PI_UPDATE, die, 0);
  	ret = this->wait(mtd, FL_WRITING);
  out:
  	this->write_word(ONENAND_CMD_RESET, this->base + ONENAND_REG_COMMAND);
  	this->wait(mtd, FL_RESETING);
  	if (!ret)
  		/* Recalculate device size on boundary change*/
  		flexonenand_get_size(mtd);
  
  	return ret;
  }
  
  /**
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2483
   * onenand_chip_probe - [OneNAND Interface] Probe the OneNAND chip
916527f48   Kyungmin Park   OneNAND support (...
2484
2485
2486
2487
2488
   * @param mtd		MTD device structure
   *
   * OneNAND detection method:
   *   Compare the the values from command with ones from register
   */
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2489
  static int onenand_chip_probe(struct mtd_info *mtd)
916527f48   Kyungmin Park   OneNAND support (...
2490
2491
  {
  	struct onenand_chip *this = mtd->priv;
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2492
  	int bram_maf_id, bram_dev_id, maf_id, dev_id;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2493
2494
2495
2496
  	int syscfg;
  
  	/* Save system configuration 1 */
  	syscfg = this->read_word(this->base + ONENAND_REG_SYS_CFG1);
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2497

ef0921d6b   Kyungmin Park   Sync with 2.6.27
2498
  	/* Clear Sync. Burst Read mode to read BootRAM */
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2499
2500
  	this->write_word((syscfg & ~ONENAND_SYS_CFG1_SYNC_READ),
  			 this->base + ONENAND_REG_SYS_CFG1);
916527f48   Kyungmin Park   OneNAND support (...
2501
2502
2503
2504
2505
2506
2507
  
  	/* Send the command for reading device ID from BootRAM */
  	this->write_word(ONENAND_CMD_READID, this->base + ONENAND_BOOTRAM);
  
  	/* Read manufacturer and device IDs from BootRAM */
  	bram_maf_id = this->read_word(this->base + ONENAND_BOOTRAM + 0x0);
  	bram_dev_id = this->read_word(this->base + ONENAND_BOOTRAM + 0x2);
916527f48   Kyungmin Park   OneNAND support (...
2508
2509
  	/* Reset OneNAND to read default register values */
  	this->write_word(ONENAND_CMD_RESET, this->base + ONENAND_BOOTRAM);
d438d5084   Kyungmin Park   Fix OneNAND build...
2510
2511
  	/* Wait reset */
  	this->wait(mtd, FL_RESETING);
916527f48   Kyungmin Park   OneNAND support (...
2512

ef0921d6b   Kyungmin Park   Sync with 2.6.27
2513
2514
2515
2516
2517
2518
  	/* Restore system configuration 1 */
  	this->write_word(syscfg, this->base + ONENAND_REG_SYS_CFG1);
  
  	/* Check manufacturer ID */
  	if (onenand_check_maf(bram_maf_id))
  		return -ENXIO;
916527f48   Kyungmin Park   OneNAND support (...
2519
2520
2521
2522
2523
2524
2525
  	/* Read manufacturer and device IDs from Register */
  	maf_id = this->read_word(this->base + ONENAND_REG_MANUFACTURER_ID);
  	dev_id = this->read_word(this->base + ONENAND_REG_DEVICE_ID);
  
  	/* Check OneNAND device */
  	if (maf_id != bram_maf_id || dev_id != bram_dev_id)
  		return -ENXIO;
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2526
2527
2528
2529
2530
2531
2532
2533
2534
2535
2536
2537
2538
  	return 0;
  }
  
  /**
   * onenand_probe - [OneNAND Interface] Probe the OneNAND device
   * @param mtd		MTD device structure
   *
   * OneNAND detection method:
   *   Compare the the values from command with ones from register
   */
  int onenand_probe(struct mtd_info *mtd)
  {
  	struct onenand_chip *this = mtd->priv;
1432c763f   Wolfgang Denk   GCC4.6: Squash wa...
2539
  	int dev_id, ver_id;
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2540
2541
2542
2543
2544
2545
  	int density;
  	int ret;
  
  	ret = this->chip_probe(mtd);
  	if (ret)
  		return ret;
1432c763f   Wolfgang Denk   GCC4.6: Squash wa...
2546
  	/* Read device IDs from Register */
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2547
2548
2549
  	dev_id = this->read_word(this->base + ONENAND_REG_DEVICE_ID);
  	ver_id = this->read_word(this->base + ONENAND_REG_VERSION_ID);
  	this->technology = this->read_word(this->base + ONENAND_REG_TECHNOLOGY);
916527f48   Kyungmin Park   OneNAND support (...
2550
  	/* Flash device information */
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2551
  	mtd->name = onenand_print_device_info(dev_id, ver_id);
916527f48   Kyungmin Park   OneNAND support (...
2552
  	this->device_id = dev_id;
8cf11f3aa   Stefan Roese   OneNAND: Save ver...
2553
  	this->version_id = ver_id;
916527f48   Kyungmin Park   OneNAND support (...
2554

e26fd3d3b   Lukasz Majewski   onenand: Replace ...
2555
2556
  	/* Check OneNAND features */
  	onenand_check_features(mtd);
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2557
  	density = onenand_get_density(dev_id);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
2570
2571
  	if (FLEXONENAND(this)) {
  		this->dies = ONENAND_IS_DDP(this) ? 2 : 1;
  		/* Maximum possible erase regions */
  		mtd->numeraseregions = this->dies << 1;
  		mtd->eraseregions = malloc(sizeof(struct mtd_erase_region_info)
  					* (this->dies << 1));
  		if (!mtd->eraseregions)
  			return -ENOMEM;
  	}
  
  	/*
  	 * For Flex-OneNAND, chipsize represents maximum possible device size.
  	 * mtd->size represents the actual device size.
  	 */
916527f48   Kyungmin Park   OneNAND support (...
2572
2573
2574
2575
  	this->chipsize = (16 << density) << 20;
  
  	/* OneNAND page size & block size */
  	/* The data buffer size is equal to page size */
d438d5084   Kyungmin Park   Fix OneNAND build...
2576
  	mtd->writesize =
916527f48   Kyungmin Park   OneNAND support (...
2577
  	    this->read_word(this->base + ONENAND_REG_DATA_BUFFER_SIZE);
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2578
  	/* We use the full BufferRAM */
e26fd3d3b   Lukasz Majewski   onenand: Replace ...
2579
  	if (ONENAND_IS_4KB_PAGE(this))
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2580
  		mtd->writesize <<= 1;
d438d5084   Kyungmin Park   Fix OneNAND build...
2581
  	mtd->oobsize = mtd->writesize >> 5;
916527f48   Kyungmin Park   OneNAND support (...
2582
  	/* Pagers per block is always 64 in OneNAND */
d438d5084   Kyungmin Park   Fix OneNAND build...
2583
  	mtd->erasesize = mtd->writesize << 6;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2584
2585
2586
2587
2588
2589
2590
  	/*
  	 * Flex-OneNAND SLC area has 64 pages per block.
  	 * Flex-OneNAND MLC area has 128 pages per block.
  	 * Expose MLC erase size to find erase_shift and page_mask.
  	 */
  	if (FLEXONENAND(this))
  		mtd->erasesize <<= 1;
916527f48   Kyungmin Park   OneNAND support (...
2591
2592
  
  	this->erase_shift = ffs(mtd->erasesize) - 1;
d438d5084   Kyungmin Park   Fix OneNAND build...
2593
  	this->page_shift = ffs(mtd->writesize) - 1;
916527f48   Kyungmin Park   OneNAND support (...
2594
  	this->ppb_shift = (this->erase_shift - this->page_shift);
d438d5084   Kyungmin Park   Fix OneNAND build...
2595
  	this->page_mask = (mtd->erasesize / mtd->writesize) - 1;
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2596
2597
2598
  	/* Set density mask. it is used for DDP */
  	if (ONENAND_IS_DDP(this))
  		this->density_mask = this->chipsize >> (this->erase_shift + 1);
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
2599
2600
  	/* It's real page size */
  	this->writesize = mtd->writesize;
916527f48   Kyungmin Park   OneNAND support (...
2601
2602
  
  	/* REVIST: Multichip handling */
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2603
2604
2605
2606
  	if (FLEXONENAND(this))
  		flexonenand_get_size(mtd);
  	else
  		mtd->size = this->chipsize;
916527f48   Kyungmin Park   OneNAND support (...
2607

d438d5084   Kyungmin Park   Fix OneNAND build...
2608
  	mtd->flags = MTD_CAP_NANDFLASH;
dfe64e2c8   Sergey Lapin   mtd: resync with ...
2609
2610
2611
2612
2613
2614
2615
2616
  	mtd->_erase = onenand_erase;
  	mtd->_read = onenand_read;
  	mtd->_write = onenand_write;
  	mtd->_read_oob = onenand_read_oob;
  	mtd->_write_oob = onenand_write_oob;
  	mtd->_sync = onenand_sync;
  	mtd->_block_isbad = onenand_block_isbad;
  	mtd->_block_markbad = onenand_block_markbad;
195ccfc59   Fathi BOUDRA   OneNAND: Fill in ...
2617

916527f48   Kyungmin Park   OneNAND support (...
2618
2619
2620
2621
2622
2623
2624
2625
2626
2627
2628
2629
2630
2631
2632
  	return 0;
  }
  
  /**
   * onenand_scan - [OneNAND Interface] Scan for the OneNAND device
   * @param mtd		MTD device structure
   * @param maxchips	Number of chips to scan for
   *
   * This fills out all the not initialized function pointers
   * with the defaults.
   * The flash ID is read and the mtd/chip structures are
   * filled with the appropriate values.
   */
  int onenand_scan(struct mtd_info *mtd, int maxchips)
  {
1ae398620   Stefan Roese   OneNAND: Addition...
2633
  	int i;
916527f48   Kyungmin Park   OneNAND support (...
2634
2635
2636
2637
2638
2639
2640
2641
2642
2643
2644
  	struct onenand_chip *this = mtd->priv;
  
  	if (!this->read_word)
  		this->read_word = onenand_readw;
  	if (!this->write_word)
  		this->write_word = onenand_writew;
  
  	if (!this->command)
  		this->command = onenand_command;
  	if (!this->wait)
  		this->wait = onenand_wait;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2645
2646
  	if (!this->bbt_wait)
  		this->bbt_wait = onenand_bbt_wait;
916527f48   Kyungmin Park   OneNAND support (...
2647
2648
2649
2650
2651
  
  	if (!this->read_bufferram)
  		this->read_bufferram = onenand_read_bufferram;
  	if (!this->write_bufferram)
  		this->write_bufferram = onenand_write_bufferram;
6b3967bbb   Lukasz Majewski   onenand:samsung T...
2652
2653
  	if (!this->chip_probe)
  		this->chip_probe = onenand_chip_probe;
1714f51a2   Kyungmin Park   Add markbad function
2654
2655
  	if (!this->block_markbad)
  		this->block_markbad = onenand_default_block_markbad;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2656
2657
  	if (!this->scan_bbt)
  		this->scan_bbt = onenand_default_bbt;
916527f48   Kyungmin Park   OneNAND support (...
2658
2659
2660
2661
2662
2663
2664
2665
2666
  	if (onenand_probe(mtd))
  		return -ENXIO;
  
  	/* Set Sync. Burst Read after probing */
  	if (this->mmcontrol) {
  		printk(KERN_INFO "OneNAND Sync. Burst Read support
  ");
  		this->read_bufferram = onenand_sync_read_bufferram;
  	}
bfd7f3861   Kyungmin Park   Fix OneNAND read_...
2667
2668
2669
2670
2671
2672
2673
2674
2675
2676
2677
2678
2679
2680
2681
2682
2683
2684
2685
2686
2687
2688
2689
  	/* Allocate buffers, if necessary */
  	if (!this->page_buf) {
  		this->page_buf = kzalloc(mtd->writesize, GFP_KERNEL);
  		if (!this->page_buf) {
  			printk(KERN_ERR "onenand_scan(): Can't allocate page_buf
  ");
  			return -ENOMEM;
  		}
  		this->options |= ONENAND_PAGEBUF_ALLOC;
  	}
  	if (!this->oob_buf) {
  		this->oob_buf = kzalloc(mtd->oobsize, GFP_KERNEL);
  		if (!this->oob_buf) {
  			printk(KERN_ERR "onenand_scan: Can't allocate oob_buf
  ");
  			if (this->options & ONENAND_PAGEBUF_ALLOC) {
  				this->options &= ~ONENAND_PAGEBUF_ALLOC;
  				kfree(this->page_buf);
  			}
  			return -ENOMEM;
  		}
  		this->options |= ONENAND_OOBBUF_ALLOC;
  	}
1ae398620   Stefan Roese   OneNAND: Addition...
2690
2691
2692
2693
2694
2695
  	this->state = FL_READY;
  
  	/*
  	 * Allow subpage writes up to oobsize.
  	 */
  	switch (mtd->oobsize) {
cacbe9195   Amul Kumar Saha   Flex-OneNAND driv...
2696
2697
2698
2699
  	case 128:
  		this->ecclayout = &onenand_oob_128;
  		mtd->subpage_sft = 0;
  		break;
1ae398620   Stefan Roese   OneNAND: Addition...
2700
2701
2702
2703
2704
2705
2706
2707
2708
2709
2710
2711
2712
2713
2714
2715
2716
2717
2718
2719
2720
2721
2722
2723
2724
2725
2726
2727
2728
2729
2730
2731
2732
2733
  	case 64:
  		this->ecclayout = &onenand_oob_64;
  		mtd->subpage_sft = 2;
  		break;
  
  	case 32:
  		this->ecclayout = &onenand_oob_32;
  		mtd->subpage_sft = 1;
  		break;
  
  	default:
  		printk(KERN_WARNING "No OOB scheme defined for oobsize %d
  ",
  			mtd->oobsize);
  		mtd->subpage_sft = 0;
  		/* To prevent kernel oops */
  		this->ecclayout = &onenand_oob_32;
  		break;
  	}
  
  	this->subpagesize = mtd->writesize >> mtd->subpage_sft;
  
  	/*
  	 * The number of bytes available for a client to place data into
  	 * the out of band area
  	 */
  	this->ecclayout->oobavail = 0;
  	for (i = 0; i < MTD_MAX_OOBFREE_ENTRIES &&
  	    this->ecclayout->oobfree[i].length; i++)
  		this->ecclayout->oobavail +=
  			this->ecclayout->oobfree[i].length;
  	mtd->oobavail = this->ecclayout->oobavail;
  
  	mtd->ecclayout = this->ecclayout;
ef0921d6b   Kyungmin Park   Sync with 2.6.27
2734
2735
  	/* Unlock whole block */
  	onenand_unlock_all(mtd);
916527f48   Kyungmin Park   OneNAND support (...
2736

ef0921d6b   Kyungmin Park   Sync with 2.6.27
2737
  	return this->scan_bbt(mtd);
916527f48   Kyungmin Park   OneNAND support (...
2738
2739
2740
2741
2742
2743
2744
2745
2746
  }
  
  /**
   * onenand_release - [OneNAND Interface] Free resources held by the OneNAND device
   * @param mtd		MTD device structure
   */
  void onenand_release(struct mtd_info *mtd)
  {
  }