sata_sx4.c 39.6 KB
Newer Older
Linus Torvalds's avatar
Linus Torvalds committed
1 2 3 4 5 6 7 8 9
/*
 *  sata_sx4.c - Promise SATA
 *
 *  Maintained by:  Jeff Garzik <jgarzik@pobox.com>
 *  		    Please ALWAYS copy linux-ide@vger.kernel.org
 *		    on emails.
 *
 *  Copyright 2003-2004 Red Hat, Inc.
 *
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2, or (at your option)
 *  any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; see the file COPYING.  If not, write to
 *  the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
 *
 *
 *  libata documentation is available via 'make {ps|pdf}docs',
 *  as Documentation/DocBook/libata.*
 *
 *  Hardware documentation available under NDA.
Linus Torvalds's avatar
Linus Torvalds committed
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
/*
	Theory of operation
	-------------------

	The SX4 (PDC20621) chip features a single Host DMA (HDMA) copy
	engine, DIMM memory, and four ATA engines (one per SATA port).
	Data is copied to/from DIMM memory by the HDMA engine, before
	handing off to one (or more) of the ATA engines.  The ATA
	engines operate solely on DIMM memory.

	The SX4 behaves like a PATA chip, with no SATA controls or
	knowledge whatsoever, leading to the presumption that
	PATA<->SATA bridges exist on SX4 boards, external to the
	PDC20621 chip itself.

	The chip is quite capable, supporting an XOR engine and linked
	hardware commands (permits a string to transactions to be
	submitted and waited-on as a single unit), and an optional
	microprocessor.

	The limiting factor is largely software.  This Linux driver was
	written to multiplex the single HDMA engine to copy disk
	transactions into a fixed DIMM memory space, from where an ATA
	engine takes over.  As a result, each WRITE looks like this:

		submit HDMA packet to hardware
		hardware copies data from system memory to DIMM
		hardware raises interrupt

		submit ATA packet to hardware
		hardware executes ATA WRITE command, w/ data in DIMM
		hardware raises interrupt
65

66 67 68 69 70
	and each READ looks like this:

		submit ATA packet to hardware
		hardware executes ATA READ command, w/ data in DIMM
		hardware raises interrupt
71

72 73 74 75 76 77 78 79 80
		submit HDMA packet to hardware
		hardware copies data from DIMM to system memory
		hardware raises interrupt

	This is a very slow, lock-step way of doing things that can
	certainly be improved by motivated kernel hackers.

 */

Linus Torvalds's avatar
Linus Torvalds committed
81 82 83
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/pci.h>
84
#include <linux/slab.h>
Linus Torvalds's avatar
Linus Torvalds committed
85 86 87 88
#include <linux/init.h>
#include <linux/blkdev.h>
#include <linux/delay.h>
#include <linux/interrupt.h>
89
#include <linux/device.h>
Linus Torvalds's avatar
Linus Torvalds committed
90
#include <scsi/scsi_host.h>
91
#include <scsi/scsi_cmnd.h>
Linus Torvalds's avatar
Linus Torvalds committed
92 93 94 95
#include <linux/libata.h>
#include "sata_promise.h"

#define DRV_NAME	"sata_sx4"
Jeff Garzik's avatar
Jeff Garzik committed
96
#define DRV_VERSION	"0.12"
Linus Torvalds's avatar
Linus Torvalds committed
97 98 99


enum {
Tejun Heo's avatar
Tejun Heo committed
100 101 102
	PDC_MMIO_BAR		= 3,
	PDC_DIMM_BAR		= 4,

Linus Torvalds's avatar
Linus Torvalds committed
103 104 105 106 107 108 109
	PDC_PRD_TBL		= 0x44,	/* Direct command DMA table addr */

	PDC_PKT_SUBMIT		= 0x40, /* Command packet pointer addr */
	PDC_HDMA_PKT_SUBMIT	= 0x100, /* Host DMA packet pointer addr */
	PDC_INT_SEQMASK		= 0x40,	/* Mask of asserted SEQ INTs */
	PDC_HDMA_CTLSTAT	= 0x12C, /* Host DMA control / status */

110 111
	PDC_CTLSTAT		= 0x60,	/* IDEn control / status */

Linus Torvalds's avatar
Linus Torvalds committed
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
	PDC_20621_SEQCTL	= 0x400,
	PDC_20621_SEQMASK	= 0x480,
	PDC_20621_GENERAL_CTL	= 0x484,
	PDC_20621_PAGE_SIZE	= (32 * 1024),

	/* chosen, not constant, values; we design our own DIMM mem map */
	PDC_20621_DIMM_WINDOW	= 0x0C,	/* page# for 32K DIMM window */
	PDC_20621_DIMM_BASE	= 0x00200000,
	PDC_20621_DIMM_DATA	= (64 * 1024),
	PDC_DIMM_DATA_STEP	= (256 * 1024),
	PDC_DIMM_WINDOW_STEP	= (8 * 1024),
	PDC_DIMM_HOST_PRD	= (6 * 1024),
	PDC_DIMM_HOST_PKT	= (128 * 0),
	PDC_DIMM_HPKT_PRD	= (128 * 1),
	PDC_DIMM_ATA_PKT	= (128 * 2),
	PDC_DIMM_APKT_PRD	= (128 * 3),
	PDC_DIMM_HEADER_SZ	= PDC_DIMM_APKT_PRD + 128,
	PDC_PAGE_WINDOW		= 0x40,
	PDC_PAGE_DATA		= PDC_PAGE_WINDOW +
				  (PDC_20621_DIMM_DATA / PDC_20621_PAGE_SIZE),
	PDC_PAGE_SET		= PDC_DIMM_DATA_STEP / PDC_20621_PAGE_SIZE,

	PDC_CHIP0_OFS		= 0xC0000, /* offset of chip #0 */

	PDC_20621_ERR_MASK	= (1<<19) | (1<<20) | (1<<21) | (1<<22) |
				  (1<<23),

	board_20621		= 0,	/* FastTrak S150 SX4 */

141 142
	PDC_MASK_INT		= (1 << 10), /* HDMA/ATA mask int */
	PDC_RESET		= (1 << 11), /* HDMA/ATA reset */
143
	PDC_DMA_ENABLE		= (1 << 7),  /* DMA start/stop */
Linus Torvalds's avatar
Linus Torvalds committed
144 145 146 147

	PDC_MAX_HDMA		= 32,
	PDC_HDMA_Q_MASK		= (PDC_MAX_HDMA - 1),

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
	PDC_DIMM0_SPD_DEV_ADDRESS	= 0x50,
	PDC_DIMM1_SPD_DEV_ADDRESS	= 0x51,
	PDC_I2C_CONTROL			= 0x48,
	PDC_I2C_ADDR_DATA		= 0x4C,
	PDC_DIMM0_CONTROL		= 0x80,
	PDC_DIMM1_CONTROL		= 0x84,
	PDC_SDRAM_CONTROL		= 0x88,
	PDC_I2C_WRITE			= 0,		/* master -> slave */
	PDC_I2C_READ			= (1 << 6),	/* master <- slave */
	PDC_I2C_START			= (1 << 7),	/* start I2C proto */
	PDC_I2C_MASK_INT		= (1 << 5),	/* mask I2C interrupt */
	PDC_I2C_COMPLETE		= (1 << 16),	/* I2C normal compl. */
	PDC_I2C_NO_ACK			= (1 << 20),	/* slave no-ack addr */
	PDC_DIMM_SPD_SUBADDRESS_START	= 0x00,
	PDC_DIMM_SPD_SUBADDRESS_END	= 0x7F,
	PDC_DIMM_SPD_ROW_NUM		= 3,
	PDC_DIMM_SPD_COLUMN_NUM		= 4,
	PDC_DIMM_SPD_MODULE_ROW		= 5,
	PDC_DIMM_SPD_TYPE		= 11,
	PDC_DIMM_SPD_FRESH_RATE		= 12,
	PDC_DIMM_SPD_BANK_NUM		= 17,
	PDC_DIMM_SPD_CAS_LATENCY	= 18,
	PDC_DIMM_SPD_ATTRIBUTE		= 21,
	PDC_DIMM_SPD_ROW_PRE_CHARGE	= 27,
	PDC_DIMM_SPD_ROW_ACTIVE_DELAY	= 28,
	PDC_DIMM_SPD_RAS_CAS_DELAY	= 29,
	PDC_DIMM_SPD_ACTIVE_PRECHARGE	= 30,
	PDC_DIMM_SPD_SYSTEM_FREQ	= 126,
	PDC_CTL_STATUS			= 0x08,
	PDC_DIMM_WINDOW_CTLR		= 0x0C,
	PDC_TIME_CONTROL		= 0x3C,
	PDC_TIME_PERIOD			= 0x40,
	PDC_TIME_COUNTER		= 0x44,
	PDC_GENERAL_CTLR		= 0x484,
	PCI_PLL_INIT			= 0x8A531824,
	PCI_X_TCOUNT			= 0xEE1E5CFF,

	/* PDC_TIME_CONTROL bits */
	PDC_TIMER_BUZZER		= (1 << 10),
	PDC_TIMER_MODE_PERIODIC		= 0,		/* bits 9:8 == 00 */
	PDC_TIMER_MODE_ONCE		= (1 << 8),	/* bits 9:8 == 01 */
	PDC_TIMER_ENABLE		= (1 << 7),
	PDC_TIMER_MASK_INT		= (1 << 5),
	PDC_TIMER_SEQ_MASK		= 0x1f,		/* SEQ ID for timer */
	PDC_TIMER_DEFAULT		= PDC_TIMER_MODE_ONCE |
					  PDC_TIMER_ENABLE |
					  PDC_TIMER_MASK_INT,
Linus Torvalds's avatar
Linus Torvalds committed
195 196
};

