Blame view

drivers/ide/tx4939ide.c 17 KB
378979892   Atsushi Nemoto   ide: Add tx4939id...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
  /*
   * TX4939 internal IDE driver
   * Based on RBTX49xx patch from CELF patch archive.
   *
   * This file is subject to the terms and conditions of the GNU General Public
   * License.  See the file "COPYING" in the main directory of this archive
   * for more details.
   *
   * (C) Copyright TOSHIBA CORPORATION 2005-2007
   */
  
  #include <linux/module.h>
  #include <linux/types.h>
  #include <linux/ide.h>
  #include <linux/init.h>
  #include <linux/delay.h>
  #include <linux/platform_device.h>
  #include <linux/io.h>
  #include <linux/scatterlist.h>
15a453a95   Bartlomiej Zolnierkiewicz   ide: include <asm...
20
  #include <asm/ide.h>
378979892   Atsushi Nemoto   ide: Add tx4939id...
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
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
92
93
94
95
96
97
98
99
100
101
102
103
  #define MODNAME	"tx4939ide"
  
  /* ATA Shadow Registers (8-bit except for Data which is 16-bit) */
  #define TX4939IDE_Data			0x000
  #define TX4939IDE_Error_Feature		0x001
  #define TX4939IDE_Sec			0x002
  #define TX4939IDE_LBA0			0x003
  #define TX4939IDE_LBA1			0x004
  #define TX4939IDE_LBA2			0x005
  #define TX4939IDE_DevHead		0x006
  #define TX4939IDE_Stat_Cmd		0x007
  #define TX4939IDE_AltStat_DevCtl	0x402
  /* H/W DMA Registers  */
  #define TX4939IDE_DMA_Cmd	0x800	/* 8-bit */
  #define TX4939IDE_DMA_Stat	0x802	/* 8-bit */
  #define TX4939IDE_PRD_Ptr	0x804	/* 32-bit */
  /* ATA100 CORE Registers (16-bit) */
  #define TX4939IDE_Sys_Ctl	0xc00
  #define TX4939IDE_Xfer_Cnt_1	0xc08
  #define TX4939IDE_Xfer_Cnt_2	0xc0a
  #define TX4939IDE_Sec_Cnt	0xc10
  #define TX4939IDE_Start_Lo_Addr	0xc18
  #define TX4939IDE_Start_Up_Addr	0xc20
  #define TX4939IDE_Add_Ctl	0xc28
  #define TX4939IDE_Lo_Burst_Cnt	0xc30
  #define TX4939IDE_Up_Burst_Cnt	0xc38
  #define TX4939IDE_PIO_Addr	0xc88
  #define TX4939IDE_H_Rst_Tim	0xc90
  #define TX4939IDE_Int_Ctl	0xc98
  #define TX4939IDE_Pkt_Cmd	0xcb8
  #define TX4939IDE_Bxfer_Cnt_Hi	0xcc0
  #define TX4939IDE_Bxfer_Cnt_Lo	0xcc8
  #define TX4939IDE_Dev_TErr	0xcd0
  #define TX4939IDE_Pkt_Xfer_Ctl	0xcd8
  #define TX4939IDE_Start_TAddr	0xce0
  
  /* bits for Int_Ctl */
  #define TX4939IDE_INT_ADDRERR	0x80
  #define TX4939IDE_INT_REACHMUL	0x40
  #define TX4939IDE_INT_DEVTIMING	0x20
  #define TX4939IDE_INT_UDMATERM	0x10
  #define TX4939IDE_INT_TIMER	0x08
  #define TX4939IDE_INT_BUSERR	0x04
  #define TX4939IDE_INT_XFEREND	0x02
  #define TX4939IDE_INT_HOST	0x01
  
  #define TX4939IDE_IGNORE_INTS	\
  	(TX4939IDE_INT_ADDRERR | TX4939IDE_INT_REACHMUL | \
  	 TX4939IDE_INT_DEVTIMING | TX4939IDE_INT_UDMATERM | \
  	 TX4939IDE_INT_TIMER | TX4939IDE_INT_XFEREND)
  
  #ifdef __BIG_ENDIAN
  #define tx4939ide_swizzlel(a)	((a) ^ 4)
  #define tx4939ide_swizzlew(a)	((a) ^ 6)
  #define tx4939ide_swizzleb(a)	((a) ^ 7)
  #else
  #define tx4939ide_swizzlel(a)	(a)
  #define tx4939ide_swizzlew(a)	(a)
  #define tx4939ide_swizzleb(a)	(a)
  #endif
  
  static u16 tx4939ide_readw(void __iomem *base, u32 reg)
  {
  	return __raw_readw(base + tx4939ide_swizzlew(reg));
  }
  static u8 tx4939ide_readb(void __iomem *base, u32 reg)
  {
  	return __raw_readb(base + tx4939ide_swizzleb(reg));
  }
  static void tx4939ide_writel(u32 val, void __iomem *base, u32 reg)
  {
  	__raw_writel(val, base + tx4939ide_swizzlel(reg));
  }
  static void tx4939ide_writew(u16 val, void __iomem *base, u32 reg)
  {
  	__raw_writew(val, base + tx4939ide_swizzlew(reg));
  }
  static void tx4939ide_writeb(u8 val, void __iomem *base, u32 reg)
  {
  	__raw_writeb(val, base + tx4939ide_swizzleb(reg));
  }
  
  #define TX4939IDE_BASE(hwif)	((void __iomem *)(hwif)->extra_base)
