pci_link.c 24.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
/*
 *  pci_link.c - ACPI PCI Interrupt Link Device Driver ($Revision: 34 $)
 *
 *  Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
 *  Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
 *  Copyright (C) 2002       Dominik Brodowski <devel@brodo.de>
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 *  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 of the License, 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; if not, write to the Free Software Foundation, Inc.,
 *  59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 * TBD: 
 *      1. Support more than one IRQ resource entry per link device (index).
 *	2. Implement start/stop mechanism and use ACPI Bus Driver facilities
 *	   for IRQ management (e.g. start()->_SRS).
 */

#include <linux/sysdev.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/types.h>
#include <linux/proc_fs.h>
#include <linux/spinlock.h>
#include <linux/pm.h>
#include <linux/pci.h>
I
Ingo Molnar 已提交
41
#include <linux/mutex.h>
L
Linus Torvalds 已提交
42 43 44 45 46

#include <acpi/acpi_bus.h>
#include <acpi/acpi_drivers.h>

#define _COMPONENT		ACPI_PCI_COMPONENT
L
Len Brown 已提交
47
ACPI_MODULE_NAME("pci_link")
L
Linus Torvalds 已提交
48 49 50 51 52 53 54
#define ACPI_PCI_LINK_CLASS		"pci_irq_routing"
#define ACPI_PCI_LINK_HID		"PNP0C0F"
#define ACPI_PCI_LINK_DRIVER_NAME	"ACPI PCI Interrupt Link Driver"
#define ACPI_PCI_LINK_DEVICE_NAME	"PCI Interrupt Link"
#define ACPI_PCI_LINK_FILE_INFO		"info"
#define ACPI_PCI_LINK_FILE_STATUS	"state"
#define ACPI_PCI_LINK_MAX_POSSIBLE 16
L
Len Brown 已提交
55 56
static int acpi_pci_link_add(struct acpi_device *device);
static int acpi_pci_link_remove(struct acpi_device *device, int type);
L
Linus Torvalds 已提交
57 58

static struct acpi_driver acpi_pci_link_driver = {
L
Len Brown 已提交
59 60 61 62 63 64 65
	.name = ACPI_PCI_LINK_DRIVER_NAME,
	.class = ACPI_PCI_LINK_CLASS,
	.ids = ACPI_PCI_LINK_HID,
	.ops = {
		.add = acpi_pci_link_add,
		.remove = acpi_pci_link_remove,
		},
L
Linus Torvalds 已提交
66 67
};

68 69 70 71
/*
 * If a link is initialized, we never change its active and initialized
 * later even the link is disable. Instead, we just repick the active irq
 */
L
Linus Torvalds 已提交
72
struct acpi_pci_link_irq {
L
Len Brown 已提交
73
	u8 active;		/* Current IRQ */
B
Bob Moore 已提交
74 75
	u8 triggering;		/* All IRQs */
	u8 polarity;	/* All IRQs */
L
Len Brown 已提交
76 77 78 79 80
	u8 resource_type;
	u8 possible_count;
	u8 possible[ACPI_PCI_LINK_MAX_POSSIBLE];
	u8 initialized:1;
	u8 reserved:7;
L
Linus Torvalds 已提交
81 82 83
};

struct acpi_pci_link {
L
Len Brown 已提交
84 85 86
	struct list_head node;
	struct acpi_device *device;
	acpi_handle handle;
L
Linus Torvalds 已提交
87
	struct acpi_pci_link_irq irq;
L
Len Brown 已提交
88
	int refcnt;
L
Linus Torvalds 已提交
89 90 91
};

static struct {
L
Len Brown 已提交
92 93 94
	int count;
	struct list_head entries;
} acpi_link;
I
Ingo Molnar 已提交
95
DEFINE_MUTEX(acpi_link_lock);
L
Linus Torvalds 已提交
96 97 98 99 100 101 102 103 104

/* --------------------------------------------------------------------------
                            PCI Link Device Management
   -------------------------------------------------------------------------- */

/*
 * set context (link) possible list from resource list
 */