197
#define ECC_ERASE_BUF_SZ (128 * 1024)
Linus Torvalds's avatar
Linus Torvalds committed
198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216

struct pdc_port_priv {
	u8			dimm_buf[(ATA_PRD_SZ * ATA_MAX_PRD) + 512];
	u8			*pkt;
	dma_addr_t		pkt_dma;
};

struct pdc_host_priv {
	unsigned int		doing_hdma;
	unsigned int		hdma_prod;
	unsigned int		hdma_cons;
	struct {
		struct ata_queued_cmd *qc;
		unsigned int	seq;
		unsigned long	pkt_ofs;
	} hdma[32];
};


217
static int pdc_sata_init_one(struct pci_dev *pdev, const struct pci_device_id *ent);
218 219 220
static void pdc_error_handler(struct ata_port *ap);
static void pdc_freeze(struct ata_port *ap);
static void pdc_thaw(struct ata_port *ap);
Linus Torvalds's avatar
Linus Torvalds committed
221 222
static int pdc_port_start(struct ata_port *ap);
static void pdc20621_qc_prep(struct ata_queued_cmd *qc);
223 224
static void pdc_tf_load_mmio(struct ata_port *ap, const struct ata_taskfile *tf);
static void pdc_exec_command_mmio(struct ata_port *ap, const struct ata_taskfile *tf);
225 226 227
static unsigned int pdc20621_dimm_init(struct ata_host *host);
static int pdc20621_detect_dimm(struct ata_host *host);
static unsigned int pdc20621_i2c_read(struct ata_host *host,
Linus Torvalds's avatar
Linus Torvalds committed
228
				      u32 device, u32 subaddr, u32 *pdata);
229 230
static int pdc20621_prog_dimm0(struct ata_host *host);
static unsigned int pdc20621_prog_dimm_global(struct ata_host *host);
Linus Torvalds's avatar
Linus Torvalds committed
231
#ifdef ATA_VERBOSE_DEBUG
232
static void pdc20621_get_from_dimm(struct ata_host *host,
Linus Torvalds's avatar
Linus Torvalds committed
233 234
				   void *psource, u32 offset, u32 size);
#endif
235
static void pdc20621_put_to_dimm(struct ata_host *host,
Linus Torvalds's avatar
Linus Torvalds committed
236 237
				 void *psource, u32 offset, u32 size);
static void pdc20621_irq_clear(struct ata_port *ap);
Tejun Heo's avatar
Tejun Heo committed
238
static unsigned int pdc20621_qc_issue(struct ata_queued_cmd *qc);
239 240 241 242
static int pdc_softreset(struct ata_link *link, unsigned int *class,
			 unsigned long deadline);
static void pdc_post_internal_cmd(struct ata_queued_cmd *qc);
static int pdc_check_atapi_dma(struct ata_queued_cmd *qc);
Linus Torvalds's avatar
Linus Torvalds committed
243 244


245
static struct scsi_host_template pdc_sata_sht = {
246
	ATA_BASE_SHT(DRV_NAME),
Linus Torvalds's avatar
Linus Torvalds committed
247 248 249 250
	.sg_tablesize		= LIBATA_MAX_PRD,
	.dma_boundary		= ATA_DMA_BOUNDARY,
};

251 252
/* TODO: inherit from base port_ops after converting to new EH */
static struct ata_port_operations pdc_20621_ops = {
253 254 255
	.inherits		= &ata_sff_port_ops,

	.check_atapi_dma	= pdc_check_atapi_dma,
Linus Torvalds's avatar
Linus Torvalds committed
256
	.qc_prep		= pdc20621_qc_prep,
Tejun Heo's avatar
Tejun Heo committed
257
	.qc_issue		= pdc20621_qc_issue,
258 259 260 261 262 263 264 265

	.freeze			= pdc_freeze,
	.thaw			= pdc_thaw,
	.softreset		= pdc_softreset,
	.error_handler		= pdc_error_handler,
	.lost_interrupt		= ATA_OP_NULL,
	.post_internal_cmd	= pdc_post_internal_cmd,

Linus Torvalds's avatar
Linus Torvalds committed
266
	.port_start		= pdc_port_start,
267 268 269 270

	.sff_tf_load		= pdc_tf_load_mmio,
	.sff_exec_command	= pdc_exec_command_mmio,
	.sff_irq_clear		= pdc20621_irq_clear,
Linus Torvalds's avatar
Linus Torvalds committed
271 272
};

273
static const struct ata_port_info pdc_port_info[] = {
Linus Torvalds's avatar
Linus Torvalds committed
274 275
	/* board_20621 */
	{
Jeff Garzik's avatar
Jeff Garzik committed
276
		.flags		= ATA_FLAG_SATA | ATA_FLAG_NO_LEGACY |
277
				  ATA_FLAG_SRST | ATA_FLAG_MMIO |
278
				  ATA_FLAG_NO_ATAPI | ATA_FLAG_PIO_POLLING,
279 280
		.pio_mask	= ATA_PIO4,
		.mwdma_mask	= ATA_MWDMA2,
281
		.udma_mask	= ATA_UDMA6,
Linus Torvalds's avatar
Linus Torvalds committed
282 283 284 285 286
		.port_ops	= &pdc_20621_ops,
	},

};

287
static const struct pci_device_id pdc_sata_pci_tbl[] = {
288 289
	{ PCI_VDEVICE(PROMISE, 0x6622), board_20621 },

Linus Torvalds's avatar
Linus Torvalds committed
290 291 292 293 294 295 296 297 298 299 300 301 302
	{ }	/* terminate list */
};

static struct pci_driver pdc_sata_pci_driver = {
	.name			= DRV_NAME,
	.id_table		= pdc_sata_pci_tbl,
	.probe			= pdc_sata_init_one,
	.remove			= ata_pci_remove_one,
};


static int pdc_port_start(struct ata_port *ap)
{
Jeff Garzik's avatar
Jeff Garzik committed
303
	struct device *dev = ap->host->dev;
Linus Torvalds's avatar
Linus Torvalds committed
304 305
	struct pdc_port_priv *pp;

306 307 308
	pp = devm_kzalloc(dev, sizeof(*pp), GFP_KERNEL);
	if (!pp)
		return -ENOMEM;
Linus Torvalds's avatar
Linus Torvalds committed
309

310 311 312
	pp->pkt = dmam_alloc_coherent(dev, 128, &pp->pkt_dma, GFP_KERNEL);
	if (!pp->pkt)
		return -ENOMEM;
Linus Torvalds's avatar
Linus Torvalds committed
313 314 315 316 317 318 319

	ap->private_data = pp;

	return 0;
}

static inline void pdc20621_ata_sg(struct ata_taskfile *tf, u8 *buf,
320
				   unsigned int portno,
Linus Torvalds's avatar
Linus Torvalds committed
321 322 323 324
					   unsigned int total_len)
{
	u32 addr;
	unsigned int dw = PDC_DIMM_APKT_PRD >> 2;
Al Viro's avatar
Al Viro committed
325
	__le32 *buf32 = (__le32 *) buf;
Linus Torvalds's avatar
Linus Torvalds committed
326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341

	/* output ATA packet S/G table */
	addr = PDC_20621_DIMM_BASE + PDC_20621_DIMM_DATA +
	       (PDC_DIMM_DATA_STEP * portno);
	VPRINTK("ATA sg addr 0x%x, %d\n", addr, addr);
	buf32[dw] = cpu_to_le32(addr);
	buf32[dw + 1] = cpu_to_le32(total_len | ATA_PRD_EOT);

	VPRINTK("ATA PSG @ %x == (0x%x, 0x%x)\n",
		PDC_20621_DIMM_BASE +
		       (PDC_DIMM_WINDOW_STEP * portno) +
		       PDC_DIMM_APKT_PRD,
		buf32[dw], buf32[dw + 1]);
}

static inline void pdc20621_host_sg(struct ata_taskfile *tf, u8 *buf,
342
				    unsigned int portno,
Linus Torvalds's avatar
Linus Torvalds committed
343 344 345 346
					    unsigned int total_len)
{
	u32 addr;
	unsigned int dw = PDC_DIMM_HPKT_PRD >> 2;
Al Viro's avatar
Al Viro committed
347
	__le32 *buf32 = (__le32 *) buf;
Linus Torvalds's avatar
Linus Torvalds committed
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367

	/* output Host DMA packet S/G table */
	addr = PDC_20621_DIMM_BASE + PDC_20621_DIMM_DATA +
	       (PDC_DIMM_DATA_STEP * portno);

	buf32[dw] = cpu_to_le32(addr);
	buf32[dw + 1] = cpu_to_le32(total_len | ATA_PRD_EOT);

	VPRINTK("HOST PSG @ %x == (0x%x, 0x%x)\n",
		PDC_20621_DIMM_BASE +
		       (PDC_DIMM_WINDOW_STEP * portno) +
		       PDC_DIMM_HPKT_PRD,
		buf32[dw], buf32[dw + 1]);
}

