setup-pci.c 16.5 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2 3 4
 *  Copyright (C) 1998-2000  Andre Hedrick <andre@linux-ide.org>
 *  Copyright (C) 1995-1998  Mark Lord
 *  Copyright (C)      2007  Bartlomiej Zolnierkiewicz
5
 *
L
Linus Torvalds 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *  May be copied or modified under the terms of the GNU General Public License
 */

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/pci.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/ide.h>
#include <linux/dma-mapping.h>

#include <asm/io.h>

/**
 *	ide_setup_pci_baseregs	-	place a PCI IDE controller native
 *	@dev: PCI device of interface to switch native
 *	@name: Name of interface
 *
 *	We attempt to place the PCI interface into PCI native mode. If
 *	we succeed the BARs are ok and the controller is in PCI mode.
26
 *	Returns 0 on success or an errno code.
L
Linus Torvalds 已提交
27 28 29 30
 *
 *	FIXME: if we program the interface and then fail to set the BARS
 *	we don't switch it back to legacy mode. Do we actually care ??
 */
31 32

static int ide_setup_pci_baseregs(struct pci_dev *dev, const char *name)
L
Linus Torvalds 已提交
33 34 35 36 37 38 39 40 41
{
	u8 progif = 0;

	/*
	 * Place both IDE interfaces into PCI "native" mode:
	 */
	if (pci_read_config_byte(dev, PCI_CLASS_PROG, &progif) ||
			 (progif & 5) != 5) {
		if ((progif & 0xa) != 0xa) {
42 43
			printk(KERN_INFO "%s %s: device not capable of full "
				"native PCI mode\n", name, pci_name(dev));
L
Linus Torvalds 已提交
44 45
			return -EOPNOTSUPP;
		}
46 47
		printk(KERN_INFO "%s %s: placing both ports into native PCI "
			"mode\n", name, pci_name(dev));
L
Linus Torvalds 已提交
48 49 50
		(void) pci_write_config_byte(dev, PCI_CLASS_PROG, progif|5);
		if (pci_read_config_byte(dev, PCI_CLASS_PROG, &progif) ||
		    (progif & 5) != 5) {
51 52 53
			printk(KERN_ERR "%s %s: rewrite of PROGIF failed, "
				"wanted 0x%04x, got 0x%04x\n",
				name, pci_name(dev), progif | 5, progif);
L
Linus Torvalds 已提交
54 55 56 57 58 59 60
			return -EOPNOTSUPP;
		}
	}
	return 0;
}

#ifdef CONFIG_BLK_DEV_IDEDMA_PCI
61
static int ide_pci_clear_simplex(unsigned long dma_base, const char *name)
62 63 64 65 66
{
	u8 dma_stat = inb(dma_base + 2);

	outb(dma_stat & 0x60, dma_base + 2);
	dma_stat = inb(dma_base + 2);
67 68

	return (dma_stat & 0x80) ? 1 : 0;
69 70
}

L
Linus Torvalds 已提交
71
/**
72
 *	ide_pci_dma_base	-	setup BMIBA
73
 *	@hwif: IDE interface
74
 *	@d: IDE port info
L
Linus Torvalds 已提交
75
 *
76
 *	Fetch the DMA Bus-Master-I/O-Base-Address (BMIBA) from PCI space.
L
Linus Torvalds 已提交
77 78
 */

79
unsigned long ide_pci_dma_base(ide_hwif_t *hwif, const struct ide_port_info *d)
L
Linus Torvalds 已提交
80
{
81 82
	struct pci_dev *dev = to_pci_dev(hwif->dev);
	unsigned long dma_base = 0;
L
Linus Torvalds 已提交
83

84
	if (hwif->host_flags & IDE_HFLAG_MMIO)
L
Linus Torvalds 已提交
85 86 87 88 89
		return hwif->dma_base;

	if (hwif->mate && hwif->mate->dma_base) {
		dma_base = hwif->mate->dma_base - (hwif->channel ? 0 : 8);
	} else {
90 91 92 93
		u8 baridx = (d->host_flags & IDE_HFLAG_CS5520) ? 2 : 4;

		dma_base = pci_resource_start(dev, baridx);

94
		if (dma_base == 0) {
95 96
			printk(KERN_ERR "%s %s: DMA base is invalid\n",
				d->name, pci_name(dev));
97 98
			return 0;
		}
L
Linus Torvalds 已提交
99 100
	}

101 102 103
	if (hwif->channel)
		dma_base += 8;

104 105 106 107 108 109
	return dma_base;
}
EXPORT_SYMBOL_GPL(ide_pci_dma_base);