static acpi_status
L
Len Brown 已提交
105
acpi_pci_link_check_possible(struct acpi_resource *resource, void *context)
L
Linus Torvalds 已提交
106
{
L
Len Brown 已提交
107 108
	struct acpi_pci_link *link = (struct acpi_pci_link *)context;
	u32 i = 0;
L
Linus Torvalds 已提交
109 110


111
	switch (resource->type) {
B
Bob Moore 已提交
112
	case ACPI_RESOURCE_TYPE_START_DEPENDENT:
113
		return AE_OK;
B
Bob Moore 已提交
114
	case ACPI_RESOURCE_TYPE_IRQ:
L
Len Brown 已提交
115 116
		{
			struct acpi_resource_irq *p = &resource->data.irq;
B
Bob Moore 已提交
117
			if (!p || !p->interrupt_count) {
118
				printk(KERN_WARNING PREFIX "Blank IRQ resource\n");
119
				return AE_OK;
L
Linus Torvalds 已提交
120
			}
L
Len Brown 已提交
121
			for (i = 0;
B
Bob Moore 已提交
122
			     (i < p->interrupt_count
L
Len Brown 已提交
123 124
			      && i < ACPI_PCI_LINK_MAX_POSSIBLE); i++) {
				if (!p->interrupts[i]) {
125 126
					printk(KERN_WARNING PREFIX "Invalid IRQ %d\n",
						      p->interrupts[i]);
L
Len Brown 已提交
127 128 129 130 131
					continue;
				}
				link->irq.possible[i] = p->interrupts[i];
				link->irq.possible_count++;
			}
B
Bob Moore 已提交
132 133 134
			link->irq.triggering = p->triggering;
			link->irq.polarity = p->polarity;
			link->irq.resource_type = ACPI_RESOURCE_TYPE_IRQ;
L
Len Brown 已提交
135
			break;
L
Linus Torvalds 已提交
136
		}
B
Bob Moore 已提交
137
	case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
L
Len Brown 已提交
138
		{
B
Bob Moore 已提交
139
			struct acpi_resource_extended_irq *p =
L
Len Brown 已提交
140
			    &resource->data.extended_irq;
B
Bob Moore 已提交
141
			if (!p || !p->interrupt_count) {
142 143
				printk(KERN_WARNING PREFIX
					      "Blank EXT IRQ resource\n");
144
				return AE_OK;
L
Len Brown 已提交
145 146
			}
			for (i = 0;
B
Bob Moore 已提交
147
			     (i < p->interrupt_count
L
Len Brown 已提交
148 149
			      && i < ACPI_PCI_LINK_MAX_POSSIBLE); i++) {
				if (!p->interrupts[i]) {
150 151
					printk(KERN_WARNING PREFIX "Invalid IRQ %d\n",
						      p->interrupts[i]);
L
Len Brown 已提交
152 153 154 155
					continue;
				}
				link->irq.possible[i] = p->interrupts[i];
				link->irq.possible_count++;
L
Linus Torvalds 已提交
156
			}
B
Bob Moore 已提交
157 158 159
			link->irq.triggering = p->triggering;
			link->irq.polarity = p->polarity;
			link->irq.resource_type = ACPI_RESOURCE_TYPE_EXTENDED_IRQ;
L
Len Brown 已提交
160
			break;
L
Linus Torvalds 已提交
161 162
		}
	default:
163
		printk(KERN_ERR PREFIX "Resource is not an IRQ entry\n");
164
		return AE_OK;
L
Linus Torvalds 已提交
165 166
	}

167
	return AE_CTRL_TERMINATE;
L
Linus Torvalds 已提交
168 169
}

L
Len Brown 已提交
170
static int acpi_pci_link_get_possible(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
171
{
L
Len Brown 已提交
172
	acpi_status status;
L
Linus Torvalds 已提交
173 174 175


	if (!link)
176
		return -EINVAL;
L
Linus Torvalds 已提交
177 178

	status = acpi_walk_resources(link->handle, METHOD_NAME__PRS,
L
Len Brown 已提交
179
				     acpi_pci_link_check_possible, link);
L
Linus Torvalds 已提交
180
	if (ACPI_FAILURE(status)) {
181
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _PRS"));
182
		return -ENODEV;
L
Linus Torvalds 已提交
183 184
	}

L
Len Brown 已提交
185 186 187
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
			  "Found %d possible IRQs\n",
			  link->irq.possible_count));
L
Linus Torvalds 已提交
188

189
	return 0;
L
Linus Torvalds 已提交
190 191 192
}

static acpi_status
L
Len Brown 已提交
193
acpi_pci_link_check_current(struct acpi_resource *resource, void *context)
L
Linus Torvalds 已提交
194
{
L
Len Brown 已提交
195
	int *irq = (int *)context;
L
Linus Torvalds 已提交
196 197


198
	switch (resource->type) {
B
Bob Moore 已提交
199
	case ACPI_RESOURCE_TYPE_IRQ:
L
Len Brown 已提交
200 201
		{
			struct acpi_resource_irq *p = &resource->data.irq;
B
Bob Moore 已提交
202
			if (!p || !p->interrupt_count) {
L
Len Brown 已提交
203 204 205 206 207 208
				/*
				 * IRQ descriptors may have no IRQ# bits set,
				 * particularly those those w/ _STA disabled
				 */
				ACPI_DEBUG_PRINT((ACPI_DB_INFO,
						  "Blank IRQ resource\n"));
209
				return AE_OK;
L
Len Brown 已提交
210 211 212
			}
			*irq = p->interrupts[0];
			break;
L
Linus Torvalds 已提交
213
		}
B
Bob Moore 已提交
214
	case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
L
Len Brown 已提交
215
		{
B
Bob Moore 已提交
216
			struct acpi_resource_extended_irq *p =
L
Len Brown 已提交
217
			    &resource->data.extended_irq;
B
Bob Moore 已提交
218
			if (!p || !p->interrupt_count) {
L
Len Brown 已提交
219 220 221 222
				/*
				 * extended IRQ descriptors must
				 * return at least 1 IRQ
				 */
223 224
				printk(KERN_WARNING PREFIX
					      "Blank EXT IRQ resource\n");
225
				return AE_OK;
L
Len Brown 已提交
226 227 228
			}
			*irq = p->interrupts[0];
			break;
L
Linus Torvalds 已提交
229
		}
230
		break;
L
Linus Torvalds 已提交
231
	default:
232
		printk(KERN_ERR PREFIX "Resource %d isn't an IRQ\n", resource->type);
233
	case ACPI_RESOURCE_TYPE_END_TAG:
234
		return AE_OK;
L
Linus Torvalds 已提交
235
	}
236
	return AE_CTRL_TERMINATE;
L
Linus Torvalds 已提交
237 238 239 240 241 242 243 244 245
}