e085b3cae   Bartlomiej Zolnierkiewicz   ide: change ->set...
104
  static void tx4939ide_set_pio_mode(ide_hwif_t *hwif, ide_drive_t *drive)
378979892   Atsushi Nemoto   ide: Add tx4939id...
105
  {
378979892   Atsushi Nemoto   ide: Add tx4939id...
106
107
  	int is_slave = drive->dn;
  	u32 mask, val;
e085b3cae   Bartlomiej Zolnierkiewicz   ide: change ->set...
108
  	const u8 pio = drive->pio_mode - XFER_PIO_0;
378979892   Atsushi Nemoto   ide: Add tx4939id...
109
110
111
112
113
  	u8 safe = pio;
  	ide_drive_t *pair;
  
  	pair = ide_get_pair_dev(drive);
  	if (pair)
cd078af65   Atsushi Nemoto   tx493xide: use mi...
114
  		safe = min_t(u8, safe, pair->pio_mode - XFER_PIO_0);
378979892   Atsushi Nemoto   ide: Add tx4939id...
115
116
117
118
119
120
121
122
123
  	/*
  	 * Update Command Transfer Mode for master/slave and Data
  	 * Transfer Mode for this drive.
  	 */
  	mask = is_slave ? 0x07f00000 : 0x000007f0;
  	val = ((safe << 8) | (pio << 4)) << (is_slave ? 16 : 0);
  	hwif->select_data = (hwif->select_data & ~mask) | val;
  	/* tx4939ide_tf_load_fixup() will set the Sys_Ctl register */
  }
8776168ca   Bartlomiej Zolnierkiewicz   ide: change ->set...
124
  static void tx4939ide_set_dma_mode(ide_hwif_t *hwif, ide_drive_t *drive)