static inline unsigned int pdc20621_ata_pkt(struct ata_taskfile *tf,
					    unsigned int devno, u8 *buf,
					    unsigned int portno)
{
	unsigned int i, dw;
Al Viro's avatar
Al Viro committed
368
	__le32 *buf32 = (__le32 *) buf;
Linus Torvalds's avatar
Linus Torvalds committed
369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419
	u8 dev_reg;

	unsigned int dimm_sg = PDC_20621_DIMM_BASE +
			       (PDC_DIMM_WINDOW_STEP * portno) +
			       PDC_DIMM_APKT_PRD;
	VPRINTK("ENTER, dimm_sg == 0x%x, %d\n", dimm_sg, dimm_sg);

	i = PDC_DIMM_ATA_PKT;

	/*
	 * Set up ATA packet
	 */
	if ((tf->protocol == ATA_PROT_DMA) && (!(tf->flags & ATA_TFLAG_WRITE)))
		buf[i++] = PDC_PKT_READ;
	else if (tf->protocol == ATA_PROT_NODATA)
		buf[i++] = PDC_PKT_NODATA;
	else
		buf[i++] = 0;
	buf[i++] = 0;			/* reserved */
	buf[i++] = portno + 1;		/* seq. id */
	buf[i++] = 0xff;		/* delay seq. id */

	/* dimm dma S/G, and next-pkt */
	dw = i >> 2;
	if (tf->protocol == ATA_PROT_NODATA)
		buf32[dw] = 0;
	else
		buf32[dw] = cpu_to_le32(dimm_sg);
	buf32[dw + 1] = 0;
	i += 8;

	if (devno == 0)
		dev_reg = ATA_DEVICE_OBS;
	else
		dev_reg = ATA_DEVICE_OBS | ATA_DEV1;

	/* select device */
	buf[i++] = (1 << 5) | PDC_PKT_CLEAR_BSY | ATA_REG_DEVICE;
	buf[i++] = dev_reg;

	/* device control register */
	buf[i++] = (1 << 5) | PDC_REG_DEVCTL;
	buf[i++] = tf->ctl;

	return i;
}

static inline void pdc20621_host_pkt(struct ata_taskfile *tf, u8 *buf,
				     unsigned int portno)
{
	unsigned int dw;
Al Viro's avatar
Al Viro committed
420 421
	u32 tmp;
	__le32 *buf32 = (__le32 *) buf;
Linus Torvalds's avatar
Linus Torvalds committed
422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458

	unsigned int host_sg = PDC_20621_DIMM_BASE +
			       (PDC_DIMM_WINDOW_STEP * portno) +
			       PDC_DIMM_HOST_PRD;
	unsigned int dimm_sg = PDC_20621_DIMM_BASE +
			       (PDC_DIMM_WINDOW_STEP * portno) +
			       PDC_DIMM_HPKT_PRD;
	VPRINTK("ENTER, dimm_sg == 0x%x, %d\n", dimm_sg, dimm_sg);
	VPRINTK("host_sg == 0x%x, %d\n", host_sg, host_sg);

	dw = PDC_DIMM_HOST_PKT >> 2;

	/*
	 * Set up Host DMA packet
	 */
	if ((tf->protocol == ATA_PROT_DMA) && (!(tf->flags & ATA_TFLAG_WRITE)))
		tmp = PDC_PKT_READ;
	else
		tmp = 0;
	tmp |= ((portno + 1 + 4) << 16);	/* seq. id */
	tmp |= (0xff << 24);			/* delay seq. id */
	buf32[dw + 0] = cpu_to_le32(tmp);
	buf32[dw + 1] = cpu_to_le32(host_sg);
	buf32[dw + 2] = cpu_to_le32(dimm_sg);
	buf32[dw + 3] = 0;

	VPRINTK("HOST PKT @ %x == (0x%x 0x%x 0x%x 0x%x)\n",
		PDC_20621_DIMM_BASE + (PDC_DIMM_WINDOW_STEP * portno) +
			PDC_DIMM_HOST_PKT,
		buf32[dw + 0],
		buf32[dw + 1],
		buf32[dw + 2],
		buf32[dw + 3]);
}