/*
 * Run _CRS and set link->irq.active
 *
 * return value:
 * 0 - success
 * !0 - failure
 */
L
Len Brown 已提交
246
static int acpi_pci_link_get_current(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
247
{
L
Len Brown 已提交
248 249 250
	int result = 0;
	acpi_status status = AE_OK;
	int irq = 0;
L
Linus Torvalds 已提交
251 252 253


	if (!link || !link->handle)
254
		return -EINVAL;
L
Linus Torvalds 已提交
255 256 257 258 259 260 261 262

	link->irq.active = 0;

	/* in practice, status disabled is meaningless, ignore it */
	if (acpi_strict) {
		/* Query _STA, set link->device->status */
		result = acpi_bus_get_status(link->device);
		if (result) {
263
			printk(KERN_ERR PREFIX "Unable to read status\n");
L
Linus Torvalds 已提交
264 265 266 267 268
			goto end;
		}

		if (!link->device->status.enabled) {
			ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Link disabled\n"));
269
			return 0;
L
Linus Torvalds 已提交
270 271 272 273 274 275 276 277
		}
	}

	/* 
	 * Query and parse _CRS to get the current IRQ assignment. 
	 */

	status = acpi_walk_resources(link->handle, METHOD_NAME__CRS,
L
Len Brown 已提交
278
				     acpi_pci_link_check_current, &irq);
L
Linus Torvalds 已提交
279
	if (ACPI_FAILURE(status)) {
280
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _CRS"));
L
Linus Torvalds 已提交
281 282 283 284 285
		result = -ENODEV;
		goto end;
	}

	if (acpi_strict && !irq) {
286
		printk(KERN_ERR PREFIX "_CRS returned 0\n");
L
Linus Torvalds 已提交
287 288 289 290 291 292 293
		result = -ENODEV;
	}

	link->irq.active = irq;

	ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Link at IRQ %d \n", link->irq.active));

L
Len Brown 已提交
294
      end:
295
	return result;
L
Linus Torvalds 已提交
296 297
}

L
Len Brown 已提交
298
static int acpi_pci_link_set(struct acpi_pci_link *link, int irq)
L
Linus Torvalds 已提交
299
{
L
Len Brown 已提交
300 301
	int result = 0;
	acpi_status status = AE_OK;
L
Linus Torvalds 已提交
302
	struct {
L
Len Brown 已提交
303 304 305 306
		struct acpi_resource res;
		struct acpi_resource end;
	} *resource;
	struct acpi_buffer buffer = { 0, NULL };
L
Linus Torvalds 已提交
307 308 309


	if (!link || !irq)
310
		return -EINVAL;
L
Linus Torvalds 已提交
311

312
	resource = kmalloc(sizeof(*resource) + 1, GFP_ATOMIC);
L
Len Brown 已提交
313
	if (!resource)
314
		return -ENOMEM;
L
Linus Torvalds 已提交
315

L
Len Brown 已提交
316 317
	memset(resource, 0, sizeof(*resource) + 1);
	buffer.length = sizeof(*resource) + 1;
L
Linus Torvalds 已提交
318 319
	buffer.pointer = resource;

L
Len Brown 已提交
320
	switch (link->irq.resource_type) {
B
Bob Moore 已提交
321 322
	case ACPI_RESOURCE_TYPE_IRQ:
		resource->res.type = ACPI_RESOURCE_TYPE_IRQ;
L
Linus Torvalds 已提交
323
		resource->res.length = sizeof(struct acpi_resource);
B
Bob Moore 已提交
324 325 326 327 328
		resource->res.data.irq.triggering = link->irq.triggering;
		resource->res.data.irq.polarity =
		    link->irq.polarity;
		if (link->irq.triggering == ACPI_EDGE_SENSITIVE)
			resource->res.data.irq.sharable =
L
Len Brown 已提交
329
			    ACPI_EXCLUSIVE;
L
Linus Torvalds 已提交
330
		else
B
Bob Moore 已提交
331 332
			resource->res.data.irq.sharable = ACPI_SHARED;
		resource->res.data.irq.interrupt_count = 1;
L
Linus Torvalds 已提交
333 334
		resource->res.data.irq.interrupts[0] = irq;
		break;
L
Len Brown 已提交
335

B
Bob Moore 已提交
336 337
	case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
		resource->res.type = ACPI_RESOURCE_TYPE_EXTENDED_IRQ;
L
Linus Torvalds 已提交
338
		resource->res.length = sizeof(struct acpi_resource);
L
Len Brown 已提交
339 340
		resource->res.data.extended_irq.producer_consumer =
		    ACPI_CONSUMER;
B
Bob Moore 已提交
341 342 343 344 345 346
		resource->res.data.extended_irq.triggering =
		    link->irq.triggering;
		resource->res.data.extended_irq.polarity =
		    link->irq.polarity;
		if (link->irq.triggering == ACPI_EDGE_SENSITIVE)
			resource->res.data.irq.sharable =
L
Len Brown 已提交
347
			    ACPI_EXCLUSIVE;
L
Linus Torvalds 已提交
348
		else
B
Bob Moore 已提交
349 350
			resource->res.data.irq.sharable = ACPI_SHARED;
		resource->res.data.extended_irq.interrupt_count = 1;
L
Linus Torvalds 已提交
351 352 353 354
		resource->res.data.extended_irq.interrupts[0] = irq;
		/* ignore resource_source, it's optional */
		break;
	default:
355
		printk(KERN_ERR PREFIX "Invalid Resource_type %d\n", link->irq.resource_type);
L
Linus Torvalds 已提交
356 357 358 359
		result = -EINVAL;
		goto end;

	}
B
Bob Moore 已提交
360
	resource->end.type = ACPI_RESOURCE_TYPE_END_TAG;
L
Linus Torvalds 已提交
361 362 363 364 365 366

	/* Attempt to set the resource */
	status = acpi_set_current_resources(link->handle, &buffer);

	/* check for total failure */
	if (ACPI_FAILURE(status)) {
367
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _SRS"));
L
Linus Torvalds 已提交
368 369 370 371 372 373 374
		result = -ENODEV;
		goto end;
	}

	/* Query _STA, set device->status */
	result = acpi_bus_get_status(link->device);
	if (result) {
375
		printk(KERN_ERR PREFIX "Unable to read status\n");
L
Linus Torvalds 已提交
376 377 378
		goto end;
	}
	if (!link->device->status.enabled) {
379 380
		printk(KERN_WARNING PREFIX
			      "%s [%s] disabled and referenced, BIOS bug\n",
381
			      acpi_device_name(link->device),
382
			      acpi_device_bid(link->device));
L
Linus Torvalds 已提交
383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
	}

	/* Query _CRS, set link->irq.active */
	result = acpi_pci_link_get_current(link);
	if (result) {
		goto end;
	}

	/*
	 * Is current setting not what we set?
	 * set link->irq.active
	 */
	if (link->irq.active != irq) {
		/*
		 * policy: when _CRS doesn't return what we just _SRS
		 * assume _SRS worked and override _CRS value.
		 */
400 401
		printk(KERN_WARNING PREFIX
			      "%s [%s] BIOS reported IRQ %d, using IRQ %d\n",
402
			      acpi_device_name(link->device),
403
			      acpi_device_bid(link->device), link->irq.active, irq);
L
Linus Torvalds 已提交
404 405 406 407
		link->irq.active = irq;
	}

	ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Set IRQ %d\n", link->irq.active));
L
Len Brown 已提交
408 409

      end:
L
Linus Torvalds 已提交
410
	kfree(resource);
411
	return result;
L
Linus Torvalds 已提交
412 413 414 415 416 417 418 419 420 421 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 459 460 461 462
}

/* --------------------------------------------------------------------------
                            PCI Link IRQ Management
   -------------------------------------------------------------------------- */

/*
 * "acpi_irq_balance" (default in APIC mode) enables ACPI to use PIC Interrupt
 * Link Devices to move the PIRQs around to minimize sharing.
 * 
 * "acpi_irq_nobalance" (default in PIC mode) tells ACPI not to move any PIC IRQs
 * that the BIOS has already set to active.  This is necessary because
 * ACPI has no automatic means of knowing what ISA IRQs are used.  Note that
 * if the BIOS doesn't set a Link Device active, ACPI needs to program it
 * even if acpi_irq_nobalance is set.
 *
 * A tables of penalties avoids directing PCI interrupts to well known
 * ISA IRQs. Boot params are available to over-ride the default table:
 *
 * List interrupts that are free for PCI use.
 * acpi_irq_pci=n[,m]
 *
 * List interrupts that should not be used for PCI:
 * acpi_irq_isa=n[,m]
 *
 * Note that PCI IRQ routers have a list of possible IRQs,
 * which may not include the IRQs this table says are available.
 * 
 * Since this heuristic can't tell the difference between a link
 * that no device will attach to, vs. a link which may be shared
 * by multiple active devices -- it is not optimal.
 *
 * If interrupt performance is that important, get an IO-APIC system
 * with a pin dedicated to each device.  Or for that matter, an MSI
 * enabled system.
 */

#define ACPI_MAX_IRQS		256
#define ACPI_MAX_ISA_IRQ	16

#define PIRQ_PENALTY_PCI_AVAILABLE	(0)
#define PIRQ_PENALTY_PCI_POSSIBLE	(16*16)
#define PIRQ_PENALTY_PCI_USING		(16*16*16)
#define PIRQ_PENALTY_ISA_TYPICAL	(16*16*16*16)
#define PIRQ_PENALTY_ISA_USED		(16*16*16*16*16)
#define PIRQ_PENALTY_ISA_ALWAYS		(16*16*16*16*16*16)

static int acpi_irq_penalty[ACPI_MAX_IRQS] = {
	PIRQ_PENALTY_ISA_ALWAYS,	/* IRQ0 timer */
	PIRQ_PENALTY_ISA_ALWAYS,	/* IRQ1 keyboard */
	PIRQ_PENALTY_ISA_ALWAYS,	/* IRQ2 cascade */
L
Len Brown 已提交
463 464
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ3 serial */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ4 serial */
L
Linus Torvalds 已提交
465 466 467 468 469 470 471 472 473 474 475
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ5 sometimes SoundBlaster */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ6 */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ7 parallel, spurious */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ8 rtc, sometimes */
	PIRQ_PENALTY_PCI_AVAILABLE,	/* IRQ9  PCI, often acpi */
	PIRQ_PENALTY_PCI_AVAILABLE,	/* IRQ10 PCI */
	PIRQ_PENALTY_PCI_AVAILABLE,	/* IRQ11 PCI */
	PIRQ_PENALTY_ISA_USED,	/* IRQ12 mouse */
	PIRQ_PENALTY_ISA_USED,	/* IRQ13 fpe, sometimes */
	PIRQ_PENALTY_ISA_USED,	/* IRQ14 ide0 */
	PIRQ_PENALTY_ISA_USED,	/* IRQ15 ide1 */
L
Len Brown 已提交
476
	/* >IRQ15 */
L
Linus Torvalds 已提交
477 478
};

L
Len Brown 已提交
479
int __init acpi_irq_penalty_init(void)
L
Linus Torvalds 已提交
480
{
L
Len Brown 已提交
481 482 483
	struct list_head *node = NULL;
	struct acpi_pci_link *link = NULL;
	int i = 0;
L
Linus Torvalds 已提交
484 485 486 487 488 489 490 491 492


	/*
	 * Update penalties to facilitate IRQ balancing.
	 */
	list_for_each(node, &acpi_link.entries) {

		link = list_entry(node, struct acpi_pci_link, node);
		if (!link) {
493
			printk(KERN_ERR PREFIX "Invalid link context\n");
L
Linus Torvalds 已提交
494 495 496 497 498 499 500 501
			continue;
		}

		/*
		 * reflect the possible and active irqs in the penalty table --
		 * useful for breaking ties.
		 */
		if (link->irq.possible_count) {
L
Len Brown 已提交
502 503 504
			int penalty =
			    PIRQ_PENALTY_PCI_POSSIBLE /
			    link->irq.possible_count;
L
Linus Torvalds 已提交
505 506 507

			for (i = 0; i < link->irq.possible_count; i++) {
				if (link->irq.possible[i] < ACPI_MAX_ISA_IRQ)
L
Len Brown 已提交
508 509 510
					acpi_irq_penalty[link->irq.
							 possible[i]] +=
					    penalty;
L
Linus Torvalds 已提交
511 512 513
			}

		} else if (link->irq.active) {
L
Len Brown 已提交
514 515
			acpi_irq_penalty[link->irq.active] +=
			    PIRQ_PENALTY_PCI_POSSIBLE;
L
Linus Torvalds 已提交
516 517 518 519 520
		}
	}
	/* Add a penalty for the SCI */
	acpi_irq_penalty[acpi_fadt.sci_int] += PIRQ_PENALTY_PCI_USING;

521
	return 0;
L
Linus Torvalds 已提交
522 523 524 525
}

static int acpi_irq_balance;	/* 0: static, 1: balance */

L
Len Brown 已提交
526
static int acpi_pci_link_allocate(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
527
{
L
Len Brown 已提交
528 529
	int irq;
	int i;
L
Linus Torvalds 已提交
530 531


532 533 534 535
	if (link->irq.initialized) {
		if (link->refcnt == 0)
			/* This means the link is disabled but initialized */
			acpi_pci_link_set(link, link->irq.active);
536
		return 0;
537
	}
L
Linus Torvalds 已提交
538 539 540 541 542 543 544 545 546 547 548 549 550

	/*
	 * search for active IRQ in list of possible IRQs.
	 */
	for (i = 0; i < link->irq.possible_count; ++i) {
		if (link->irq.active == link->irq.possible[i])
			break;
	}
	/*
	 * forget active IRQ that is not in possible list
	 */
	if (i == link->irq.possible_count) {
		if (acpi_strict)
551 552
			printk(KERN_WARNING PREFIX "_CRS %d not found"
				      " in _PRS\n", link->irq.active);
L
Linus Torvalds 已提交
553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570
		link->irq.active = 0;
	}

	/*
	 * if active found, use it; else pick entry from end of possible list.
	 */
	if (link->irq.active) {
		irq = link->irq.active;
	} else {
		irq = link->irq.possible[link->irq.possible_count - 1];
	}

	if (acpi_irq_balance || !link->irq.active) {
		/*
		 * Select the best IRQ.  This is done in reverse to promote
		 * the use of IRQs 9, 10, 11, and >15.
		 */
		for (i = (link->irq.possible_count - 1); i >= 0; i--) {
L
Len Brown 已提交
571 572
			if (acpi_irq_penalty[irq] >
			    acpi_irq_penalty[link->irq.possible[i]])
L
Linus Torvalds 已提交
573 574 575 576 577 578
				irq = link->irq.possible[i];
		}
	}

	/* Attempt to enable the link device at this IRQ. */
	if (acpi_pci_link_set(link, irq)) {
579 580
		printk(KERN_ERR PREFIX "Unable to set IRQ for %s [%s]. "
			    "Try pci=noacpi or acpi=off\n",
581
			    acpi_device_name(link->device),
582
			    acpi_device_bid(link->device));
583
		return -ENODEV;
L
Linus Torvalds 已提交
584 585
	} else {
		acpi_irq_penalty[link->irq.active] += PIRQ_PENALTY_PCI_USING;
L
Len Brown 已提交
586 587 588
		printk(PREFIX "%s [%s] enabled at IRQ %d\n",
		       acpi_device_name(link->device),
		       acpi_device_bid(link->device), link->irq.active);
L
Linus Torvalds 已提交
589 590 591 592
	}

	link->irq.initialized = 1;

593
	return 0;
L
Linus Torvalds 已提交
594 595 596
}

/*
597
 * acpi_pci_link_allocate_irq
L
Linus Torvalds 已提交
598 599 600 601 602
 * success: return IRQ >= 0
 * failure: return -1
 */

int
L
Len Brown 已提交
603 604
acpi_pci_link_allocate_irq(acpi_handle handle,
			   int index,
B
Bob Moore 已提交
605
			   int *triggering, int *polarity, char **name)
L
Linus Torvalds 已提交
606
{
L
Len Brown 已提交
607 608 609
	int result = 0;
	struct acpi_device *device = NULL;
	struct acpi_pci_link *link = NULL;
L
Linus Torvalds 已提交
610 611 612 613


	result = acpi_bus_get_device(handle, &device);
	if (result) {
614
		printk(KERN_ERR PREFIX "Invalid link device\n");
615
		return -1;
L
Linus Torvalds 已提交
616 617
	}

L
Len Brown 已提交
618
	link = (struct acpi_pci_link *)acpi_driver_data(device);
L
Linus Torvalds 已提交
619
	if (!link) {
620
		printk(KERN_ERR PREFIX "Invalid link context\n");
621
		return -1;
L
Linus Torvalds 已提交
622 623 624 625
	}

	/* TBD: Support multiple index (IRQ) entries per Link Device */
	if (index) {
626
		printk(KERN_ERR PREFIX "Invalid index %d\n", index);
627
		return -1;
L
Linus Torvalds 已提交
628 629
	}

I
Ingo Molnar 已提交
630
	mutex_lock(&acpi_link_lock);
631
	if (acpi_pci_link_allocate(link)) {
I
Ingo Molnar 已提交
632
		mutex_unlock(&acpi_link_lock);
633
		return -1;
634
	}
L
Len Brown 已提交
635

L
Linus Torvalds 已提交
636
	if (!link->irq.active) {
I
Ingo Molnar 已提交
637
		mutex_unlock(&acpi_link_lock);
638
		printk(KERN_ERR PREFIX "Link active IRQ is 0!\n");
639
		return -1;
L
Linus Torvalds 已提交
640
	}
L
Len Brown 已提交
641
	link->refcnt++;
I
Ingo Molnar 已提交
642
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
643

B
Bob Moore 已提交
644 645 646 647
	if (triggering)
		*triggering = link->irq.triggering;
	if (polarity)
		*polarity = link->irq.polarity;
L
Len Brown 已提交
648 649
	if (name)
		*name = acpi_device_bid(link->device);
650
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
651 652
			  "Link %s is referenced\n",
			  acpi_device_bid(link->device)));
653
	return (link->irq.active);
L
Linus Torvalds 已提交
654 655
}