int ide_pci_check_simplex(ide_hwif_t *hwif, const struct ide_port_info *d)
{
110
	struct pci_dev *dev = to_pci_dev(hwif->dev);
111 112 113
	u8 dma_stat;

	if (d->host_flags & (IDE_HFLAG_MMIO | IDE_HFLAG_CS5520))
114 115 116
		goto out;

	if (d->host_flags & IDE_HFLAG_CLEAR_SIMPLEX) {
117 118 119
		if (ide_pci_clear_simplex(hwif->dma_base, d->name))
			printk(KERN_INFO "%s %s: simplex device: DMA forced\n",
				d->name, pci_name(dev));
120 121 122 123 124 125 126 127 128 129 130 131 132
		goto out;
	}

	/*
	 * If the device claims "simplex" DMA, this means that only one of
	 * the two interfaces can be trusted with DMA at any point in time
	 * (so we should enable DMA only on one of the two interfaces).
	 *
	 * FIXME: At this point we haven't probed the drives so we can't make
	 * the appropriate decision.  Really we should defer this problem until
	 * we tune the drive then try to grab DMA ownership if we want to be
	 * the DMA end.  This has to be become dynamic to handle hot-plug.
	 */
133
	dma_stat = hwif->tp_ops->read_sff_dma_status(hwif);
134
	if ((dma_stat & 0x80) && hwif->mate && hwif->mate->dma_base) {
135 136
		printk(KERN_INFO "%s %s: simplex device: DMA disabled\n",
			d->name, pci_name(dev));
137
		return -1;
L
Linus Torvalds 已提交
138
	}
139
out:
140
	return 0;
L
Linus Torvalds 已提交
141
}
142
EXPORT_SYMBOL_GPL(ide_pci_check_simplex);
143 144 145 146

/*
 * Set up BM-DMA capability (PnP BIOS should have done this)
 */
147
int ide_pci_set_master(struct pci_dev *dev, const char *name)
148 149 150 151 152 153 154 155 156 157
{
	u16 pcicmd;

	pci_read_config_word(dev, PCI_COMMAND, &pcicmd);

	if ((pcicmd & PCI_COMMAND_MASTER) == 0) {
		pci_set_master(dev);

		if (pci_read_config_word(dev, PCI_COMMAND, &pcicmd) ||
		    (pcicmd & PCI_COMMAND_MASTER) == 0) {
158 159
			printk(KERN_ERR "%s %s: error updating PCICMD\n",
				name, pci_name(dev));
160 161 162 163 164 165
			return -EIO;
		}
	}

	return 0;
}
166
EXPORT_SYMBOL_GPL(ide_pci_set_master);
L
Linus Torvalds 已提交
167 168
#endif /* CONFIG_BLK_DEV_IDEDMA_PCI */

169
void ide_setup_pci_noise(struct pci_dev *dev, const struct ide_port_info *d)
L
Linus Torvalds 已提交
170
{
171 172 173
	printk(KERN_INFO "%s %s: IDE controller (0x%04x:0x%04x rev 0x%02x)\n",
		d->name, pci_name(dev),
		dev->vendor, dev->device, dev->revision);
L
Linus Torvalds 已提交
174 175 176 177 178 179 180
}
EXPORT_SYMBOL_GPL(ide_setup_pci_noise);