378979892   Atsushi Nemoto   ide: Add tx4939id...
125
  {
378979892   Atsushi Nemoto   ide: Add tx4939id...
126
  	u32 mask, val;
8776168ca   Bartlomiej Zolnierkiewicz   ide: change ->set...
127
  	const u8 mode = drive->dma_mode;
378979892   Atsushi Nemoto   ide: Add tx4939id...
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
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
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
  
  	/* Update Data Transfer Mode for this drive. */
  	if (mode >= XFER_UDMA_0)
  		val = mode - XFER_UDMA_0 + 8;
  	else
  		val = mode - XFER_MW_DMA_0 + 5;
  	if (drive->dn) {
  		mask = 0x00f00000;
  		val <<= 20;
  	} else {
  		mask = 0x000000f0;
  		val <<= 4;
  	}
  	hwif->select_data = (hwif->select_data & ~mask) | val;
  	/* tx4939ide_tf_load_fixup() will set the Sys_Ctl register */
  }
  
  static u16 tx4939ide_check_error_ints(ide_hwif_t *hwif)
  {
  	void __iomem *base = TX4939IDE_BASE(hwif);
  	u16 ctl = tx4939ide_readw(base, TX4939IDE_Int_Ctl);
  
  	if (ctl & TX4939IDE_INT_BUSERR) {
  		/* reset FIFO */
  		u16 sysctl = tx4939ide_readw(base, TX4939IDE_Sys_Ctl);
  
  		tx4939ide_writew(sysctl | 0x4000, base, TX4939IDE_Sys_Ctl);
  		mmiowb();
  		/* wait 12GBUSCLK (typ. 60ns @ GBUS200MHz, max 270ns) */
  		ndelay(270);
  		tx4939ide_writew(sysctl, base, TX4939IDE_Sys_Ctl);
  	}
  	if (ctl & (TX4939IDE_INT_ADDRERR |
  		   TX4939IDE_INT_DEVTIMING | TX4939IDE_INT_BUSERR))
  		pr_err("%s: Error interrupt %#x (%s%s%s )
  ",
  		       hwif->name, ctl,
  		       ctl & TX4939IDE_INT_ADDRERR ? " Address-Error" : "",
  		       ctl & TX4939IDE_INT_DEVTIMING ? " DEV-Timing" : "",
  		       ctl & TX4939IDE_INT_BUSERR ? " Bus-Error" : "");
  	return ctl;
  }
  
  static void tx4939ide_clear_irq(ide_drive_t *drive)
  {
  	ide_hwif_t *hwif;
  	void __iomem *base;
  	u16 ctl;
  
  	/*
  	 * tx4939ide_dma_test_irq() and tx4939ide_dma_end() do all job
  	 * for DMA case.
  	 */
  	if (drive->waiting_for_dma)
  		return;
  	hwif = drive->hwif;
  	base = TX4939IDE_BASE(hwif);
  	ctl = tx4939ide_check_error_ints(hwif);
  	tx4939ide_writew(ctl, base, TX4939IDE_Int_Ctl);
  }
  
  static u8 tx4939ide_cable_detect(ide_hwif_t *hwif)
  {
  	void __iomem *base = TX4939IDE_BASE(hwif);
  
  	return tx4939ide_readw(base, TX4939IDE_Sys_Ctl) & 0x2000 ?
  		ATA_CBL_PATA40 : ATA_CBL_PATA80;
  }
  
  #ifdef __BIG_ENDIAN
  static void tx4939ide_dma_host_set(ide_drive_t *drive, int on)
  {
  	ide_hwif_t *hwif = drive->hwif;
  	u8 unit = drive->dn;
  	void __iomem *base = TX4939IDE_BASE(hwif);
  	u8 dma_stat = tx4939ide_readb(base, TX4939IDE_DMA_Stat);
  
  	if (on)
  		dma_stat |= (1 << (5 + unit));
  	else
  		dma_stat &= ~(1 << (5 + unit));
  
  	tx4939ide_writeb(dma_stat, base, TX4939IDE_DMA_Stat);
  }
  #else
  #define tx4939ide_dma_host_set	ide_dma_host_set
  #endif
  
  static u8 tx4939ide_clear_dma_status(void __iomem *base)
  {
  	u8 dma_stat;
  
  	/* read DMA status for INTR & ERROR flags */
  	dma_stat = tx4939ide_readb(base, TX4939IDE_DMA_Stat);
  	/* clear INTR & ERROR flags */
  	tx4939ide_writeb(dma_stat | ATA_DMA_INTR | ATA_DMA_ERR, base,
  			 TX4939IDE_DMA_Stat);
  	/* recover intmask cleared by writing to bit2 of DMA_Stat */
  	tx4939ide_writew(TX4939IDE_IGNORE_INTS << 8, base, TX4939IDE_Int_Ctl);
  	return dma_stat;
  }
  
  #ifdef __BIG_ENDIAN
  /* custom ide_build_dmatable to handle swapped layout */
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
232
  static int tx4939ide_build_dmatable(ide_drive_t *drive, struct ide_cmd *cmd)