656 657 658 659
/*
 * We don't change link's irq information here.  After it is reenabled, we
 * continue use the info
 */
L
Len Brown 已提交
660
int acpi_pci_link_free_irq(acpi_handle handle)
661
{
L
Len Brown 已提交
662 663 664
	struct acpi_device *device = NULL;
	struct acpi_pci_link *link = NULL;
	acpi_status result;
665 666 667 668


	result = acpi_bus_get_device(handle, &device);
	if (result) {
669
		printk(KERN_ERR PREFIX "Invalid link device\n");
670
		return -1;
671
	}
L
Linus Torvalds 已提交
672

L
Len Brown 已提交
673
	link = (struct acpi_pci_link *)acpi_driver_data(device);
674
	if (!link) {
675
		printk(KERN_ERR PREFIX "Invalid link context\n");
676
		return -1;
677 678
	}

I
Ingo Molnar 已提交
679
	mutex_lock(&acpi_link_lock);
680
	if (!link->irq.initialized) {
I
Ingo Molnar 已提交
681
		mutex_unlock(&acpi_link_lock);
682
		printk(KERN_ERR PREFIX "Link isn't initialized\n");
683
		return -1;
684
	}
685 686 687 688 689 690 691 692 693 694
#ifdef	FUTURE_USE
	/*
	 * The Link reference count allows us to _DISable an unused link
	 * and suspend time, and set it again  on resume.
	 * However, 2.6.12 still has irq_router.resume
	 * which blindly restores the link state.
	 * So we disable the reference count method
	 * to prevent duplicate acpi_pci_link_set()
	 * which would harm some systems
	 */
L
Len Brown 已提交
695
	link->refcnt--;
696
#endif
697
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
698 699
			  "Link %s is dereferenced\n",
			  acpi_device_bid(link->device)));