/**
 *	ide_pci_enable	-	do PCI enables
 *	@dev: PCI device
181
 *	@d: IDE port info
L
Linus Torvalds 已提交
182 183
 *
 *	Enable the IDE PCI device. We attempt to enable the device in full
184 185 186
 *	but if that fails then we only need IO space. The PCI code should
 *	have setup the proper resources for us already for controllers in
 *	legacy mode.
187
 *
L
Linus Torvalds 已提交
188 189
 *	Returns zero on success or an error code
 */
190

191
static int ide_pci_enable(struct pci_dev *dev, const struct ide_port_info *d)
L
Linus Torvalds 已提交
192
{
193
	int ret, bars;
L
Linus Torvalds 已提交
194 195

	if (pci_enable_device(dev)) {
196
		ret = pci_enable_device_io(dev);
L
Linus Torvalds 已提交
197
		if (ret < 0) {
198 199
			printk(KERN_WARNING "%s %s: couldn't enable device\n",
				d->name, pci_name(dev));
L
Linus Torvalds 已提交
200 201
			goto out;
		}
202 203
		printk(KERN_WARNING "%s %s: BIOS configuration fixed\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
204 205 206
	}

	/*
207 208 209
	 * assume all devices can do 32-bit DMA for now, we can add
	 * a DMA mask field to the struct ide_port_info if we need it
	 * (or let lower level driver set the DMA mask)
L
Linus Torvalds 已提交
210 211 212
	 */
	ret = pci_set_dma_mask(dev, DMA_32BIT_MASK);
	if (ret < 0) {
213 214
		printk(KERN_ERR "%s %s: can't set DMA mask\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
215 216 217
		goto out;
	}

218 219 220 221
	if (d->host_flags & IDE_HFLAG_SINGLE)
		bars = (1 << 2) - 1;
	else
		bars = (1 << 4) - 1;
L
Linus Torvalds 已提交
222

223 224 225 226 227 228 229 230 231
	if ((d->host_flags & IDE_HFLAG_NO_DMA) == 0) {
		if (d->host_flags & IDE_HFLAG_CS5520)
			bars |= (1 << 2);
		else
			bars |= (1 << 4);
	}

	ret = pci_request_selected_regions(dev, bars, d->name);
	if (ret < 0)
232 233
		printk(KERN_ERR "%s %s: can't reserve resources\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
234 235 236 237 238 239 240
out:
	return ret;
}

/**
 *	ide_pci_configure	-	configure an unconfigured device
 *	@dev: PCI device
241
 *	@d: IDE port info
L
Linus Torvalds 已提交
242 243 244 245
 *
 *	Enable and configure the PCI device we have been passed.
 *	Returns zero on success or an error code.
 */
246

247
static int ide_pci_configure(struct pci_dev *dev, const struct ide_port_info *d)
L
Linus Torvalds 已提交
248 249 250 251 252 253 254 255 256
{
	u16 pcicmd = 0;
	/*
	 * PnP BIOS was *supposed* to have setup this device, but we
	 * can do it ourselves, so long as the BIOS has assigned an IRQ
	 * (or possibly the device is using a "legacy header" for IRQs).
	 * Maybe the user deliberately *disabled* the device,
	 * but we'll eventually ignore it again if no drives respond.
	 */
257 258
	if (ide_setup_pci_baseregs(dev, d->name) ||
	    pci_write_config_word(dev, PCI_COMMAND, pcicmd | PCI_COMMAND_IO)) {
259 260
		printk(KERN_INFO "%s %s: device disabled (BIOS)\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
261 262 263
		return -ENODEV;
	}
	if (pci_read_config_word(dev, PCI_COMMAND, &pcicmd)) {
264 265
		printk(KERN_ERR "%s %s: error accessing PCI regs\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
266 267 268
		return -EIO;
	}
	if (!(pcicmd & PCI_COMMAND_IO)) {
269 270
		printk(KERN_ERR "%s %s: unable to enable IDE controller\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
271 272 273 274 275 276 277
		return -ENXIO;
	}
	return 0;
}

/**
 *	ide_pci_check_iomem	-	check a register is I/O
278 279 280
 *	@dev: PCI device
 *	@d: IDE port info
 *	@bar: BAR number
L
Linus Torvalds 已提交
281
 *
282 283
 *	Checks if a BAR is configured and points to MMIO space. If so,
 *	return an error code. Otherwise return 0
L
Linus Torvalds 已提交
284
 */
285

286 287
static int ide_pci_check_iomem(struct pci_dev *dev, const struct ide_port_info *d,
			       int bar)
L
Linus Torvalds 已提交
288 289
{
	ulong flags = pci_resource_flags(dev, bar);
290

L
Linus Torvalds 已提交
291 292 293 294
	/* Unconfigured ? */
	if (!flags || pci_resource_len(dev, bar) == 0)
		return 0;

295 296
	/* I/O space */
	if (flags & IORESOURCE_IO)
L
Linus Torvalds 已提交
297
		return 0;
298

L
Linus Torvalds 已提交
299 300 301 302 303
	/* Bad */
	return -EINVAL;
}

/**
304
 *	ide_hw_configure	-	configure a hw_regs_t instance
L
Linus Torvalds 已提交
305
 *	@dev: PCI device holding interface
306
 *	@d: IDE port info
307 308
 *	@port: port number
 *	@irq: PCI IRQ
309
 *	@hw: hw_regs_t instance corresponding to this port
L
Linus Torvalds 已提交
310 311 312 313 314
 *
 *	Perform the initial set up for the hardware interface structure. This
 *	is done per interface port rather than per PCI device. There may be
 *	more than one port per device.
 *
315
 *	Returns zero on success or an error code.
L
Linus Torvalds 已提交
316
 */
317

318 319
static int ide_hw_configure(struct pci_dev *dev, const struct ide_port_info *d,
			    unsigned int port, int irq, hw_regs_t *hw)
L
Linus Torvalds 已提交
320 321 322
{
	unsigned long ctl = 0, base = 0;

323
	if ((d->host_flags & IDE_HFLAG_ISA_PORTS) == 0) {
324 325
		if (ide_pci_check_iomem(dev, d, 2 * port) ||
		    ide_pci_check_iomem(dev, d, 2 * port + 1)) {
326 327 328
			printk(KERN_ERR "%s %s: I/O baseregs (BIOS) are "
				"reported as MEM for port %d!\n",
				d->name, pci_name(dev), port);
329
			return -EINVAL;
330
		}
331

L
Linus Torvalds 已提交
332 333
		ctl  = pci_resource_start(dev, 2*port+1);
		base = pci_resource_start(dev, 2*port);
334
	} else {
L
Linus Torvalds 已提交
335 336 337 338
		/* Use default values */
		ctl = port ? 0x374 : 0x3f4;
		base = port ? 0x170 : 0x1f0;
	}
339

340
	if (!base || !ctl) {
341 342
		printk(KERN_ERR "%s %s: bad PCI BARs for port %d, skipping\n",
			d->name, pci_name(dev), port);
343
		return -EINVAL;
344 345
	}

346 347 348 349 350 351
	memset(hw, 0, sizeof(*hw));
	hw->irq = irq;
	hw->dev = &dev->dev;
	hw->chipset = d->chipset ? d->chipset : ide_pci;
	ide_std_init_ports(hw, base, ctl | 2);

352
	return 0;
L
Linus Torvalds 已提交
353 354
}

355
#ifdef CONFIG_BLK_DEV_IDEDMA_PCI
L
Linus Torvalds 已提交
356 357
/**
 *	ide_hwif_setup_dma	-	configure DMA interface
358
 *	@hwif: IDE interface
359
 *	@d: IDE port info
L
Linus Torvalds 已提交
360 361 362 363 364
 *
 *	Set up the DMA base for the interface. Enable the master bits as
 *	necessary and attempt to bring the device DMA into a ready to use
 *	state
 */
365

366
int ide_hwif_setup_dma(ide_hwif_t *hwif, const struct ide_port_info *d)
L
Linus Torvalds 已提交
367
{
368
	struct pci_dev *dev = to_pci_dev(hwif->dev);
L
Linus Torvalds 已提交
369

370
	if ((d->host_flags & IDE_HFLAG_NO_AUTODMA) == 0 ||
L
Linus Torvalds 已提交
371 372
	    ((dev->class >> 8) == PCI_CLASS_STORAGE_IDE &&
	     (dev->class & 0x80))) {
373
		unsigned long base = ide_pci_dma_base(hwif, d);
374

375 376 377 378 379 380 381 382 383
		if (base == 0)
			return -1;

		hwif->dma_base = base;

		if (ide_pci_check_simplex(hwif, d) < 0)
			return -1;

		if (ide_pci_set_master(dev, d->name) < 0)
384
			return -1;
385

386
		if (hwif->host_flags & IDE_HFLAG_MMIO)
387 388 389 390 391 392 393
			printk(KERN_INFO "    %s: MMIO-DMA\n", hwif->name);
		else
			printk(KERN_INFO "    %s: BM-DMA at 0x%04lx-0x%04lx\n",
					 hwif->name, base, base + 7);

		hwif->extra_base = base + (hwif->channel ? 8 : 16);

394 395
		if (ide_allocate_dma_engine(hwif))
			return -1;
396

397
		hwif->dma_ops = &sff_dma_ops;
398
	}
399

400
	return 0;
401
}
402
#endif /* CONFIG_BLK_DEV_IDEDMA_PCI */
L
Linus Torvalds 已提交
403 404 405 406

/**
 *	ide_setup_pci_controller	-	set up IDE PCI
 *	@dev: PCI device
407
 *	@d: IDE port info
L
Linus Torvalds 已提交
408 409 410 411 412 413
 *	@noisy: verbose flag
 *
 *	Set up the PCI and controller side of the IDE interface. This brings
 *	up the PCI side of the device, checks that the device is enabled
 *	and enables it if need be
 */
414

415 416
static int ide_setup_pci_controller(struct pci_dev *dev,
				    const struct ide_port_info *d, int noisy)
L
Linus Torvalds 已提交
417 418 419 420 421 422 423 424 425 426 427 428 429
{
	int ret;
	u16 pcicmd;

	if (noisy)
		ide_setup_pci_noise(dev, d);

	ret = ide_pci_enable(dev, d);
	if (ret < 0)
		goto out;

	ret = pci_read_config_word(dev, PCI_COMMAND, &pcicmd);
	if (ret < 0) {
430 431
		printk(KERN_ERR "%s %s: error accessing PCI regs\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
432 433 434 435 436 437
		goto out;
	}
	if (!(pcicmd & PCI_COMMAND_IO)) {	/* is device disabled? */
		ret = ide_pci_configure(dev, d);
		if (ret < 0)
			goto out;
438 439
		printk(KERN_INFO "%s %s: device enabled (Linux)\n",
			d->name, pci_name(dev));
L
Linus Torvalds 已提交
440 441 442 443 444 445 446 447 448
	}

out:
	return ret;
}

/**
 *	ide_pci_setup_ports	-	configure ports/devices on PCI IDE
 *	@dev: PCI device
449
 *	@d: IDE port info
L
Linus Torvalds 已提交
450
 *	@pciirq: IRQ line
451 452
 *	@hw: hw_regs_t instances corresponding to this PCI IDE device
 *	@hws: hw_regs_t pointers table to update
L
Linus Torvalds 已提交
453 454 455 456 457 458 459 460 461
 *
 *	Scan the interfaces attached to this device and do any
 *	necessary per port setup. Attach the devices and ask the
 *	generic DMA layer to do its work for us.
 *
 *	Normally called automaticall from do_ide_pci_setup_device,
 *	but is also used directly as a helper function by some controllers
 *	where the chipset setup is not the default PCI IDE one.
 */
462

463
void ide_pci_setup_ports(struct pci_dev *dev, const struct ide_port_info *d,
464
			 int pciirq, hw_regs_t *hw, hw_regs_t **hws)
L
Linus Torvalds 已提交
465
{
466
	int channels = (d->host_flags & IDE_HFLAG_SINGLE) ? 1 : 2, port;
L
Linus Torvalds 已提交
467 468 469 470 471
	u8 tmp;

	/*
	 * Set up the IDE ports
	 */
472

473
	for (port = 0; port < channels; ++port) {
474 475
		const ide_pci_enablebit_t *e = &(d->enablebits[port]);

L
Linus Torvalds 已提交
476
		if (e->reg && (pci_read_config_byte(dev, e->reg, &tmp) ||
477
		    (tmp & e->mask) != e->val)) {
478 479
			printk(KERN_INFO "%s %s: IDE port disabled\n",
				d->name, pci_name(dev));
L
Linus Torvalds 已提交
480
			continue;	/* port not enabled */
481
		}
L
Linus Torvalds 已提交
482

483
		if (ide_hw_configure(dev, d, port, pciirq, hw + port))
L
Linus Torvalds 已提交
484 485
			continue;

486
		*(hws + port) = hw + port;
487
	}
L
Linus Torvalds 已提交
488 489 490 491 492 493 494 495 496 497
}
EXPORT_SYMBOL_GPL(ide_pci_setup_ports);

/*
 * ide_setup_pci_device() looks at the primary/secondary interfaces
 * on a PCI IDE device and, if they are enabled, prepares the IDE driver
 * for use with them.  This generic code works for most PCI chipsets.
 *
 * One thing that is not standardized is the location of the
 * primary/secondary interface "enable/disable" bits.  For chipsets that
498
 * we "know" about, this information is in the struct ide_port_info;
L
Linus Torvalds 已提交
499 500
 * for all other chipsets, we just assume both interfaces are enabled.
 */
501
static int do_ide_setup_pci_device(struct pci_dev *dev,
502
				   const struct ide_port_info *d,
503
				   u8 noisy)
L
Linus Torvalds 已提交
504 505 506 507 508 509 510 511
{
	int pciirq, ret;

	/*
	 * Can we trust the reported IRQ?
	 */
	pciirq = dev->irq;

512 513 514 515 516 517 518 519 520 521
	/*
	 * This allows offboard ide-pci cards the enable a BIOS,
	 * verify interrupt settings of split-mirror pci-config
	 * space, place chipset into init-mode, and/or preserve
	 * an interrupt if the card is not native ide support.
	 */
	ret = d->init_chipset ? d->init_chipset(dev, d->name) : 0;
	if (ret < 0)
		goto out;

L
Linus Torvalds 已提交
522 523 524
	/* Is it an "IDE storage" device in non-PCI mode? */
	if ((dev->class >> 8) == PCI_CLASS_STORAGE_IDE && (dev->class & 5) != 5) {
		if (noisy)
525 526
			printk(KERN_INFO "%s %s: not 100%% native mode: will "
				"probe irqs later\n", d->name, pci_name(dev));
L
Linus Torvalds 已提交
527
		pciirq = ret;
528 529 530 531 532 533
	} else if (!pciirq && noisy) {
		printk(KERN_WARNING "%s %s: bad irq (%d): will probe later\n",
			d->name, pci_name(dev), pciirq);
	} else if (noisy) {
		printk(KERN_INFO "%s %s: 100%% native mode on irq %d\n",
			d->name, pci_name(dev), pciirq);
L
Linus Torvalds 已提交
534 535
	}

536
	ret = pciirq;
L
Linus Torvalds 已提交
537 538 539 540
out:
	return ret;
}

541 542
int ide_pci_init_one(struct pci_dev *dev, const struct ide_port_info *d,
		     void *priv)
L
Linus Torvalds 已提交
543
{
544
	struct ide_host *host;
545
	hw_regs_t hw[4], *hws[] = { NULL, NULL, NULL, NULL };
L
Linus Torvalds 已提交
546 547
	int ret;

548 549 550 551
	ret = ide_setup_pci_controller(dev, d, 1);
	if (ret < 0)
		goto out;

552 553
	ide_pci_setup_ports(dev, d, 0, &hw[0], &hws[0]);

554 555 556 557 558 559 560 561 562 563
	host = ide_host_alloc(d, hws);
	if (host == NULL) {
		ret = -ENOMEM;
		goto out;
	}

	host->dev[0] = &dev->dev;

	host->host_priv = priv;

564
	pci_set_drvdata(dev, host);
565

566
	ret = do_ide_setup_pci_device(dev, d, 1);
567 568
	if (ret < 0)
		goto out;
569

570 571
	/* fixup IRQ */
	hw[1].irq = hw[0].irq = ret;
L
Linus Torvalds 已提交
572

573 574 575
	ret = ide_host_register(host, d, hws);
	if (ret)
		ide_host_free(host);
576
out:
L
Linus Torvalds 已提交
577 578
	return ret;
}
579
EXPORT_SYMBOL_GPL(ide_pci_init_one);
L
Linus Torvalds 已提交
580

581 582
int ide_pci_init_two(struct pci_dev *dev1, struct pci_dev *dev2,
		     const struct ide_port_info *d, void *priv)
L
Linus Torvalds 已提交
583 584
{
	struct pci_dev *pdev[] = { dev1, dev2 };
585
	struct ide_host *host;
L
Linus Torvalds 已提交
586
	int ret, i;
587
	hw_regs_t hw[4], *hws[] = { NULL, NULL, NULL, NULL };
L
Linus Torvalds 已提交
588 589

	for (i = 0; i < 2; i++) {
590 591 592 593
		ret = ide_setup_pci_controller(pdev[i], d, !i);
		if (ret < 0)
			goto out;

594
		ide_pci_setup_ports(pdev[i], d, 0, &hw[i*2], &hws[i*2]);
595
	}
596

597 598 599 600 601 602 603 604 605 606 607
	host = ide_host_alloc(d, hws);
	if (host == NULL) {
		ret = -ENOMEM;
		goto out;
	}

	host->dev[0] = &dev1->dev;
	host->dev[1] = &dev2->dev;

	host->host_priv = priv;

608 609
	pci_set_drvdata(pdev[0], host);
	pci_set_drvdata(pdev[1], host);
610 611

	for (i = 0; i < 2; i++) {
612 613
		ret = do_ide_setup_pci_device(pdev[i], d, !i);

L
Linus Torvalds 已提交
614 615 616 617 618 619
		/*
		 * FIXME: Mom, mom, they stole me the helper function to undo
		 * do_ide_setup_pci_device() on the first device!
		 */
		if (ret < 0)
			goto out;
620

621 622
		/* fixup IRQ */
		hw[i*2 + 1].irq = hw[i*2].irq = ret;
L
Linus Torvalds 已提交
623 624
	}

625 626 627
	ret = ide_host_register(host, d, hws);
	if (ret)
		ide_host_free(host);
L
Linus Torvalds 已提交
628 629 630
out:
	return ret;
}
631
EXPORT_SYMBOL_GPL(ide_pci_init_two);
632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661

void ide_pci_remove(struct pci_dev *dev)
{
	struct ide_host *host = pci_get_drvdata(dev);
	struct pci_dev *dev2 = host->dev[1] ? to_pci_dev(host->dev[1]) : NULL;
	int bars;

	if (host->host_flags & IDE_HFLAG_SINGLE)
		bars = (1 << 2) - 1;
	else
		bars = (1 << 4) - 1;

	if ((host->host_flags & IDE_HFLAG_NO_DMA) == 0) {
		if (host->host_flags & IDE_HFLAG_CS5520)
			bars |= (1 << 2);
		else
			bars |= (1 << 4);
	}

	ide_host_remove(host);

	if (dev2)
		pci_release_selected_regions(dev2, bars);
	pci_release_selected_regions(dev, bars);

	if (dev2)
		pci_disable_device(dev2);
	pci_disable_device(dev);
}
EXPORT_SYMBOL_GPL(ide_pci_remove);