378979892   Atsushi Nemoto   ide: Add tx4939id...
233
234
235
236
237
238
  {
  	ide_hwif_t *hwif = drive->hwif;
  	u32 *table = (u32 *)hwif->dmatable_cpu;
  	unsigned int count = 0;
  	int i;
  	struct scatterlist *sg;
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
239
  	for_each_sg(hwif->sg_table, sg, cmd->sg_nents, i) {
378979892   Atsushi Nemoto   ide: Add tx4939id...
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
  		u32 cur_addr, cur_len, bcount;
  
  		cur_addr = sg_dma_address(sg);
  		cur_len = sg_dma_len(sg);
  
  		/*
  		 * Fill in the DMA table, without crossing any 64kB boundaries.
  		 */
  
  		while (cur_len) {
  			if (count++ >= PRD_ENTRIES)
  				goto use_pio_instead;
  
  			bcount = 0x10000 - (cur_addr & 0xffff);
  			if (bcount > cur_len)
  				bcount = cur_len;
a0fce792b   Atsushi Nemoto   tx4939ide: Do not...
256
257
  			/*
  			 * This workaround for zero count seems required.
9711a5372   Atsushi Nemoto   tx4939ide: typo f...
258
  			 * (standard ide_build_dmatable does it too)
a0fce792b   Atsushi Nemoto   tx4939ide: Do not...
259
  			 */
9711a5372   Atsushi Nemoto   tx4939ide: typo f...
260
  			if (bcount == 0x10000)
a0fce792b   Atsushi Nemoto   tx4939ide: Do not...
261
  				bcount = 0x8000;
378979892   Atsushi Nemoto   ide: Add tx4939id...
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
  			*table++ = bcount & 0xffff;
  			*table++ = cur_addr;
  			cur_addr += bcount;
  			cur_len -= bcount;
  		}
  	}
  
  	if (count) {
  		*(table - 2) |= 0x80000000;
  		return count;
  	}
  
  use_pio_instead:
  	printk(KERN_ERR "%s: %s
  ", drive->name,
  		count ? "DMA table too small" : "empty DMA table?");
378979892   Atsushi Nemoto   ide: Add tx4939id...
278
279
280
281
282
  	return 0; /* revert to PIO for this request */
  }
  #else
  #define tx4939ide_build_dmatable	ide_build_dmatable
  #endif
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
283
  static int tx4939ide_dma_setup(ide_drive_t *drive, struct ide_cmd *cmd)
378979892   Atsushi Nemoto   ide: Add tx4939id...
284
285
286
  {
  	ide_hwif_t *hwif = drive->hwif;
  	void __iomem *base = TX4939IDE_BASE(hwif);
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
287
  	u8 rw = (cmd->tf_flags & IDE_TFLAG_WRITE) ? 0 : ATA_DMA_WR;
378979892   Atsushi Nemoto   ide: Add tx4939id...
288
289
  
  	/* fall back to PIO! */
11998b316   Bartlomiej Zolnierkiewicz   ide: move ide_map...
290
  	if (tx4939ide_build_dmatable(drive, cmd) == 0)
378979892   Atsushi Nemoto   ide: Add tx4939id...
291
  		return 1;
378979892   Atsushi Nemoto   ide: Add tx4939id...
292
293
294
295
296
  
  	/* PRD table */
  	tx4939ide_writel(hwif->dmatable_dma, base, TX4939IDE_PRD_Ptr);
  
  	/* specify r/w */
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
297
  	tx4939ide_writeb(rw, base, TX4939IDE_DMA_Cmd);
378979892   Atsushi Nemoto   ide: Add tx4939id...
298
299
300
  
  	/* clear INTR & ERROR flags */
  	tx4939ide_clear_dma_status(base);
378979892   Atsushi Nemoto   ide: Add tx4939id...
301
302
  	tx4939ide_writew(SECTOR_SIZE / 2, base, drive->dn ?
  			 TX4939IDE_Xfer_Cnt_2 : TX4939IDE_Xfer_Cnt_1);
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
303

9780e2dd8   Tejun Heo   ide: convert to r...
304
  	tx4939ide_writew(blk_rq_sectors(cmd->rq), base, TX4939IDE_Sec_Cnt);
229816941   Bartlomiej Zolnierkiewicz   ide: pass command...
305

378979892   Atsushi Nemoto   ide: Add tx4939id...
306
307
308
309
310
311
312
313
314
  	return 0;
  }
  
  static int tx4939ide_dma_end(ide_drive_t *drive)
  {
  	ide_hwif_t *hwif = drive->hwif;
  	u8 dma_stat, dma_cmd;
  	void __iomem *base = TX4939IDE_BASE(hwif);
  	u16 ctl = tx4939ide_readw(base, TX4939IDE_Int_Ctl);
378979892   Atsushi Nemoto   ide: Add tx4939id...
315
316
317
318
319
320
321
  	/* get DMA command mode */
  	dma_cmd = tx4939ide_readb(base, TX4939IDE_DMA_Cmd);
  	/* stop DMA */
  	tx4939ide_writeb(dma_cmd & ~ATA_DMA_START, base, TX4939IDE_DMA_Cmd);
  
  	/* read and clear the INTR & ERROR bits */
  	dma_stat = tx4939ide_clear_dma_status(base);
253275c52   Atsushi Nemoto   tx4939ide: remove...
322
  #define CHECK_DMA_MASK (ATA_DMA_ACTIVE | ATA_DMA_ERR | ATA_DMA_INTR)
378979892   Atsushi Nemoto   ide: Add tx4939id...
323

4453011f9   Bartlomiej Zolnierkiewicz   ide: destroy DMA ...
324
  	/* verify good DMA status */
253275c52   Atsushi Nemoto   tx4939ide: remove...
325
  	if ((dma_stat & CHECK_DMA_MASK) == 0 &&
378979892   Atsushi Nemoto   ide: Add tx4939id...
326
327
328
329
  	    (ctl & (TX4939IDE_INT_XFEREND | TX4939IDE_INT_HOST)) ==
  	    (TX4939IDE_INT_XFEREND | TX4939IDE_INT_HOST))
  		/* INT_IDE lost... bug? */
  		return 0;
253275c52   Atsushi Nemoto   tx4939ide: remove...
330
  	return ((dma_stat & CHECK_DMA_MASK) !=
378979892   Atsushi Nemoto   ide: Add tx4939id...
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
  		ATA_DMA_INTR) ? 0x10 | dma_stat : 0;
  }
  
  /* returns 1 if DMA IRQ issued, 0 otherwise */
  static int tx4939ide_dma_test_irq(ide_drive_t *drive)
  {
  	ide_hwif_t *hwif = drive->hwif;
  	void __iomem *base = TX4939IDE_BASE(hwif);
  	u16 ctl, ide_int;
  	u8 dma_stat, stat;
  	int found = 0;
  
  	ctl = tx4939ide_check_error_ints(hwif);
  	ide_int = ctl & (TX4939IDE_INT_XFEREND | TX4939IDE_INT_HOST);
  	switch (ide_int) {
  	case TX4939IDE_INT_HOST:
  		/* On error, XFEREND might not be asserted. */
  		stat = tx4939ide_readb(base, TX4939IDE_AltStat_DevCtl);
  		if ((stat & (ATA_BUSY | ATA_DRQ | ATA_ERR)) == ATA_ERR)
  			found = 1;
  		else
  			/* Wait for XFEREND (Mask HOST and unmask XFEREND) */
  			ctl &= ~TX4939IDE_INT_XFEREND << 8;
  		ctl |= ide_int << 8;
  		break;
  	case TX4939IDE_INT_HOST | TX4939IDE_INT_XFEREND:
  		dma_stat = tx4939ide_readb(base, TX4939IDE_DMA_Stat);
  		if (!(dma_stat & ATA_DMA_INTR))
  			pr_warning("%s: weird interrupt status. "
  				   "DMA_Stat %#02x int_ctl %#04x
  ",
  				   hwif->name, dma_stat, ctl);
  		found = 1;
  		break;
  	}
  	/*
  	 * Do not clear XFEREND, HOST now.  They will be cleared by
  	 * clearing bit2 of DMA_Stat.
  	 */
  	ctl &= ~ide_int;
  	tx4939ide_writew(ctl, base, TX4939IDE_Int_Ctl);
  	return found;
  }