700 701 702 703

	if (link->refcnt == 0) {
		acpi_ut_evaluate_object(link->handle, "_DIS", 0, NULL);
	}
I
Ingo Molnar 已提交
704
	mutex_unlock(&acpi_link_lock);
705
	return (link->irq.active);
706
}
L
Len Brown 已提交
707

L
Linus Torvalds 已提交
708 709 710 711
/* --------------------------------------------------------------------------
                                 Driver Interface
   -------------------------------------------------------------------------- */

L
Len Brown 已提交
712
static int acpi_pci_link_add(struct acpi_device *device)
L
Linus Torvalds 已提交
713
{
L
Len Brown 已提交
714 715 716 717
	int result = 0;
	struct acpi_pci_link *link = NULL;
	int i = 0;
	int found = 0;
L
Linus Torvalds 已提交
718 719 720


	if (!device)
721
		return -EINVAL;
L
Linus Torvalds 已提交
722 723 724

	link = kmalloc(sizeof(struct acpi_pci_link), GFP_KERNEL);
	if (!link)
725
		return -ENOMEM;
L
Linus Torvalds 已提交
726 727 728 729 730 731 732 733
	memset(link, 0, sizeof(struct acpi_pci_link));

	link->device = device;
	link->handle = device->handle;
	strcpy(acpi_device_name(device), ACPI_PCI_LINK_DEVICE_NAME);
	strcpy(acpi_device_class(device), ACPI_PCI_LINK_CLASS);
	acpi_driver_data(device) = link;

I
Ingo Molnar 已提交
734
	mutex_lock(&acpi_link_lock);
L
Linus Torvalds 已提交
735 736 737 738 739 740 741 742
	result = acpi_pci_link_get_possible(link);
	if (result)
		goto end;

	/* query and set link->irq.active */
	acpi_pci_link_get_current(link);

	printk(PREFIX "%s [%s] (IRQs", acpi_device_name(device),
L
Len Brown 已提交
743
	       acpi_device_bid(device));
L
Linus Torvalds 已提交
744 745 746 747
	for (i = 0; i < link->irq.possible_count; i++) {
		if (link->irq.active == link->irq.possible[i]) {
			printk(" *%d", link->irq.possible[i]);
			found = 1;
L
Len Brown 已提交
748
		} else
L
Linus Torvalds 已提交
749 750 751 752 753 754 755 756
			printk(" %d", link->irq.possible[i]);
	}

	printk(")");

	if (!found)
		printk(" *%d", link->irq.active);

L
Len Brown 已提交
757
	if (!link->device->status.enabled)
L
Linus Torvalds 已提交
758 759 760 761 762 763 764 765
		printk(", disabled.");

	printk("\n");

	/* TBD: Acquire/release lock */
	list_add_tail(&link->node, &acpi_link.entries);
	acpi_link.count++;

L
Len Brown 已提交
766
      end:
L
Linus Torvalds 已提交
767 768
	/* disable all links -- to be activated on use */
	acpi_ut_evaluate_object(link->handle, "_DIS", 0, NULL);
I
Ingo Molnar 已提交
769
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
770 771 772 773

	if (result)
		kfree(link);

774
	return result;
L
Linus Torvalds 已提交
775 776
}