static void pdc20621_dma_prep(struct ata_queued_cmd *qc)
{
459
	struct scatterlist *sg;
Linus Torvalds's avatar
Linus Torvalds committed
460 461
	struct ata_port *ap = qc->ap;
	struct pdc_port_priv *pp = ap->private_data;
Tejun Heo's avatar
Tejun Heo committed
462 463
	void __iomem *mmio = ap->host->iomap[PDC_MMIO_BAR];
	void __iomem *dimm_mmio = ap->host->iomap[PDC_DIMM_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
464
	unsigned int portno = ap->port_no;
Tejun Heo's avatar
Tejun Heo committed
465
	unsigned int i, si, idx, total_len = 0, sgt_len;
Al Viro's avatar
Al Viro committed
466
	__le32 *buf = (__le32 *) &pp->dimm_buf[PDC_DIMM_HEADER_SZ];
Linus Torvalds's avatar
Linus Torvalds committed
467

468
	WARN_ON(!(qc->flags & ATA_QCFLAG_DMAMAP));
Linus Torvalds's avatar
Linus Torvalds committed
469

Tejun Heo's avatar
Tejun Heo committed
470
	VPRINTK("ata%u: ENTER\n", ap->print_id);
Linus Torvalds's avatar
Linus Torvalds committed
471 472 473 474 475 476 477 478

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

	/*
	 * Build S/G table
	 */
	idx = 0;
Tejun Heo's avatar
Tejun Heo committed
479
	for_each_sg(qc->sg, sg, qc->n_elem, si) {
480 481 482
		buf[idx++] = cpu_to_le32(sg_dma_address(sg));
		buf[idx++] = cpu_to_le32(sg_dma_len(sg));
		total_len += sg_dma_len(sg);
Linus Torvalds's avatar
Linus Torvalds committed
483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521
	}
	buf[idx - 1] |= cpu_to_le32(ATA_PRD_EOT);
	sgt_len = idx * 4;

	/*
	 * Build ATA, host DMA packets
	 */
	pdc20621_host_sg(&qc->tf, &pp->dimm_buf[0], portno, total_len);
	pdc20621_host_pkt(&qc->tf, &pp->dimm_buf[0], portno);

	pdc20621_ata_sg(&qc->tf, &pp->dimm_buf[0], portno, total_len);
	i = pdc20621_ata_pkt(&qc->tf, qc->dev->devno, &pp->dimm_buf[0], portno);

	if (qc->tf.flags & ATA_TFLAG_LBA48)
		i = pdc_prep_lba48(&qc->tf, &pp->dimm_buf[0], i);
	else
		i = pdc_prep_lba28(&qc->tf, &pp->dimm_buf[0], i);

	pdc_pkt_footer(&qc->tf, &pp->dimm_buf[0], i);

	/* copy three S/G tables and two packets to DIMM MMIO window */
	memcpy_toio(dimm_mmio + (portno * PDC_DIMM_WINDOW_STEP),
		    &pp->dimm_buf, PDC_DIMM_HEADER_SZ);
	memcpy_toio(dimm_mmio + (portno * PDC_DIMM_WINDOW_STEP) +
		    PDC_DIMM_HOST_PRD,
		    &pp->dimm_buf[PDC_DIMM_HEADER_SZ], sgt_len);

	/* force host FIFO dump */
	writel(0x00000001, mmio + PDC_20621_GENERAL_CTL);

	readl(dimm_mmio);	/* MMIO PCI posting flush */

	VPRINTK("ata pkt buf ofs %u, prd size %u, mmio copied\n", i, sgt_len);
}

static void pdc20621_nodata_prep(struct ata_queued_cmd *qc)
{
	struct ata_port *ap = qc->ap;
	struct pdc_port_priv *pp = ap->private_data;
Tejun Heo's avatar
Tejun Heo committed
522 523
	void __iomem *mmio = ap->host->iomap[PDC_MMIO_BAR];
	void __iomem *dimm_mmio = ap->host->iomap[PDC_DIMM_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
524 525 526
	unsigned int portno = ap->port_no;
	unsigned int i;

Tejun Heo's avatar
Tejun Heo committed
527
	VPRINTK("ata%u: ENTER\n", ap->print_id);
Linus Torvalds's avatar
Linus Torvalds committed
528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 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 569 570 571

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

	i = pdc20621_ata_pkt(&qc->tf, qc->dev->devno, &pp->dimm_buf[0], portno);

	if (qc->tf.flags & ATA_TFLAG_LBA48)
		i = pdc_prep_lba48(&qc->tf, &pp->dimm_buf[0], i);
	else
		i = pdc_prep_lba28(&qc->tf, &pp->dimm_buf[0], i);

	pdc_pkt_footer(&qc->tf, &pp->dimm_buf[0], i);

	/* copy three S/G tables and two packets to DIMM MMIO window */
	memcpy_toio(dimm_mmio + (portno * PDC_DIMM_WINDOW_STEP),
		    &pp->dimm_buf, PDC_DIMM_HEADER_SZ);

	/* force host FIFO dump */
	writel(0x00000001, mmio + PDC_20621_GENERAL_CTL);

	readl(dimm_mmio);	/* MMIO PCI posting flush */

	VPRINTK("ata pkt buf ofs %u, mmio copied\n", i);
}

static void pdc20621_qc_prep(struct ata_queued_cmd *qc)
{
	switch (qc->tf.protocol) {
	case ATA_PROT_DMA:
		pdc20621_dma_prep(qc);
		break;
	case ATA_PROT_NODATA:
		pdc20621_nodata_prep(qc);
		break;
	default:
		break;
	}
}

static void __pdc20621_push_hdma(struct ata_queued_cmd *qc,
				 unsigned int seq,
				 u32 pkt_ofs)
{
	struct ata_port *ap = qc->ap;
Jeff Garzik's avatar
Jeff Garzik committed
572
	struct ata_host *host = ap->host;
Tejun Heo's avatar
Tejun Heo committed
573
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

	writel(0x00000001, mmio + PDC_20621_SEQCTL + (seq * 4));
	readl(mmio + PDC_20621_SEQCTL + (seq * 4));	/* flush */

	writel(pkt_ofs, mmio + PDC_HDMA_PKT_SUBMIT);
	readl(mmio + PDC_HDMA_PKT_SUBMIT);	/* flush */
}

static void pdc20621_push_hdma(struct ata_queued_cmd *qc,
				unsigned int seq,
				u32 pkt_ofs)
{
	struct ata_port *ap = qc->ap;
Jeff Garzik's avatar
Jeff Garzik committed
590
	struct pdc_host_priv *pp = ap->host->private_data;
Linus Torvalds's avatar
Linus Torvalds committed
591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607
	unsigned int idx = pp->hdma_prod & PDC_HDMA_Q_MASK;

	if (!pp->doing_hdma) {
		__pdc20621_push_hdma(qc, seq, pkt_ofs);
		pp->doing_hdma = 1;
		return;
	}

	pp->hdma[idx].qc = qc;
	pp->hdma[idx].seq = seq;
	pp->hdma[idx].pkt_ofs = pkt_ofs;
	pp->hdma_prod++;
}

static void pdc20621_pop_hdma(struct ata_queued_cmd *qc)
{
	struct ata_port *ap = qc->ap;
Jeff Garzik's avatar
Jeff Garzik committed
608
	struct pdc_host_priv *pp = ap->host->private_data;
Linus Torvalds's avatar
Linus Torvalds committed
609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626
	unsigned int idx = pp->hdma_cons & PDC_HDMA_Q_MASK;

	/* if nothing on queue, we're done */
	if (pp->hdma_prod == pp->hdma_cons) {
		pp->doing_hdma = 0;
		return;
	}

	__pdc20621_push_hdma(pp->hdma[idx].qc, pp->hdma[idx].seq,
			     pp->hdma[idx].pkt_ofs);
	pp->hdma_cons++;
}

#ifdef ATA_VERBOSE_DEBUG
static void pdc20621_dump_hdma(struct ata_queued_cmd *qc)
{
	struct ata_port *ap = qc->ap;
	unsigned int port_no = ap->port_no;
Tejun Heo's avatar
Tejun Heo committed
627
	void __iomem *dimm_mmio = ap->host->iomap[PDC_DIMM_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643

	dimm_mmio += (port_no * PDC_DIMM_WINDOW_STEP);
	dimm_mmio += PDC_DIMM_HOST_PKT;

	printk(KERN_ERR "HDMA[0] == 0x%08X\n", readl(dimm_mmio));
	printk(KERN_ERR "HDMA[1] == 0x%08X\n", readl(dimm_mmio + 4));
	printk(KERN_ERR "HDMA[2] == 0x%08X\n", readl(dimm_mmio + 8));
	printk(KERN_ERR "HDMA[3] == 0x%08X\n", readl(dimm_mmio + 12));
}
#else
static inline void pdc20621_dump_hdma(struct ata_queued_cmd *qc) { }
#endif /* ATA_VERBOSE_DEBUG */

static void pdc20621_packet_start(struct ata_queued_cmd *qc)
{
	struct ata_port *ap = qc->ap;
Jeff Garzik's avatar
Jeff Garzik committed
644
	struct ata_host *host = ap->host;
Linus Torvalds's avatar
Linus Torvalds committed
645
	unsigned int port_no = ap->port_no;
Tejun Heo's avatar
Tejun Heo committed
646
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
647 648 649 650 651 652 653
	unsigned int rw = (qc->tf.flags & ATA_TFLAG_WRITE);
	u8 seq = (u8) (port_no + 1);
	unsigned int port_ofs;

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

Tejun Heo's avatar
Tejun Heo committed
654
	VPRINTK("ata%u: ENTER\n", ap->print_id);
Linus Torvalds's avatar
Linus Torvalds committed
655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674

	wmb();			/* flush PRD, pkt writes */

	port_ofs = PDC_20621_DIMM_BASE + (PDC_DIMM_WINDOW_STEP * port_no);

	/* if writing, we (1) DMA to DIMM, then (2) do ATA command */
	if (rw && qc->tf.protocol == ATA_PROT_DMA) {
		seq += 4;

		pdc20621_dump_hdma(qc);
		pdc20621_push_hdma(qc, seq, port_ofs + PDC_DIMM_HOST_PKT);
		VPRINTK("queued ofs 0x%x (%u), seq %u\n",
			port_ofs + PDC_DIMM_HOST_PKT,
			port_ofs + PDC_DIMM_HOST_PKT,
			seq);
	} else {
		writel(0x00000001, mmio + PDC_20621_SEQCTL + (seq * 4));
		readl(mmio + PDC_20621_SEQCTL + (seq * 4));	/* flush */

		writel(port_ofs + PDC_DIMM_ATA_PKT,
Tejun Heo's avatar
Tejun Heo committed
675 676
		       ap->ioaddr.cmd_addr + PDC_PKT_SUBMIT);
		readl(ap->ioaddr.cmd_addr + PDC_PKT_SUBMIT);
Linus Torvalds's avatar
Linus Torvalds committed
677 678 679 680 681 682 683
		VPRINTK("submitted ofs 0x%x (%u), seq %u\n",
			port_ofs + PDC_DIMM_ATA_PKT,
			port_ofs + PDC_DIMM_ATA_PKT,
			seq);
	}
}

Tejun Heo's avatar
Tejun Heo committed
684
static unsigned int pdc20621_qc_issue(struct ata_queued_cmd *qc)
Linus Torvalds's avatar
Linus Torvalds committed
685 686 687
{
	switch (qc->tf.protocol) {
	case ATA_PROT_NODATA:
688 689 690 691
		if (qc->tf.flags & ATA_TFLAG_POLLING)
			break;
		/*FALLTHROUGH*/
	case ATA_PROT_DMA:
Linus Torvalds's avatar
Linus Torvalds committed
692 693 694
		pdc20621_packet_start(qc);
		return 0;

695
	case ATAPI_PROT_DMA:
Linus Torvalds's avatar
Linus Torvalds committed
696 697 698 699 700 701 702
		BUG();
		break;

	default:
		break;
	}

Tejun Heo's avatar
Tejun Heo committed
703
	return ata_sff_qc_issue(qc);
Linus Torvalds's avatar
Linus Torvalds committed
704 705
}

706 707
static inline unsigned int pdc20621_host_intr(struct ata_port *ap,
					  struct ata_queued_cmd *qc,
Linus Torvalds's avatar
Linus Torvalds committed
708
					  unsigned int doing_hdma,
709
					  void __iomem *mmio)
Linus Torvalds's avatar
Linus Torvalds committed
710 711 712 713 714 715 716 717 718 719 720 721 722 723
{
	unsigned int port_no = ap->port_no;
	unsigned int port_ofs =
		PDC_20621_DIMM_BASE + (PDC_DIMM_WINDOW_STEP * port_no);
	u8 status;
	unsigned int handled = 0;

	VPRINTK("ENTER\n");

	if ((qc->tf.protocol == ATA_PROT_DMA) &&	/* read */
	    (!(qc->tf.flags & ATA_TFLAG_WRITE))) {

		/* step two - DMA from DIMM to host */
		if (doing_hdma) {
Tejun Heo's avatar
Tejun Heo committed
724
			VPRINTK("ata%u: read hdma, 0x%x 0x%x\n", ap->print_id,
Linus Torvalds's avatar
Linus Torvalds committed
725 726
				readl(mmio + 0x104), readl(mmio + PDC_HDMA_CTLSTAT));
			/* get drive status; clear intr; complete txn */
727 728
			qc->err_mask |= ac_err_mask(ata_wait_idle(ap));
			ata_qc_complete(qc);
Linus Torvalds's avatar
Linus Torvalds committed
729 730 731 732 733 734
			pdc20621_pop_hdma(qc);
		}

		/* step one - exec ATA command */
		else {
			u8 seq = (u8) (port_no + 1 + 4);
Tejun Heo's avatar
Tejun Heo committed
735
			VPRINTK("ata%u: read ata, 0x%x 0x%x\n", ap->print_id,
Linus Torvalds's avatar
Linus Torvalds committed
736 737 738 739 740 741 742 743 744 745 746 747 748 749
				readl(mmio + 0x104), readl(mmio + PDC_HDMA_CTLSTAT));

			/* submit hdma pkt */
			pdc20621_dump_hdma(qc);
			pdc20621_push_hdma(qc, seq,
					   port_ofs + PDC_DIMM_HOST_PKT);
		}
		handled = 1;

	} else if (qc->tf.protocol == ATA_PROT_DMA) {	/* write */

		/* step one - DMA from host to DIMM */
		if (doing_hdma) {
			u8 seq = (u8) (port_no + 1);
Tejun Heo's avatar
Tejun Heo committed
750
			VPRINTK("ata%u: write hdma, 0x%x 0x%x\n", ap->print_id,
Linus Torvalds's avatar
Linus Torvalds committed
751 752 753 754 755 756
				readl(mmio + 0x104), readl(mmio + PDC_HDMA_CTLSTAT));

			/* submit ata pkt */
			writel(0x00000001, mmio + PDC_20621_SEQCTL + (seq * 4));
			readl(mmio + PDC_20621_SEQCTL + (seq * 4));
			writel(port_ofs + PDC_DIMM_ATA_PKT,
Tejun Heo's avatar
Tejun Heo committed
757 758
			       ap->ioaddr.cmd_addr + PDC_PKT_SUBMIT);
			readl(ap->ioaddr.cmd_addr + PDC_PKT_SUBMIT);
Linus Torvalds's avatar
Linus Torvalds committed
759 760 761 762
		}

		/* step two - execute ATA command */
		else {
Tejun Heo's avatar
Tejun Heo committed
763
			VPRINTK("ata%u: write ata, 0x%x 0x%x\n", ap->print_id,
Linus Torvalds's avatar
Linus Torvalds committed
764 765
				readl(mmio + 0x104), readl(mmio + PDC_HDMA_CTLSTAT));
			/* get drive status; clear intr; complete txn */
766 767
			qc->err_mask |= ac_err_mask(ata_wait_idle(ap));
			ata_qc_complete(qc);
Linus Torvalds's avatar
Linus Torvalds committed
768 769 770 771 772 773 774
			pdc20621_pop_hdma(qc);
		}
		handled = 1;

	/* command completion, but no data xfer */
	} else if (qc->tf.protocol == ATA_PROT_NODATA) {

Tejun Heo's avatar
Tejun Heo committed
775
		status = ata_sff_busy_wait(ap, ATA_BUSY | ATA_DRQ, 1000);
Linus Torvalds's avatar
Linus Torvalds committed
776
		DPRINTK("BUS_NODATA (drv_stat 0x%X)\n", status);
777 778
		qc->err_mask |= ac_err_mask(status);
		ata_qc_complete(qc);
Linus Torvalds's avatar
Linus Torvalds committed
779 780 781 782 783 784 785 786 787 788 789
		handled = 1;

	} else {
		ap->stats.idle_irq++;
	}

	return handled;
}

static void pdc20621_irq_clear(struct ata_port *ap)
{
790
	ioread8(ap->ioaddr.status_addr);
Linus Torvalds's avatar
Linus Torvalds committed
791 792
}

793
static irqreturn_t pdc20621_interrupt(int irq, void *dev_instance)
Linus Torvalds's avatar
Linus Torvalds committed
794
{
Jeff Garzik's avatar
Jeff Garzik committed
795
	struct ata_host *host = dev_instance;
Linus Torvalds's avatar
Linus Torvalds committed
796 797 798 799
	struct ata_port *ap;
	u32 mask = 0;
	unsigned int i, tmp, port_no;
	unsigned int handled = 0;
800
	void __iomem *mmio_base;
Linus Torvalds's avatar
Linus Torvalds committed
801 802 803

	VPRINTK("ENTER\n");

Tejun Heo's avatar
Tejun Heo committed
804
	if (!host || !host->iomap[PDC_MMIO_BAR]) {
Linus Torvalds's avatar
Linus Torvalds committed
805 806 807 808
		VPRINTK("QUICK EXIT\n");
		return IRQ_NONE;
	}

Tejun Heo's avatar
Tejun Heo committed
809
	mmio_base = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825

	/* reading should also clear interrupts */
	mmio_base += PDC_CHIP0_OFS;
	mask = readl(mmio_base + PDC_20621_SEQMASK);
	VPRINTK("mask == 0x%x\n", mask);

	if (mask == 0xffffffff) {
		VPRINTK("QUICK EXIT 2\n");
		return IRQ_NONE;
	}
	mask &= 0xffff;		/* only 16 tags possible */
	if (!mask) {
		VPRINTK("QUICK EXIT 3\n");
		return IRQ_NONE;
	}

826
	spin_lock(&host->lock);
Linus Torvalds's avatar
Linus Torvalds committed
827

828
	for (i = 1; i < 9; i++) {
Linus Torvalds's avatar
Linus Torvalds committed
829 830 831
		port_no = i - 1;
		if (port_no > 3)
			port_no -= 4;
Jeff Garzik's avatar
Jeff Garzik committed
832
		if (port_no >= host->n_ports)
Linus Torvalds's avatar
Linus Torvalds committed
833 834
			ap = NULL;
		else
Jeff Garzik's avatar
Jeff Garzik committed
835
			ap = host->ports[port_no];
Linus Torvalds's avatar
Linus Torvalds committed
836 837
		tmp = mask & (1 << i);
		VPRINTK("seq %u, port_no %u, ap %p, tmp %x\n", i, port_no, ap, tmp);
Tejun Heo's avatar
Tejun Heo committed
838
		if (tmp && ap) {
Linus Torvalds's avatar
Linus Torvalds committed
839 840
			struct ata_queued_cmd *qc;

Tejun Heo's avatar
Tejun Heo committed
841
			qc = ata_qc_from_tag(ap, ap->link.active_tag);
842
			if (qc && (!(qc->tf.flags & ATA_TFLAG_POLLING)))
Linus Torvalds's avatar
Linus Torvalds committed
843 844 845 846 847
				handled += pdc20621_host_intr(ap, qc, (i > 4),
							      mmio_base);
		}
	}

848
	spin_unlock(&host->lock);
Linus Torvalds's avatar
Linus Torvalds committed
849 850 851 852 853 854 855 856

	VPRINTK("mask == 0x%x\n", mask);

	VPRINTK("EXIT\n");

	return IRQ_RETVAL(handled);
}

857
static void pdc_freeze(struct ata_port *ap)
Linus Torvalds's avatar
Linus Torvalds committed
858
{
859 860
	void __iomem *mmio = ap->ioaddr.cmd_addr;
	u32 tmp;
Linus Torvalds's avatar
Linus Torvalds committed
861

862
	/* FIXME: if all 4 ATA engines are stopped, also stop HDMA engine */
Linus Torvalds's avatar
Linus Torvalds committed
863

864 865 866 867 868 869
	tmp = readl(mmio + PDC_CTLSTAT);
	tmp |= PDC_MASK_INT;
	tmp &= ~PDC_DMA_ENABLE;
	writel(tmp, mmio + PDC_CTLSTAT);
	readl(mmio + PDC_CTLSTAT); /* flush */
}
Jeff Garzik's avatar
Jeff Garzik committed
870

871 872 873 874
static void pdc_thaw(struct ata_port *ap)
{
	void __iomem *mmio = ap->ioaddr.cmd_addr;
	u32 tmp;
Linus Torvalds's avatar
Linus Torvalds committed
875

876
	/* FIXME: start HDMA engine, if zero ATA engines running */
Linus Torvalds's avatar
Linus Torvalds committed
877

878 879
	/* clear IRQ */
	ioread8(ap->ioaddr.status_addr);
Linus Torvalds's avatar
Linus Torvalds committed
880

881 882 883 884 885 886
	/* turn IRQ back on */
	tmp = readl(mmio + PDC_CTLSTAT);
	tmp &= ~PDC_MASK_INT;
	writel(tmp, mmio + PDC_CTLSTAT);
	readl(mmio + PDC_CTLSTAT); /* flush */
}
Linus Torvalds's avatar
Linus Torvalds committed
887

888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904
static void pdc_reset_port(struct ata_port *ap)
{
	void __iomem *mmio = ap->ioaddr.cmd_addr + PDC_CTLSTAT;
	unsigned int i;
	u32 tmp;

	/* FIXME: handle HDMA copy engine */

	for (i = 11; i > 0; i--) {
		tmp = readl(mmio);
		if (tmp & PDC_RESET)
			break;

		udelay(100);

		tmp |= PDC_RESET;
		writel(tmp, mmio);
Linus Torvalds's avatar
Linus Torvalds committed
905 906
	}

907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923
	tmp &= ~PDC_RESET;
	writel(tmp, mmio);
	readl(mmio);	/* flush */
}

static int pdc_softreset(struct ata_link *link, unsigned int *class,
			 unsigned long deadline)
{
	pdc_reset_port(link->ap);
	return ata_sff_softreset(link, class, deadline);
}

static void pdc_error_handler(struct ata_port *ap)
{
	if (!(ap->pflags & ATA_PFLAG_FROZEN))
		pdc_reset_port(ap);

924
	ata_sff_error_handler(ap);
925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963
}

static void pdc_post_internal_cmd(struct ata_queued_cmd *qc)
{
	struct ata_port *ap = qc->ap;

	/* make DMA engine forget about the failed command */
	if (qc->flags & ATA_QCFLAG_FAILED)
		pdc_reset_port(ap);
}

static int pdc_check_atapi_dma(struct ata_queued_cmd *qc)
{
	u8 *scsicmd = qc->scsicmd->cmnd;
	int pio = 1; /* atapi dma off by default */

	/* Whitelist commands that may use DMA. */
	switch (scsicmd[0]) {
	case WRITE_12:
	case WRITE_10:
	case WRITE_6:
	case READ_12:
	case READ_10:
	case READ_6:
	case 0xad: /* READ_DVD_STRUCTURE */
	case 0xbe: /* READ_CD */
		pio = 0;
	}
	/* -45150 (FFFF4FA2) to -1 (FFFFFFFF) shall use PIO mode */
	if (scsicmd[0] == WRITE_10) {
		unsigned int lba =
			(scsicmd[2] << 24) |
			(scsicmd[3] << 16) |
			(scsicmd[4] << 8) |
			scsicmd[5];
		if (lba >= 0xFFFF4FA2)
			pio = 1;
	}
	return pio;
Linus Torvalds's avatar
Linus Torvalds committed
964 965
}

966
static void pdc_tf_load_mmio(struct ata_port *ap, const struct ata_taskfile *tf)
Linus Torvalds's avatar
Linus Torvalds committed
967
{
968
	WARN_ON(tf->protocol == ATA_PROT_DMA ||
969
		tf->protocol == ATAPI_PROT_DMA);
Tejun Heo's avatar
Tejun Heo committed
970
	ata_sff_tf_load(ap, tf);
Linus Torvalds's avatar
Linus Torvalds committed
971 972 973
}


974
static void pdc_exec_command_mmio(struct ata_port *ap, const struct ata_taskfile *tf)
Linus Torvalds's avatar
Linus Torvalds committed
975
{
976
	WARN_ON(tf->protocol == ATA_PROT_DMA ||
977
		tf->protocol == ATAPI_PROT_DMA);
Tejun Heo's avatar
Tejun Heo committed
978
	ata_sff_exec_command(ap, tf);
Linus Torvalds's avatar
Linus Torvalds committed
979 980 981
}


Tejun Heo's avatar
Tejun Heo committed
982
static void pdc_sata_setup_port(struct ata_ioports *port, void __iomem *base)
Linus Torvalds's avatar
Linus Torvalds committed
983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000
{
	port->cmd_addr		= base;
	port->data_addr		= base;
	port->feature_addr	=
	port->error_addr	= base + 0x4;
	port->nsect_addr	= base + 0x8;
	port->lbal_addr		= base + 0xc;
	port->lbam_addr		= base + 0x10;
	port->lbah_addr		= base + 0x14;
	port->device_addr	= base + 0x18;
	port->command_addr	=
	port->status_addr	= base + 0x1c;
	port->altstatus_addr	=
	port->ctl_addr		= base + 0x38;
}


#ifdef ATA_VERBOSE_DEBUG
1001
static void pdc20621_get_from_dimm(struct ata_host *host, void *psource,
Linus Torvalds's avatar
Linus Torvalds committed
1002 1003 1004 1005 1006 1007
				   u32 offset, u32 size)
{
	u32 window_size;
	u16 idx;
	u8 page_mask;
	long dist;
1008 1009
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
	void __iomem *dimm_mmio = host->iomap[PDC_DIMM_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1010 1011 1012 1013

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

1014
	page_mask = 0x00;
1015
	window_size = 0x2000 * 4; /* 32K byte uchar size */
1016
	idx = (u16) (offset / window_size);
Linus Torvalds's avatar
Linus Torvalds committed
1017 1018 1019 1020 1021 1022 1023 1024

	writel(0x01, mmio + PDC_GENERAL_CTLR);
	readl(mmio + PDC_GENERAL_CTLR);
	writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
	readl(mmio + PDC_DIMM_WINDOW_CTLR);

	offset -= (idx * window_size);
	idx++;
1025
	dist = ((long) (window_size - (offset + size))) >= 0 ? size :
Linus Torvalds's avatar
Linus Torvalds committed
1026
		(long) (window_size - offset);
1027
	memcpy_fromio((char *) psource, (char *) (dimm_mmio + offset / 4),
Linus Torvalds's avatar
Linus Torvalds committed
1028 1029
		      dist);

1030
	psource += dist;
Linus Torvalds's avatar
Linus Torvalds committed
1031 1032 1033 1034 1035 1036
	size -= dist;
	for (; (long) size >= (long) window_size ;) {
		writel(0x01, mmio + PDC_GENERAL_CTLR);
		readl(mmio + PDC_GENERAL_CTLR);
		writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
		readl(mmio + PDC_DIMM_WINDOW_CTLR);
1037
		memcpy_fromio((char *) psource, (char *) (dimm_mmio),
Linus Torvalds's avatar
Linus Torvalds committed
1038 1039 1040
			      window_size / 4);
		psource += window_size;
		size -= window_size;
1041
		idx++;
Linus Torvalds's avatar
Linus Torvalds committed
1042 1043 1044 1045 1046 1047 1048
	}

	if (size) {
		writel(0x01, mmio + PDC_GENERAL_CTLR);
		readl(mmio + PDC_GENERAL_CTLR);
		writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
		readl(mmio + PDC_DIMM_WINDOW_CTLR);
1049
		memcpy_fromio((char *) psource, (char *) (dimm_mmio),
Linus Torvalds's avatar
Linus Torvalds committed
1050 1051 1052 1053 1054 1055
			      size / 4);
	}
}
#endif


1056
static void pdc20621_put_to_dimm(struct ata_host *host, void *psource,
Linus Torvalds's avatar
Linus Torvalds committed
1057 1058 1059 1060 1061 1062
				 u32 offset, u32 size)
{
	u32 window_size;
	u16 idx;
	u8 page_mask;
	long dist;
1063 1064
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
	void __iomem *dimm_mmio = host->iomap[PDC_DIMM_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1065

1066
	/* hard-code chip #0 */
Linus Torvalds's avatar
Linus Torvalds committed
1067 1068
	mmio += PDC_CHIP0_OFS;

1069
	page_mask = 0x00;
1070
	window_size = 0x2000 * 4;       /* 32K byte uchar size */
Linus Torvalds's avatar
Linus Torvalds committed
1071 1072 1073 1074
	idx = (u16) (offset / window_size);

	writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
	readl(mmio + PDC_DIMM_WINDOW_CTLR);
1075
	offset -= (idx * window_size);
Linus Torvalds's avatar
Linus Torvalds committed
1076 1077 1078
	idx++;
	dist = ((long)(s32)(window_size - (offset + size))) >= 0 ? size :
		(long) (window_size - offset);
1079
	memcpy_toio(dimm_mmio + offset / 4, psource, dist);
Linus Torvalds's avatar
Linus Torvalds committed
1080 1081 1082
	writel(0x01, mmio + PDC_GENERAL_CTLR);
	readl(mmio + PDC_GENERAL_CTLR);

1083
	psource += dist;
Linus Torvalds's avatar
Linus Torvalds committed
1084 1085 1086 1087
	size -= dist;
	for (; (long) size >= (long) window_size ;) {
		writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
		readl(mmio + PDC_DIMM_WINDOW_CTLR);
1088
		memcpy_toio(dimm_mmio, psource, window_size / 4);
Linus Torvalds's avatar
Linus Torvalds committed
1089 1090 1091 1092
		writel(0x01, mmio + PDC_GENERAL_CTLR);
		readl(mmio + PDC_GENERAL_CTLR);
		psource += window_size;
		size -= window_size;
1093
		idx++;
Linus Torvalds's avatar
Linus Torvalds committed
1094
	}
1095

Linus Torvalds's avatar
Linus Torvalds committed
1096 1097 1098
	if (size) {
		writel(((idx) << page_mask), mmio + PDC_DIMM_WINDOW_CTLR);
		readl(mmio + PDC_DIMM_WINDOW_CTLR);
1099
		memcpy_toio(dimm_mmio, psource, size / 4);
Linus Torvalds's avatar
Linus Torvalds committed
1100 1101 1102 1103 1104 1105
		writel(0x01, mmio + PDC_GENERAL_CTLR);
		readl(mmio + PDC_GENERAL_CTLR);
	}
}


1106
static unsigned int pdc20621_i2c_read(struct ata_host *host, u32 device,
Linus Torvalds's avatar
Linus Torvalds committed
1107 1108
				      u32 subaddr, u32 *pdata)
{
1109
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1110
	u32 i2creg  = 0;
1111
	u32 status;
1112
	u32 count = 0;
Linus Torvalds's avatar
Linus Torvalds committed
1113 1114 1115 1116 1117 1118 1119 1120

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

	i2creg |= device << 24;
	i2creg |= subaddr << 16;

	/* Set the device and subaddress */
1121 1122
	writel(i2creg, mmio + PDC_I2C_ADDR_DATA);
	readl(mmio + PDC_I2C_ADDR_DATA);
Linus Torvalds's avatar
Linus Torvalds committed
1123 1124

	/* Write Control to perform read operation, mask int */
1125
	writel(PDC_I2C_READ | PDC_I2C_START | PDC_I2C_MASK_INT,
1126
	       mmio + PDC_I2C_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1127 1128

	for (count = 0; count <= 1000; count ++) {
1129
		status = readl(mmio + PDC_I2C_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1130
		if (status & PDC_I2C_COMPLETE) {
1131
			status = readl(mmio + PDC_I2C_ADDR_DATA);
Linus Torvalds's avatar
Linus Torvalds committed
1132 1133 1134 1135 1136 1137
			break;
		} else if (count == 1000)
			return 0;
	}

	*pdata = (status >> 8) & 0x000000ff;
1138
	return 1;
Linus Torvalds's avatar
Linus Torvalds committed
1139 1140 1141
}


1142
static int pdc20621_detect_dimm(struct ata_host *host)
Linus Torvalds's avatar
Linus Torvalds committed
1143
{
1144
	u32 data = 0;
1145
	if (pdc20621_i2c_read(host, PDC_DIMM0_SPD_DEV_ADDRESS,
Linus Torvalds's avatar
Linus Torvalds committed
1146
			     PDC_DIMM_SPD_SYSTEM_FREQ, &data)) {
1147
		if (data == 100)
Linus Torvalds's avatar
Linus Torvalds committed
1148
			return 100;
1149
	} else
Linus Torvalds's avatar
Linus Torvalds committed
1150
		return 0;
1151

1152
	if (pdc20621_i2c_read(host, PDC_DIMM0_SPD_DEV_ADDRESS, 9, &data)) {
1153
		if (data <= 0x75)
Linus Torvalds's avatar
Linus Torvalds committed
1154
			return 133;
1155
	} else
Linus Torvalds's avatar
Linus Torvalds committed
1156
		return 0;
1157

1158
	return 0;
Linus Torvalds's avatar
Linus Torvalds committed
1159 1160 1161
}


1162
static int pdc20621_prog_dimm0(struct ata_host *host)
Linus Torvalds's avatar
Linus Torvalds committed
1163 1164 1165
{
	u32 spd0[50];
	u32 data = 0;
1166 1167
	int size, i;
	u8 bdimmsize;
1168
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1169 1170 1171 1172
	static const struct {
		unsigned int reg;
		unsigned int ofs;
	} pdc_i2c_read_data [] = {
1173
		{ PDC_DIMM_SPD_TYPE, 11 },
Linus Torvalds's avatar
Linus Torvalds committed
1174
		{ PDC_DIMM_SPD_FRESH_RATE, 12 },
1175
		{ PDC_DIMM_SPD_COLUMN_NUM, 4 },
Linus Torvalds's avatar
Linus Torvalds committed
1176 1177 1178 1179 1180 1181 1182 1183
		{ PDC_DIMM_SPD_ATTRIBUTE, 21 },
		{ PDC_DIMM_SPD_ROW_NUM, 3 },
		{ PDC_DIMM_SPD_BANK_NUM, 17 },
		{ PDC_DIMM_SPD_MODULE_ROW, 5 },
		{ PDC_DIMM_SPD_ROW_PRE_CHARGE, 27 },
		{ PDC_DIMM_SPD_ROW_ACTIVE_DELAY, 28 },
		{ PDC_DIMM_SPD_RAS_CAS_DELAY, 29 },
		{ PDC_DIMM_SPD_ACTIVE_PRECHARGE, 30 },
1184
		{ PDC_DIMM_SPD_CAS_LATENCY, 18 },
Linus Torvalds's avatar
Linus Torvalds committed
1185 1186 1187 1188 1189
	};

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

1190
	for (i = 0; i < ARRAY_SIZE(pdc_i2c_read_data); i++)
1191
		pdc20621_i2c_read(host, PDC_DIMM0_SPD_DEV_ADDRESS,
1192
				  pdc_i2c_read_data[i].reg,
Linus Torvalds's avatar
Linus Torvalds committed
1193
				  &spd0[pdc_i2c_read_data[i].ofs]);
1194

1195 1196
	data |= (spd0[4] - 8) | ((spd0[21] != 0) << 3) | ((spd0[3]-11) << 4);
	data |= ((spd0[17] / 4) << 6) | ((spd0[5] / 2) << 7) |
Linus Torvalds's avatar
Linus Torvalds committed
1197
		((((spd0[27] + 9) / 10) - 1) << 8) ;
1198
	data |= (((((spd0[29] > spd0[28])
1199
		    ? spd0[29] : spd0[28]) + 9) / 10) - 1) << 10;
1200
	data |= ((spd0[30] - spd0[29] + 9) / 10 - 2) << 12;
1201

1202
	if (spd0[18] & 0x08)
Linus Torvalds's avatar
Linus Torvalds committed
1203
		data |= ((0x03) << 14);
1204
	else if (spd0[18] & 0x04)
Linus Torvalds's avatar
Linus Torvalds committed
1205
		data |= ((0x02) << 14);
1206
	else if (spd0[18] & 0x01)
Linus Torvalds's avatar
Linus Torvalds committed
1207
		data |= ((0x01) << 14);
1208
	else
Linus Torvalds's avatar
Linus Torvalds committed
1209 1210
		data |= (0 << 14);

1211
	/*
Linus Torvalds's avatar
Linus Torvalds committed
1212 1213 1214 1215
	   Calculate the size of bDIMMSize (power of 2) and
	   merge the DIMM size by program start/end address.
	*/

1216 1217 1218 1219
	bdimmsize = spd0[4] + (spd0[5] / 2) + spd0[3] + (spd0[17] / 2) + 3;
	size = (1 << bdimmsize) >> 20;	/* size = xxx(MB) */
	data |= (((size / 16) - 1) << 16);
	data |= (0 << 23);
Linus Torvalds's avatar
Linus Torvalds committed
1220
	data |= 8;
1221
	writel(data, mmio + PDC_DIMM0_CONTROL);
1222
	readl(mmio + PDC_DIMM0_CONTROL);
1223
	return size;
Linus Torvalds's avatar
Linus Torvalds committed
1224 1225 1226
}


1227
static unsigned int pdc20621_prog_dimm_global(struct ata_host *host)
Linus Torvalds's avatar
Linus Torvalds committed
1228 1229
{
	u32 data, spd0;
Tejun Heo's avatar
Tejun Heo committed
1230
	int error, i;
1231
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1232 1233

	/* hard-code chip #0 */
1234
	mmio += PDC_CHIP0_OFS;
Linus Torvalds's avatar
Linus Torvalds committed
1235

1236
	/*
Linus Torvalds's avatar
Linus Torvalds committed
1237 1238 1239 1240 1241 1242
	  Set To Default : DIMM Module Global Control Register (0x022259F1)
	  DIMM Arbitration Disable (bit 20)
	  DIMM Data/Control Output Driving Selection (bit12 - bit15)
	  Refresh Enable (bit 17)
	*/

1243
	data = 0x022259F1;
1244 1245
	writel(data, mmio + PDC_SDRAM_CONTROL);
	readl(mmio + PDC_SDRAM_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1246 1247

	/* Turn on for ECC */
1248
	pdc20621_i2c_read(host, PDC_DIMM0_SPD_DEV_ADDRESS,
Linus Torvalds's avatar
Linus Torvalds committed
1249 1250 1251
			  PDC_DIMM_SPD_TYPE, &spd0);
	if (spd0 == 0x02) {
		data |= (0x01 << 16);
1252 1253
		writel(data, mmio + PDC_SDRAM_CONTROL);
		readl(mmio + PDC_SDRAM_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1254
		printk(KERN_ERR "Local DIMM ECC Enabled\n");
1255
	}
Linus Torvalds's avatar
Linus Torvalds committed
1256

1257 1258 1259 1260
	/* DIMM Initialization Select/Enable (bit 18/19) */
	data &= (~(1<<18));
	data |= (1<<19);
	writel(data, mmio + PDC_SDRAM_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1261

1262 1263
	error = 1;
	for (i = 1; i <= 10; i++) {   /* polling ~5 secs */
1264
		data = readl(mmio + PDC_SDRAM_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1265
		if (!(data & (1<<19))) {
1266 1267
			error = 0;
			break;
Linus Torvalds's avatar
Linus Torvalds committed
1268 1269
		}
		msleep(i*100);
1270 1271
	}
	return error;
Linus Torvalds's avatar
Linus Torvalds committed
1272
}
1273

Linus Torvalds's avatar
Linus Torvalds committed
1274

1275
static unsigned int pdc20621_dimm_init(struct ata_host *host)
Linus Torvalds's avatar
Linus Torvalds committed
1276
{
1277
	int speed, size, length;
1278 1279 1280 1281 1282 1283
	u32 addr, spd0, pci_status;
	u32 time_period = 0;
	u32 tcount = 0;
	u32 ticks = 0;
	u32 clock = 0;
	u32 fparam = 0;
1284
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1285 1286

	/* hard-code chip #0 */
1287
	mmio += PDC_CHIP0_OFS;
Linus Torvalds's avatar
Linus Torvalds committed
1288 1289 1290 1291 1292 1293 1294 1295 1296

	/* Initialize PLL based upon PCI Bus Frequency */

	/* Initialize Time Period Register */
	writel(0xffffffff, mmio + PDC_TIME_PERIOD);
	time_period = readl(mmio + PDC_TIME_PERIOD);
	VPRINTK("Time Period Register (0x40): 0x%x\n", time_period);

	/* Enable timer */
1297
	writel(PDC_TIMER_DEFAULT, mmio + PDC_TIME_CONTROL);
Linus Torvalds's avatar
Linus Torvalds committed
1298 1299 1300 1301 1302
	readl(mmio + PDC_TIME_CONTROL);

	/* Wait 3 seconds */
	msleep(3000);

1303
	/*
Linus Torvalds's avatar
Linus Torvalds committed
1304 1305 1306 1307 1308 1309 1310
	   When timer is enabled, counter is decreased every internal
	   clock cycle.
	*/

	tcount = readl(mmio + PDC_TIME_COUNTER);
	VPRINTK("Time Counter Register (0x44): 0x%x\n", tcount);

1311
	/*
Linus Torvalds's avatar
Linus Torvalds committed
1312 1313 1314
	   If SX4 is on PCI-X bus, after 3 seconds, the timer counter
	   register should be >= (0xffffffff - 3x10^8).
	*/
1315
	if (tcount >= PCI_X_TCOUNT) {
Linus Torvalds's avatar
Linus Torvalds committed
1316 1317
		ticks = (time_period - tcount);
		VPRINTK("Num counters 0x%x (%d)\n", ticks, ticks);
1318

Linus Torvalds's avatar
Linus Torvalds committed
1319 1320
		clock = (ticks / 300000);
		VPRINTK("10 * Internal clk = 0x%x (%d)\n", clock, clock);
1321

Linus Torvalds's avatar
Linus Torvalds committed
1322 1323 1324 1325 1326 1327
		clock = (clock * 33);
		VPRINTK("10 * Internal clk * 33 = 0x%x (%d)\n", clock, clock);

		/* PLL F Param (bit 22:16) */
		fparam = (1400000 / clock) - 2;
		VPRINTK("PLL F Param: 0x%x (%d)\n", fparam, fparam);
1328

Linus Torvalds's avatar
Linus Torvalds committed
1329 1330 1331 1332 1333 1334 1335 1336 1337 1338
		/* OD param = 0x2 (bit 31:30), R param = 0x5 (bit 29:25) */
		pci_status = (0x8a001824 | (fparam << 16));
	} else
		pci_status = PCI_PLL_INIT;

	/* Initialize PLL. */
	VPRINTK("pci_status: 0x%x\n", pci_status);
	writel(pci_status, mmio + PDC_CTL_STATUS);
	readl(mmio + PDC_CTL_STATUS);

1339
	/*
Linus Torvalds's avatar
Linus Torvalds committed
1340 1341 1342
	   Read SPD of DIMM by I2C interface,
	   and program the DIMM Module Controller.
	*/
1343
	if (!(speed = pdc20621_detect_dimm(host))) {
1344
		printk(KERN_ERR "Detect Local DIMM Fail\n");
Linus Torvalds's avatar
Linus Torvalds committed
1345
		return 1;	/* DIMM error */
1346 1347
	}
	VPRINTK("Local DIMM Speed = %d\n", speed);
Linus Torvalds's avatar
Linus Torvalds committed
1348

1349
	/* Programming DIMM0 Module Control Register (index_CID0:80h) */
1350
	size = pdc20621_prog_dimm0(host);
1351
	VPRINTK("Local DIMM Size = %dMB\n", size);
Linus Torvalds's avatar
Linus Torvalds committed
1352

1353
	/* Programming DIMM Module Global Control Register (index_CID0:88h) */
1354
	if (pdc20621_prog_dimm_global(host)) {
Linus Torvalds's avatar
Linus Torvalds committed
1355 1356
		printk(KERN_ERR "Programming DIMM Module Global Control Register Fail\n");
		return 1;
1357
	}
Linus Torvalds's avatar
Linus Torvalds committed
1358 1359 1360

#ifdef ATA_VERBOSE_DEBUG
	{
1361 1362 1363 1364 1365 1366
		u8 test_parttern1[40] =
			{0x55,0xAA,'P','r','o','m','i','s','e',' ',
			'N','o','t',' ','Y','e','t',' ',
			'D','e','f','i','n','e','d',' ',
			'1','.','1','0',
			'9','8','0','3','1','6','1','2',0,0};
Linus Torvalds's avatar
Linus Torvalds committed
1367 1368
		u8 test_parttern2[40] = {0};

1369 1370
		pdc20621_put_to_dimm(host, test_parttern2, 0x10040, 40);
		pdc20621_put_to_dimm(host, test_parttern2, 0x40, 40);
Linus Torvalds's avatar
Linus Torvalds committed
1371

1372 1373
		pdc20621_put_to_dimm(host, test_parttern1, 0x10040, 40);
		pdc20621_get_from_dimm(host, test_parttern2, 0x40, 40);
1374
		printk(KERN_ERR "%x, %x, %s\n", test_parttern2[0],
Linus Torvalds's avatar
Linus Torvalds committed
1375
		       test_parttern2[1], &(test_parttern2[2]));
1376
		pdc20621_get_from_dimm(host, test_parttern2, 0x10040,
Linus Torvalds's avatar
Linus Torvalds committed
1377
				       40);
1378
		printk(KERN_ERR "%x, %x, %s\n", test_parttern2[0],
Linus Torvalds's avatar
Linus Torvalds committed
1379 1380
		       test_parttern2[1], &(test_parttern2[2]));

1381 1382
		pdc20621_put_to_dimm(host, test_parttern1, 0x40, 40);
		pdc20621_get_from_dimm(host, test_parttern2, 0x40, 40);
1383
		printk(KERN_ERR "%x, %x, %s\n", test_parttern2[0],
Linus Torvalds's avatar
Linus Torvalds committed
1384 1385 1386 1387 1388 1389
		       test_parttern2[1], &(test_parttern2[2]));
	}
#endif

	/* ECC initiliazation. */

1390
	pdc20621_i2c_read(host, PDC_DIMM0_SPD_DEV_ADDRESS,
Linus Torvalds's avatar
Linus Torvalds committed
1391 1392
			  PDC_DIMM_SPD_TYPE, &spd0);
	if (spd0 == 0x02) {
1393
		void *buf;
Linus Torvalds's avatar
Linus Torvalds committed
1394 1395 1396
		VPRINTK("Start ECC initialization\n");
		addr = 0;
		length = size * 1024 * 1024;
1397
		buf = kzalloc(ECC_ERASE_BUF_SZ, GFP_KERNEL);
Linus Torvalds's avatar
Linus Torvalds committed
1398
		while (addr < length) {
1399 1400 1401
			pdc20621_put_to_dimm(host, buf, addr,
					     ECC_ERASE_BUF_SZ);
			addr += ECC_ERASE_BUF_SZ;
Linus Torvalds's avatar
Linus Torvalds committed
1402
		}
1403
		kfree(buf);
Linus Torvalds's avatar
Linus Torvalds committed
1404 1405 1406 1407 1408 1409
		VPRINTK("Finish ECC initialization\n");
	}
	return 0;
}


1410
static void pdc_20621_init(struct ata_host *host)
Linus Torvalds's avatar
Linus Torvalds committed
1411 1412
{
	u32 tmp;
1413
	void __iomem *mmio = host->iomap[PDC_MMIO_BAR];
Linus Torvalds's avatar
Linus Torvalds committed
1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440

	/* hard-code chip #0 */
	mmio += PDC_CHIP0_OFS;

	/*
	 * Select page 0x40 for our 32k DIMM window
	 */
	tmp = readl(mmio + PDC_20621_DIMM_WINDOW) & 0xffff0000;
	tmp |= PDC_PAGE_WINDOW;	/* page 40h; arbitrarily selected */
	writel(tmp, mmio + PDC_20621_DIMM_WINDOW);

	/*
	 * Reset Host DMA
	 */
	tmp = readl(mmio + PDC_HDMA_CTLSTAT);
	tmp |= PDC_RESET;
	writel(tmp, mmio + PDC_HDMA_CTLSTAT);
	readl(mmio + PDC_HDMA_CTLSTAT);		/* flush */

	udelay(10);

	tmp = readl(mmio + PDC_HDMA_CTLSTAT);
	tmp &= ~PDC_RESET;
	writel(tmp, mmio + PDC_HDMA_CTLSTAT);
	readl(mmio + PDC_HDMA_CTLSTAT);		/* flush */
}

1441 1442
static int pdc_sata_init_one(struct pci_dev *pdev,
			     const struct pci_device_id *ent)
Linus Torvalds's avatar
Linus Torvalds committed
1443 1444
{
	static int printed_version;
1445 1446 1447
	const struct ata_port_info *ppi[] =
		{ &pdc_port_info[ent->driver_data], NULL };
	struct ata_host *host;
1448
	struct pdc_host_priv *hpriv;
1449
	int i, rc;
Linus Torvalds's avatar
Linus Torvalds committed
1450 1451

	if (!printed_version++)
1452
		dev_printk(KERN_DEBUG, &pdev->dev, "version " DRV_VERSION "\n");
Linus Torvalds's avatar
Linus Torvalds committed
1453

1454 1455 1456 1457 1458 1459 1460 1461 1462
	/* allocate host */
	host = ata_host_alloc_pinfo(&pdev->dev, ppi, 4);
	hpriv = devm_kzalloc(&pdev->dev, sizeof(*hpriv), GFP_KERNEL);
	if (!host || !hpriv)
		return -ENOMEM;

	host->private_data = hpriv;

	/* acquire resources and fill host */
1463
	rc = pcim_enable_device(pdev);
Linus Torvalds's avatar
Linus Torvalds committed
1464 1465 1466
	if (rc)
		return rc;

Tejun Heo's avatar
Tejun Heo committed
1467 1468 1469
	rc = pcim_iomap_regions(pdev, (1 << PDC_MMIO_BAR) | (1 << PDC_DIMM_BAR),
				DRV_NAME);
	if (rc == -EBUSY)
1470
		pcim_pin_device(pdev);
Tejun Heo's avatar
Tejun Heo committed
1471
	if (rc)
1472
		return rc;
1473 1474
	host->iomap = pcim_iomap_table(pdev);

1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485
	for (i = 0; i < 4; i++) {
		struct ata_port *ap = host->ports[i];
		void __iomem *base = host->iomap[PDC_MMIO_BAR] + PDC_CHIP0_OFS;
		unsigned int offset = 0x200 + i * 0x80;

		pdc_sata_setup_port(&ap->ioaddr, base + offset);

		ata_port_pbar_desc(ap, PDC_MMIO_BAR, -1, "mmio");
		ata_port_pbar_desc(ap, PDC_DIMM_BAR, -1, "dimm");
		ata_port_pbar_desc(ap, PDC_MMIO_BAR, offset, "port");
	}
Linus Torvalds's avatar
Linus Torvalds committed
1486

1487
	/* configure and activate */
Linus Torvalds's avatar
Linus Torvalds committed
1488 1489
	rc = pci_set_dma_mask(pdev, ATA_DMA_MASK);
	if (rc)
1490
		return rc;
Linus Torvalds's avatar
Linus Torvalds committed
1491 1492
	rc = pci_set_consistent_dma_mask(pdev, ATA_DMA_MASK);
	if (rc)
1493
		return rc;
Linus Torvalds's avatar
Linus Torvalds committed
1494

1495
	if (pdc20621_dimm_init(host))
1496
		return -ENOMEM;
1497
	pdc_20621_init(host);
Linus Torvalds's avatar
Linus Torvalds committed
1498 1499

	pci_set_master(pdev);
1500 1501
	return ata_host_activate(host, pdev->irq, pdc20621_interrupt,
				 IRQF_SHARED, &pdc_sata_sht);
Linus Torvalds's avatar
Linus Torvalds committed
1502 1503 1504 1505 1506
}


static int __init pdc_sata_init(void)
{
1507
	return pci_register_driver(&pdc_sata_pci_driver);
Linus Torvalds's avatar
Linus Torvalds committed
1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524
}


static void __exit pdc_sata_exit(void)
{
	pci_unregister_driver(&pdc_sata_pci_driver);
}


MODULE_AUTHOR("Jeff Garzik");
MODULE_DESCRIPTION("Promise SATA low-level driver");
MODULE_LICENSE("GPL");
MODULE_DEVICE_TABLE(pci, pdc_sata_pci_tbl);
MODULE_VERSION(DRV_VERSION);

module_init(pdc_sata_init);
module_exit(pdc_sata_exit);