592b53152   Sergei Shtylyov   ide: move read_sf...
374
375
376
377
378
379
380
381
382
383
  #ifdef __BIG_ENDIAN
  static u8 tx4939ide_dma_sff_read_status(ide_hwif_t *hwif)
  {
  	void __iomem *base = TX4939IDE_BASE(hwif);
  
  	return tx4939ide_readb(base, TX4939IDE_DMA_Stat);
  }
  #else
  #define tx4939ide_dma_sff_read_status ide_dma_sff_read_status
  #endif
378979892   Atsushi Nemoto   ide: Add tx4939id...
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
  static void tx4939ide_init_hwif(ide_hwif_t *hwif)
  {
  	void __iomem *base = TX4939IDE_BASE(hwif);
  
  	/* Soft Reset */
  	tx4939ide_writew(0x8000, base, TX4939IDE_Sys_Ctl);
  	mmiowb();
  	/* at least 20 GBUSCLK (typ. 100ns @ GBUS200MHz, max 450ns) */
  	ndelay(450);
  	tx4939ide_writew(0x0000, base, TX4939IDE_Sys_Ctl);
  	/* mask some interrupts and clear all interrupts */
  	tx4939ide_writew((TX4939IDE_IGNORE_INTS << 8) | 0xff, base,
  			 TX4939IDE_Int_Ctl);
  
  	tx4939ide_writew(0x0008, base, TX4939IDE_Lo_Burst_Cnt);
  	tx4939ide_writew(0, base, TX4939IDE_Up_Burst_Cnt);
  }
  
  static int tx4939ide_init_dma(ide_hwif_t *hwif, const struct ide_port_info *d)
  {
  	hwif->dma_base =
  		hwif->extra_base + tx4939ide_swizzleb(TX4939IDE_DMA_Cmd);
  	/*
  	 * Note that we cannot use ATA_DMA_TABLE_OFS, ATA_DMA_STATUS
  	 * for big endian.
  	 */
  	return ide_allocate_dma_engine(hwif);
  }