L
Len Brown 已提交
777
static int acpi_pci_link_resume(struct acpi_pci_link *link)
778 779 780
{

	if (link->refcnt && link->irq.active && link->irq.initialized)
781
		return (acpi_pci_link_set(link, link->irq.active));
782
	else
783
		return 0;
784 785
}

786 787 788 789 790
/*
 * FIXME: this is a workaround to avoid nasty warning.  It will be removed
 * after every device calls pci_disable_device in .resume.
 */
int acpi_in_resume;
L
Len Brown 已提交
791
static int irqrouter_resume(struct sys_device *dev)
L
Linus Torvalds 已提交
792
{
L
Len Brown 已提交
793 794
	struct list_head *node = NULL;
	struct acpi_pci_link *link = NULL;
L
Linus Torvalds 已提交
795 796


797 798 799
	/* Make sure SCI is enabled again (Apple firmware bug?) */
	acpi_set_register(ACPI_BITREG_SCI_ENABLE, 1, ACPI_MTX_DO_NOT_LOCK);

800
	acpi_in_resume = 1;
L
Linus Torvalds 已提交
801 802 803
	list_for_each(node, &acpi_link.entries) {
		link = list_entry(node, struct acpi_pci_link, node);
		if (!link) {
804
			printk(KERN_ERR PREFIX "Invalid link context\n");
L
Linus Torvalds 已提交
805 806
			continue;
		}
807
		acpi_pci_link_resume(link);
L
Linus Torvalds 已提交
808
	}
809
	acpi_in_resume = 0;
810
	return 0;
L
Linus Torvalds 已提交
811 812
}

L
Len Brown 已提交
813
static int acpi_pci_link_remove(struct acpi_device *device, int type)
L
Linus Torvalds 已提交
814 815 816 817 818
{
	struct acpi_pci_link *link = NULL;


	if (!device || !acpi_driver_data(device))
819
		return -EINVAL;
L
Linus Torvalds 已提交
820

L
Len Brown 已提交
821
	link = (struct acpi_pci_link *)acpi_driver_data(device);
L
Linus Torvalds 已提交
822

I
Ingo Molnar 已提交
823
	mutex_lock(&acpi_link_lock);
L
Linus Torvalds 已提交
824
	list_del(&link->node);
I
Ingo Molnar 已提交
825
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
826 827 828

	kfree(link);

829
	return 0;
L
Linus Torvalds 已提交
830 831 832 833 834 835 836 837 838 839 840 841 842
}

/*
 * modify acpi_irq_penalty[] from cmdline
 */
static int __init acpi_irq_penalty_update(char *str, int used)
{
	int i;

	for (i = 0; i < 16; i++) {
		int retval;
		int irq;

L
Len Brown 已提交
843
		retval = get_option(&str, &irq);
L
Linus Torvalds 已提交
844 845 846 847 848 849

		if (!retval)
			break;	/* no number found */

		if (irq < 0)
			continue;
L
Len Brown 已提交
850

L
Linus Torvalds 已提交
851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871
		if (irq >= ACPI_MAX_IRQS)
			continue;

		if (used)
			acpi_irq_penalty[irq] += PIRQ_PENALTY_ISA_USED;
		else
			acpi_irq_penalty[irq] = PIRQ_PENALTY_PCI_AVAILABLE;

		if (retval != 2)	/* no next number */
			break;
	}
	return 1;
}