22aa4b32a   Bartlomiej Zolnierkiewicz   ide: remove ide_t...
412
  static void tx4939ide_tf_load_fixup(ide_drive_t *drive)
378979892   Atsushi Nemoto   ide: Add tx4939id...
413
414
415
416
417
418
419
420
421
  {
  	ide_hwif_t *hwif = drive->hwif;
  	void __iomem *base = TX4939IDE_BASE(hwif);
  	u16 sysctl = hwif->select_data >> (drive->dn ? 16 : 0);
  
  	/*
  	 * Fix ATA100 CORE System Control Register. (The write to the
  	 * Device/Head register may write wrong data to the System
  	 * Control Register)
abb596b25   Sergei Shtylyov   ide: turn selectp...
422
  	 * While Sys_Ctl is written here, dev_select() is not needed.
378979892   Atsushi Nemoto   ide: Add tx4939id...
423
424
425
  	 */
  	tx4939ide_writew(sysctl, base, TX4939IDE_Sys_Ctl);
  }
c9ff9e7b6   Sergei Shtylyov   ide: refactor tf_...
426
427
  static void tx4939ide_tf_load(ide_drive_t *drive, struct ide_taskfile *tf,
  			      u8 valid)
378979892   Atsushi Nemoto   ide: Add tx4939id...
428
  {
c9ff9e7b6   Sergei Shtylyov   ide: refactor tf_...
429
  	ide_tf_load(drive, tf, valid);
d68bab503   Atsushi Nemoto   tx493[89]ide: Rem...
430

c9ff9e7b6   Sergei Shtylyov   ide: refactor tf_...
431
  	if (valid & IDE_VALID_DEVICE)
22aa4b32a   Bartlomiej Zolnierkiewicz   ide: remove ide_t...
432
  		tx4939ide_tf_load_fixup(drive);
378979892   Atsushi Nemoto   ide: Add tx4939id...
433
  }
d68bab503   Atsushi Nemoto   tx493[89]ide: Rem...
434
  #ifdef __BIG_ENDIAN
378979892   Atsushi Nemoto   ide: Add tx4939id...
435

d68bab503   Atsushi Nemoto   tx493[89]ide: Rem...
436
  /* custom iops (independent from SWAP_IO_SPACE) */
aa24d9783   Atsushi Nemoto   tx4939ide: Fix tx...
437
  static void tx4939ide_input_data_swap(ide_drive_t *drive, struct ide_cmd *cmd,
378979892   Atsushi Nemoto   ide: Add tx4939id...
438
439
440
441
442
443
444
445
  				void *buf, unsigned int len)
  {
  	unsigned long port = drive->hwif->io_ports.data_addr;
  	unsigned short *ptr = buf;
  	unsigned int count = (len + 1) / 2;
  
  	while (count--)
  		*ptr++ = cpu_to_le16(__raw_readw((void __iomem *)port));
f26f6ceac   Atsushi Nemoto   tx493[89]ide: Fix...
446
  	__ide_flush_dcache_range((unsigned long)buf, roundup(len, 2));
378979892   Atsushi Nemoto   ide: Add tx4939id...
447
  }
aa24d9783   Atsushi Nemoto   tx4939ide: Fix tx...
448
  static void tx4939ide_output_data_swap(ide_drive_t *drive, struct ide_cmd *cmd,
378979892   Atsushi Nemoto   ide: Add tx4939id...
449
450
451
452
453
454
455
456
457
458
  				void *buf, unsigned int len)
  {
  	unsigned long port = drive->hwif->io_ports.data_addr;
  	unsigned short *ptr = buf;
  	unsigned int count = (len + 1) / 2;
  
  	while (count--) {
  		__raw_writew(le16_to_cpu(*ptr), (void __iomem *)port);
  		ptr++;
  	}
f26f6ceac   Atsushi Nemoto   tx493[89]ide: Fix...
459
  	__ide_flush_dcache_range((unsigned long)buf, roundup(len, 2));
378979892   Atsushi Nemoto   ide: Add tx4939id...
460
461
462
463
464
465
  }
  
  static const struct ide_tp_ops tx4939ide_tp_ops = {
  	.exec_command		= ide_exec_command,
  	.read_status		= ide_read_status,
  	.read_altstatus		= ide_read_altstatus,
ecf3a31d2   Sergei Shtylyov   ide: turn set_irq...
466
  	.write_devctl		= ide_write_devctl,
378979892   Atsushi Nemoto   ide: Add tx4939id...
467

abb596b25   Sergei Shtylyov   ide: turn selectp...
468
  	.dev_select		= ide_dev_select,
378979892   Atsushi Nemoto   ide: Add tx4939id...
469
  	.tf_load		= tx4939ide_tf_load,
d68bab503   Atsushi Nemoto   tx493[89]ide: Rem...
470
  	.tf_read		= ide_tf_read,
378979892   Atsushi Nemoto   ide: Add tx4939id...
471
472
473
474
475
476
  
  	.input_data		= tx4939ide_input_data_swap,
  	.output_data		= tx4939ide_output_data_swap,
  };
  
  #else	/* __LITTLE_ENDIAN */
378979892   Atsushi Nemoto   ide: Add tx4939id...
477
478
479
480
  static const struct ide_tp_ops tx4939ide_tp_ops = {
  	.exec_command		= ide_exec_command,
  	.read_status		= ide_read_status,
  	.read_altstatus		= ide_read_altstatus,
ecf3a31d2   Sergei Shtylyov   ide: turn set_irq...
481
  	.write_devctl		= ide_write_devctl,
378979892   Atsushi Nemoto   ide: Add tx4939id...
482

abb596b25   Sergei Shtylyov   ide: turn selectp...
483
  	.dev_select		= ide_dev_select,
378979892   Atsushi Nemoto   ide: Add tx4939id...
484
485
486
487
488
489
490
491
492
493
  	.tf_load		= tx4939ide_tf_load,
  	.tf_read		= ide_tf_read,
  
  	.input_data		= ide_input_data,
  	.output_data		= ide_output_data,
  };
  
  #endif	/* __LITTLE_ENDIAN */
  
  static const struct ide_port_ops tx4939ide_port_ops = {
3ee86dcdd   Bartlomiej Zolnierkiewicz   tx493x: fix inden...
494
495
496
497
  	.set_pio_mode		= tx4939ide_set_pio_mode,
  	.set_dma_mode		= tx4939ide_set_dma_mode,
  	.clear_irq		= tx4939ide_clear_irq,
  	.cable_detect		= tx4939ide_cable_detect,
378979892   Atsushi Nemoto   ide: Add tx4939id...
498
499
500
  };
  
  static const struct ide_dma_ops tx4939ide_dma_ops = {
3ee86dcdd   Bartlomiej Zolnierkiewicz   tx493x: fix inden...
501
502
  	.dma_host_set		= tx4939ide_dma_host_set,
  	.dma_setup		= tx4939ide_dma_setup,
3ee86dcdd   Bartlomiej Zolnierkiewicz   tx493x: fix inden...
503
504
505
506
  	.dma_start		= ide_dma_start,
  	.dma_end		= tx4939ide_dma_end,
  	.dma_test_irq		= tx4939ide_dma_test_irq,
  	.dma_lost_irq		= ide_dma_lost_irq,
22117d6ea   Bartlomiej Zolnierkiewicz   ide: add ->dma_ti...
507
  	.dma_timer_expiry	= ide_dma_sff_timer_expiry,
592b53152   Sergei Shtylyov   ide: move read_sf...
508
  	.dma_sff_read_status	= tx4939ide_dma_sff_read_status,
378979892   Atsushi Nemoto   ide: Add tx4939id...
509
  };