/*
 * We'd like PNP to call this routine for the
 * single ISA_USED value for each legacy device.
 * But instead it calls us with each POSSIBLE setting.
 * There is no ISA_POSSIBLE weight, so we simply use
 * the (small) PCI_USING penalty.
 */
D
David Shaohua Li 已提交
872
void acpi_penalize_isa_irq(int irq, int active)
L
Linus Torvalds 已提交
873
{
D
David Shaohua Li 已提交
874 875 876 877
	if (active)
		acpi_irq_penalty[irq] += PIRQ_PENALTY_ISA_USED;
	else
		acpi_irq_penalty[irq] += PIRQ_PENALTY_PCI_USING;
L
Linus Torvalds 已提交
878 879 880 881 882 883 884 885 886 887 888
}

/*
 * Over-ride default table to reserve additional IRQs for use by ISA
 * e.g. acpi_irq_isa=5
 * Useful for telling ACPI how not to interfere with your ISA sound card.
 */
static int __init acpi_irq_isa(char *str)
{
	return acpi_irq_penalty_update(str, 1);
}
L
Len Brown 已提交
889

L
Linus Torvalds 已提交
890 891 892 893 894 895 896 897 898 899 900
__setup("acpi_irq_isa=", acpi_irq_isa);

/*
 * Over-ride default table to free additional IRQs for use by PCI
 * e.g. acpi_irq_pci=7,15
 * Used for acpi_irq_balance to free up IRQs to reduce PCI IRQ sharing.
 */
static int __init acpi_irq_pci(char *str)
{
	return acpi_irq_penalty_update(str, 0);
}
L
Len Brown 已提交
901

L
Linus Torvalds 已提交
902 903 904 905 906 907 908
__setup("acpi_irq_pci=", acpi_irq_pci);

static int __init acpi_irq_nobalance_set(char *str)
{
	acpi_irq_balance = 0;
	return 1;
}
L
Len Brown 已提交
909

L
Linus Torvalds 已提交
910 911 912 913 914 915 916 917
__setup("acpi_irq_nobalance", acpi_irq_nobalance_set);

int __init acpi_irq_balance_set(char *str)
{
	acpi_irq_balance = 1;
	return 1;
}

L
Len Brown 已提交
918
__setup("acpi_irq_balance", acpi_irq_balance_set);
L
Linus Torvalds 已提交
919

920
/* FIXME: we will remove this interface after all drivers call pci_disable_device */
L
Linus Torvalds 已提交
921
static struct sysdev_class irqrouter_sysdev_class = {
L
Len Brown 已提交
922 923
	set_kset_name("irqrouter"),
	.resume = irqrouter_resume,
L
Linus Torvalds 已提交
924 925 926
};

static struct sys_device device_irqrouter = {
L
Len Brown 已提交
927 928
	.id = 0,
	.cls = &irqrouter_sysdev_class,
L
Linus Torvalds 已提交
929 930 931 932 933 934 935 936
};

static int __init irqrouter_init_sysfs(void)
{
	int error;


	if (acpi_disabled || acpi_noirq)
937
		return 0;
L
Linus Torvalds 已提交
938 939 940 941 942

	error = sysdev_class_register(&irqrouter_sysdev_class);
	if (!error)
		error = sysdev_register(&device_irqrouter);

943
	return error;
L
Len Brown 已提交
944
}
L
Linus Torvalds 已提交
945 946 947

device_initcall(irqrouter_init_sysfs);

L
Len Brown 已提交
948
static int __init acpi_pci_link_init(void)
L
Linus Torvalds 已提交
949 950 951
{

	if (acpi_noirq)
952
		return 0;
L
Linus Torvalds 已提交
953 954 955 956 957

	acpi_link.count = 0;
	INIT_LIST_HEAD(&acpi_link.entries);

	if (acpi_bus_register_driver(&acpi_pci_link_driver) < 0)
958
		return -ENODEV;
L
Linus Torvalds 已提交
959

960
	return 0;
L
Linus Torvalds 已提交
961 962 963
}

subsys_initcall(acpi_pci_link_init);