e6b53703b   Andi Kleen   sections: fix sec...
510
  static const struct ide_port_info tx4939ide_port_info __initconst = {
3ee86dcdd   Bartlomiej Zolnierkiewicz   tx493x: fix inden...
511
512
513
514
515
516
517
518
519
  	.init_hwif		= tx4939ide_init_hwif,
  	.init_dma		= tx4939ide_init_dma,
  	.port_ops		= &tx4939ide_port_ops,
  	.dma_ops		= &tx4939ide_dma_ops,
  	.tp_ops			= &tx4939ide_tp_ops,
  	.host_flags		= IDE_HFLAG_MMIO,
  	.pio_mask		= ATA_PIO4,
  	.mwdma_mask		= ATA_MWDMA2,
  	.udma_mask		= ATA_UDMA5,
b1d249e84   Bartlomiej Zolnierkiewicz   ide: remove chips...
520
  	.chipset		= ide_generic,
378979892   Atsushi Nemoto   ide: Add tx4939id...
521
522
523
524
  };
  
  static int __init tx4939ide_probe(struct platform_device *pdev)
  {
9f36d3143   Bartlomiej Zolnierkiewicz   ide: remove hw_re...
525
  	struct ide_hw hw, *hws[] = { &hw };
378979892   Atsushi Nemoto   ide: Add tx4939id...
526
527
528
529
530
531
532
533
534
535
536
537
538
  	struct ide_host *host;
  	struct resource *res;
  	int irq, ret;
  	unsigned long mapbase;
  
  	irq = platform_get_irq(pdev, 0);
  	if (irq < 0)
  		return -ENODEV;
  	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  	if (!res)
  		return -ENODEV;
  
  	if (!devm_request_mem_region(&pdev->dev, res->start,
28f65c11f   Joe Perches   treewide: Convert...
539
  				     resource_size(res), "tx4938ide"))
378979892   Atsushi Nemoto   ide: Add tx4939id...
540
541
  		return -EBUSY;
  	mapbase = (unsigned long)devm_ioremap(&pdev->dev, res->start,
28f65c11f   Joe Perches   treewide: Convert...
542
  					      resource_size(res));
378979892   Atsushi Nemoto   ide: Add tx4939id...
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
  	if (!mapbase)
  		return -EBUSY;
  	memset(&hw, 0, sizeof(hw));
  	hw.io_ports.data_addr =
  		mapbase + tx4939ide_swizzlew(TX4939IDE_Data);
  	hw.io_ports.error_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_Error_Feature);
  	hw.io_ports.nsect_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_Sec);
  	hw.io_ports.lbal_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_LBA0);
  	hw.io_ports.lbam_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_LBA1);
  	hw.io_ports.lbah_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_LBA2);
  	hw.io_ports.device_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_DevHead);
  	hw.io_ports.command_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_Stat_Cmd);
  	hw.io_ports.ctl_addr =
  		mapbase + tx4939ide_swizzleb(TX4939IDE_AltStat_DevCtl);
  	hw.irq = irq;
  	hw.dev = &pdev->dev;
  
  	pr_info("TX4939 IDE interface (base %#lx, irq %d)
  ", mapbase, irq);
dca398305   Bartlomiej Zolnierkiewicz   ide: pass number ...
569
  	host = ide_host_alloc(&tx4939ide_port_info, hws, 1);
378979892   Atsushi Nemoto   ide: Add tx4939id...
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
  	if (!host)
  		return -ENOMEM;
  	/* use extra_base for base address of the all registers */
  	host->ports[0]->extra_base = mapbase;
  	ret = ide_host_register(host, &tx4939ide_port_info, hws);
  	if (ret) {
  		ide_host_free(host);
  		return ret;
  	}
  	platform_set_drvdata(pdev, host);
  	return 0;
  }
  
  static int __exit tx4939ide_remove(struct platform_device *pdev)
  {
  	struct ide_host *host = platform_get_drvdata(pdev);
  
  	ide_host_remove(host);
  	return 0;
  }
  
  #ifdef CONFIG_PM
  static int tx4939ide_resume(struct platform_device *dev)
  {
  	struct ide_host *host = platform_get_drvdata(dev);
  	ide_hwif_t *hwif = host->ports[0];
  
  	tx4939ide_init_hwif(hwif);
  	return 0;
  }
  #else
  #define tx4939ide_resume	NULL
  #endif
  
  static struct platform_driver tx4939ide_driver = {
  	.driver = {
  		.name = MODNAME,
378979892   Atsushi Nemoto   ide: Add tx4939id...
607
608
609
610
  	},
  	.remove = __exit_p(tx4939ide_remove),
  	.resume = tx4939ide_resume,
  };
477c17dba   Jingoo Han   ide: tx4939ide: u...
611
  module_platform_driver_probe(tx4939ide_driver, tx4939ide_probe);
378979892   Atsushi Nemoto   ide: Add tx4939id...
612
613
614
615
  
  MODULE_DESCRIPTION("TX4939 internal IDE driver");
  MODULE_LICENSE("GPL");
  MODULE_ALIAS("platform:tx4939ide");