pci_link.c 24.0 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
47
ACPI_MODULE_NAME("pci_link");
L
Linus Torvalds 已提交
48 49 50 51 52
#define ACPI_PCI_LINK_CLASS		"pci_irq_routing"
#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 已提交
53 54
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 已提交
55

56 57 58 59 60 61
static struct acpi_device_id link_device_ids[] = {
	{"PNP0C0F", 0},
	{"", 0},
};
MODULE_DEVICE_TABLE(acpi, link_device_ids);

L
Linus Torvalds 已提交
62
static struct acpi_driver acpi_pci_link_driver = {
L
Len Brown 已提交
63
	.name = "pci_link",
L
Len Brown 已提交
64
	.class = ACPI_PCI_LINK_CLASS,
65
	.ids = link_device_ids,
L
Len Brown 已提交
66 67 68 69
	.ops = {
		.add = acpi_pci_link_add,
		.remove = acpi_pci_link_remove,
		},
L
Linus Torvalds 已提交
70 71
};

72 73 74 75
/*
 * 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 已提交
76
struct acpi_pci_link_irq {
L
Len Brown 已提交
77
	u8 active;		/* Current IRQ */
B
Bob Moore 已提交
78 79
	u8 triggering;		/* All IRQs */
	u8 polarity;	/* All IRQs */
L
Len Brown 已提交
80 81 82 83 84
	u8 resource_type;
	u8 possible_count;
	u8 possible[ACPI_PCI_LINK_MAX_POSSIBLE];
	u8 initialized:1;
	u8 reserved:7;
L
Linus Torvalds 已提交
85 86 87
};

struct acpi_pci_link {
L
Len Brown 已提交
88 89
	struct list_head node;
	struct acpi_device *device;
L
Linus Torvalds 已提交
90
	struct acpi_pci_link_irq irq;
L
Len Brown 已提交
91
	int refcnt;
L
Linus Torvalds 已提交
92 93 94
};

static struct {
L
Len Brown 已提交
95 96 97
	int count;
	struct list_head entries;
} acpi_link;
A
Adrian Bunk 已提交
98
static DEFINE_MUTEX(acpi_link_lock);
L
Linus Torvalds 已提交
99 100 101 102 103 104 105 106 107

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

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


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

175
	return AE_CTRL_TERMINATE;
L
Linus Torvalds 已提交
176 177
}

L
Len Brown 已提交
178
static int acpi_pci_link_get_possible(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
179
{
L
Len Brown 已提交
180
	acpi_status status;
L
Linus Torvalds 已提交
181 182 183


	if (!link)
184
		return -EINVAL;
L
Linus Torvalds 已提交
185

186
	status = acpi_walk_resources(link->device->handle, METHOD_NAME__PRS,
L
Len Brown 已提交
187
				     acpi_pci_link_check_possible, link);
L
Linus Torvalds 已提交
188
	if (ACPI_FAILURE(status)) {
189
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _PRS"));
190
		return -ENODEV;
L
Linus Torvalds 已提交
191 192
	}

L
Len Brown 已提交
193 194 195
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
			  "Found %d possible IRQs\n",
			  link->irq.possible_count));
L
Linus Torvalds 已提交
196

197
	return 0;
L
Linus Torvalds 已提交
198 199 200
}

static acpi_status
L
Len Brown 已提交
201
acpi_pci_link_check_current(struct acpi_resource *resource, void *context)
L
Linus Torvalds 已提交
202
{
L
Len Brown 已提交
203
	int *irq = (int *)context;
L
Linus Torvalds 已提交
204 205


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

248
	return AE_CTRL_TERMINATE;
L
Linus Torvalds 已提交
249 250 251 252 253 254 255 256 257
}

/*
 * Run _CRS and set link->irq.active
 *
 * return value:
 * 0 - success
 * !0 - failure
 */
L
Len Brown 已提交
258
static int acpi_pci_link_get_current(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
259
{
L
Len Brown 已提交
260 261 262
	int result = 0;
	acpi_status status = AE_OK;
	int irq = 0;
L
Linus Torvalds 已提交
263

264
	if (!link)
265
		return -EINVAL;
L
Linus Torvalds 已提交
266 267 268 269 270 271 272 273

	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) {
274
			printk(KERN_ERR PREFIX "Unable to read status\n");
L
Linus Torvalds 已提交
275 276 277 278 279
			goto end;
		}

		if (!link->device->status.enabled) {
			ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Link disabled\n"));
280
			return 0;
L
Linus Torvalds 已提交
281 282 283 284 285 286 287
		}
	}

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

288
	status = acpi_walk_resources(link->device->handle, METHOD_NAME__CRS,
L
Len Brown 已提交
289
				     acpi_pci_link_check_current, &irq);
L
Linus Torvalds 已提交
290
	if (ACPI_FAILURE(status)) {
291
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _CRS"));
L
Linus Torvalds 已提交
292 293 294 295 296
		result = -ENODEV;
		goto end;
	}

	if (acpi_strict && !irq) {
297
		printk(KERN_ERR PREFIX "_CRS returned 0\n");
L
Linus Torvalds 已提交
298 299 300 301 302 303 304
		result = -ENODEV;
	}

	link->irq.active = irq;

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

L
Len Brown 已提交
305
      end:
306
	return result;
L
Linus Torvalds 已提交
307 308
}

L
Len Brown 已提交
309
static int acpi_pci_link_set(struct acpi_pci_link *link, int irq)
L
Linus Torvalds 已提交
310
{
L
Len Brown 已提交
311 312
	int result = 0;
	acpi_status status = AE_OK;
L
Linus Torvalds 已提交
313
	struct {
L
Len Brown 已提交
314 315 316 317
		struct acpi_resource res;
		struct acpi_resource end;
	} *resource;
	struct acpi_buffer buffer = { 0, NULL };
L
Linus Torvalds 已提交
318 319 320


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

323
	resource = kzalloc(sizeof(*resource) + 1, irqs_disabled() ? GFP_ATOMIC: GFP_KERNEL);
L
Len Brown 已提交
324
	if (!resource)
325
		return -ENOMEM;
L
Linus Torvalds 已提交
326

L
Len Brown 已提交
327
	buffer.length = sizeof(*resource) + 1;
L
Linus Torvalds 已提交
328 329
	buffer.pointer = resource;

L
Len Brown 已提交
330
	switch (link->irq.resource_type) {
B
Bob Moore 已提交
331 332
	case ACPI_RESOURCE_TYPE_IRQ:
		resource->res.type = ACPI_RESOURCE_TYPE_IRQ;
L
Linus Torvalds 已提交
333
		resource->res.length = sizeof(struct acpi_resource);
B
Bob Moore 已提交
334 335 336 337 338
		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 已提交
339
			    ACPI_EXCLUSIVE;
L
Linus Torvalds 已提交
340
		else
B
Bob Moore 已提交
341 342
			resource->res.data.irq.sharable = ACPI_SHARED;
		resource->res.data.irq.interrupt_count = 1;
L
Linus Torvalds 已提交
343 344
		resource->res.data.irq.interrupts[0] = irq;
		break;
L
Len Brown 已提交
345

B
Bob Moore 已提交
346 347
	case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
		resource->res.type = ACPI_RESOURCE_TYPE_EXTENDED_IRQ;
L
Linus Torvalds 已提交
348
		resource->res.length = sizeof(struct acpi_resource);
L
Len Brown 已提交
349 350
		resource->res.data.extended_irq.producer_consumer =
		    ACPI_CONSUMER;
B
Bob Moore 已提交
351 352 353 354 355 356
		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 已提交
357
			    ACPI_EXCLUSIVE;
L
Linus Torvalds 已提交
358
		else
B
Bob Moore 已提交
359 360
			resource->res.data.irq.sharable = ACPI_SHARED;
		resource->res.data.extended_irq.interrupt_count = 1;
L
Linus Torvalds 已提交
361 362 363 364
		resource->res.data.extended_irq.interrupts[0] = irq;
		/* ignore resource_source, it's optional */
		break;
	default:
365
		printk(KERN_ERR PREFIX "Invalid Resource_type %d\n", link->irq.resource_type);
L
Linus Torvalds 已提交
366 367 368 369
		result = -EINVAL;
		goto end;

	}
B
Bob Moore 已提交
370
	resource->end.type = ACPI_RESOURCE_TYPE_END_TAG;
L
Linus Torvalds 已提交
371 372

	/* Attempt to set the resource */
373
	status = acpi_set_current_resources(link->device->handle, &buffer);
L
Linus Torvalds 已提交
374 375 376

	/* check for total failure */
	if (ACPI_FAILURE(status)) {
377
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _SRS"));
L
Linus Torvalds 已提交
378 379 380 381 382 383 384
		result = -ENODEV;
		goto end;
	}

	/* Query _STA, set device->status */
	result = acpi_bus_get_status(link->device);
	if (result) {
385
		printk(KERN_ERR PREFIX "Unable to read status\n");
L
Linus Torvalds 已提交
386 387 388
		goto end;
	}
	if (!link->device->status.enabled) {
389 390
		printk(KERN_WARNING PREFIX
			      "%s [%s] disabled and referenced, BIOS bug\n",
391
			      acpi_device_name(link->device),
392
			      acpi_device_bid(link->device));
L
Linus Torvalds 已提交
393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409
	}

	/* 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.
		 */
410 411
		printk(KERN_WARNING PREFIX
			      "%s [%s] BIOS reported IRQ %d, using IRQ %d\n",
412
			      acpi_device_name(link->device),
413
			      acpi_device_bid(link->device), link->irq.active, irq);
L
Linus Torvalds 已提交
414 415 416 417
		link->irq.active = irq;
	}

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

      end:
L
Linus Torvalds 已提交
420
	kfree(resource);
421
	return result;
L
Linus Torvalds 已提交
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 463 464 465 466 467 468 469 470 471 472
}

/* --------------------------------------------------------------------------
                            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 已提交
473 474
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ3 serial */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ4 serial */
L
Linus Torvalds 已提交
475 476 477 478 479 480 481 482 483 484 485
	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 已提交
486
	/* >IRQ15 */
L
Linus Torvalds 已提交
487 488
};

L
Len Brown 已提交
489
int __init acpi_irq_penalty_init(void)
L
Linus Torvalds 已提交
490
{
L
Len Brown 已提交
491 492 493
	struct list_head *node = NULL;
	struct acpi_pci_link *link = NULL;
	int i = 0;
L
Linus Torvalds 已提交
494 495 496 497 498 499 500 501 502


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

		link = list_entry(node, struct acpi_pci_link, node);
		if (!link) {
503
			printk(KERN_ERR PREFIX "Invalid link context\n");
L
Linus Torvalds 已提交
504 505 506 507 508 509 510 511
			continue;
		}

		/*
		 * reflect the possible and active irqs in the penalty table --
		 * useful for breaking ties.
		 */
		if (link->irq.possible_count) {
L
Len Brown 已提交
512 513 514
			int penalty =
			    PIRQ_PENALTY_PCI_POSSIBLE /
			    link->irq.possible_count;
L
Linus Torvalds 已提交
515 516 517

			for (i = 0; i < link->irq.possible_count; i++) {
				if (link->irq.possible[i] < ACPI_MAX_ISA_IRQ)
L
Len Brown 已提交
518 519 520
					acpi_irq_penalty[link->irq.
							 possible[i]] +=
					    penalty;
L
Linus Torvalds 已提交
521 522 523
			}

		} else if (link->irq.active) {
L
Len Brown 已提交
524 525
			acpi_irq_penalty[link->irq.active] +=
			    PIRQ_PENALTY_PCI_POSSIBLE;
L
Linus Torvalds 已提交
526 527 528
		}
	}
	/* Add a penalty for the SCI */
529
	acpi_irq_penalty[acpi_gbl_FADT.sci_interrupt] += PIRQ_PENALTY_PCI_USING;
L
Linus Torvalds 已提交
530

531
	return 0;
L
Linus Torvalds 已提交
532 533 534 535
}

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

L
Len Brown 已提交
536
static int acpi_pci_link_allocate(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
537
{
L
Len Brown 已提交
538 539
	int irq;
	int i;
L
Linus Torvalds 已提交
540 541


542 543 544 545
	if (link->irq.initialized) {
		if (link->refcnt == 0)
			/* This means the link is disabled but initialized */
			acpi_pci_link_set(link, link->irq.active);
546
		return 0;
547
	}
L
Linus Torvalds 已提交
548 549 550 551 552 553 554 555 556 557 558 559 560

	/*
	 * 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)
561 562
			printk(KERN_WARNING PREFIX "_CRS %d not found"
				      " in _PRS\n", link->irq.active);
L
Linus Torvalds 已提交
563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580
		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 已提交
581 582
			if (acpi_irq_penalty[irq] >
			    acpi_irq_penalty[link->irq.possible[i]])
L
Linus Torvalds 已提交
583 584 585 586 587 588
				irq = link->irq.possible[i];
		}
	}

	/* Attempt to enable the link device at this IRQ. */
	if (acpi_pci_link_set(link, irq)) {
589 590
		printk(KERN_ERR PREFIX "Unable to set IRQ for %s [%s]. "
			    "Try pci=noacpi or acpi=off\n",
591
			    acpi_device_name(link->device),
592
			    acpi_device_bid(link->device));
593
		return -ENODEV;
L
Linus Torvalds 已提交
594 595
	} else {
		acpi_irq_penalty[link->irq.active] += PIRQ_PENALTY_PCI_USING;
L
Len Brown 已提交
596 597 598
		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 已提交
599 600 601 602
	}

	link->irq.initialized = 1;

603
	return 0;
L
Linus Torvalds 已提交
604 605 606
}

/*
607
 * acpi_pci_link_allocate_irq
L
Linus Torvalds 已提交
608 609 610 611 612
 * success: return IRQ >= 0
 * failure: return -1
 */

int
L
Len Brown 已提交
613 614
acpi_pci_link_allocate_irq(acpi_handle handle,
			   int index,
B
Bob Moore 已提交
615
			   int *triggering, int *polarity, char **name)
L
Linus Torvalds 已提交
616
{
L
Len Brown 已提交
617 618 619
	int result = 0;
	struct acpi_device *device = NULL;
	struct acpi_pci_link *link = NULL;
L
Linus Torvalds 已提交
620 621 622 623


	result = acpi_bus_get_device(handle, &device);
	if (result) {
624
		printk(KERN_ERR PREFIX "Invalid link device\n");
625
		return -1;
L
Linus Torvalds 已提交
626 627
	}

628
	link = acpi_driver_data(device);
L
Linus Torvalds 已提交
629
	if (!link) {
630
		printk(KERN_ERR PREFIX "Invalid link context\n");
631
		return -1;
L
Linus Torvalds 已提交
632 633 634 635
	}

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

I
Ingo Molnar 已提交
640
	mutex_lock(&acpi_link_lock);
641
	if (acpi_pci_link_allocate(link)) {
I
Ingo Molnar 已提交
642
		mutex_unlock(&acpi_link_lock);
643
		return -1;
644
	}
L
Len Brown 已提交
645

L
Linus Torvalds 已提交
646
	if (!link->irq.active) {
I
Ingo Molnar 已提交
647
		mutex_unlock(&acpi_link_lock);
648
		printk(KERN_ERR PREFIX "Link active IRQ is 0!\n");
649
		return -1;
L
Linus Torvalds 已提交
650
	}
L
Len Brown 已提交
651
	link->refcnt++;
I
Ingo Molnar 已提交
652
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
653

B
Bob Moore 已提交
654 655 656 657
	if (triggering)
		*triggering = link->irq.triggering;
	if (polarity)
		*polarity = link->irq.polarity;
L
Len Brown 已提交
658 659
	if (name)
		*name = acpi_device_bid(link->device);
660
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
661 662
			  "Link %s is referenced\n",
			  acpi_device_bid(link->device)));
663
	return (link->irq.active);
L
Linus Torvalds 已提交
664 665
}

666 667 668 669
/*
 * We don't change link's irq information here.  After it is reenabled, we
 * continue use the info
 */
L
Len Brown 已提交
670
int acpi_pci_link_free_irq(acpi_handle handle)
671
{
L
Len Brown 已提交
672 673 674
	struct acpi_device *device = NULL;
	struct acpi_pci_link *link = NULL;
	acpi_status result;
675 676 677 678


	result = acpi_bus_get_device(handle, &device);
	if (result) {
679
		printk(KERN_ERR PREFIX "Invalid link device\n");
680
		return -1;
681
	}
L
Linus Torvalds 已提交
682

683
	link = acpi_driver_data(device);
684
	if (!link) {
685
		printk(KERN_ERR PREFIX "Invalid link context\n");
686
		return -1;
687 688
	}

I
Ingo Molnar 已提交
689
	mutex_lock(&acpi_link_lock);
690
	if (!link->irq.initialized) {
I
Ingo Molnar 已提交
691
		mutex_unlock(&acpi_link_lock);
692
		printk(KERN_ERR PREFIX "Link isn't initialized\n");
693
		return -1;
694
	}
695 696 697 698 699 700 701 702 703 704
#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 已提交
705
	link->refcnt--;
706
#endif
707
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
708 709
			  "Link %s is dereferenced\n",
			  acpi_device_bid(link->device)));
710 711

	if (link->refcnt == 0) {
712
		acpi_ut_evaluate_object(link->device->handle, "_DIS", 0, NULL);
713
	}
I
Ingo Molnar 已提交
714
	mutex_unlock(&acpi_link_lock);
715
	return (link->irq.active);
716
}
L
Len Brown 已提交
717

L
Linus Torvalds 已提交
718 719 720 721
/* --------------------------------------------------------------------------
                                 Driver Interface
   -------------------------------------------------------------------------- */

L
Len Brown 已提交
722
static int acpi_pci_link_add(struct acpi_device *device)
L
Linus Torvalds 已提交
723
{
L
Len Brown 已提交
724 725 726 727
	int result = 0;
	struct acpi_pci_link *link = NULL;
	int i = 0;
	int found = 0;
L
Linus Torvalds 已提交
728 729 730


	if (!device)
731
		return -EINVAL;
L
Linus Torvalds 已提交
732

733
	link = kzalloc(sizeof(struct acpi_pci_link), GFP_KERNEL);
L
Linus Torvalds 已提交
734
	if (!link)
735
		return -ENOMEM;
L
Linus Torvalds 已提交
736 737 738 739 740 741

	link->device = device;
	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 已提交
742
	mutex_lock(&acpi_link_lock);
L
Linus Torvalds 已提交
743 744 745 746 747 748 749
	result = acpi_pci_link_get_possible(link);
	if (result)
		goto end;

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

750
	printk(KERN_INFO PREFIX "%s [%s] (IRQs", acpi_device_name(device),
L
Len Brown 已提交
751
	       acpi_device_bid(device));
L
Linus Torvalds 已提交
752 753 754 755
	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 已提交
756
		} else
L
Linus Torvalds 已提交
757 758 759 760 761 762 763 764
			printk(" %d", link->irq.possible[i]);
	}

	printk(")");

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

L
Len Brown 已提交
765
	if (!link->device->status.enabled)
L
Linus Torvalds 已提交
766 767 768 769 770 771 772 773
		printk(", disabled.");

	printk("\n");

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

L
Len Brown 已提交
774
      end:
L
Linus Torvalds 已提交
775
	/* disable all links -- to be activated on use */
776
	acpi_ut_evaluate_object(device->handle, "_DIS", 0, NULL);
I
Ingo Molnar 已提交
777
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
778 779 780 781

	if (result)
		kfree(link);

782
	return result;
L
Linus Torvalds 已提交
783 784
}

L
Len Brown 已提交
785
static int acpi_pci_link_resume(struct acpi_pci_link *link)
786 787 788
{

	if (link->refcnt && link->irq.active && link->irq.initialized)
789
		return (acpi_pci_link_set(link, link->irq.active));
790
	else
791
		return 0;
792 793
}

L
Len Brown 已提交
794
static int irqrouter_resume(struct sys_device *dev)
L
Linus Torvalds 已提交
795
{
L
Len Brown 已提交
796 797
	struct list_head *node = NULL;
	struct acpi_pci_link *link = NULL;
L
Linus Torvalds 已提交
798 799


800
	/* Make sure SCI is enabled again (Apple firmware bug?) */
801
	acpi_set_register(ACPI_BITREG_SCI_ENABLE, 1);
802

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

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


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

822
	link = acpi_driver_data(device);
L
Linus Torvalds 已提交
823

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

	kfree(link);

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

/*
 * 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 已提交
844
		retval = get_option(&str, &irq);
L
Linus Torvalds 已提交
845 846 847 848 849 850

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

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

852
		if (irq >= ARRAY_SIZE(acpi_irq_penalty))
L
Linus Torvalds 已提交
853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872
			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 已提交
873
void acpi_penalize_isa_irq(int irq, int active)
L
Linus Torvalds 已提交
874
{
875 876 877 878 879 880
	if (irq >= 0 && irq < ARRAY_SIZE(acpi_irq_penalty)) {
		if (active)
			acpi_irq_penalty[irq] += PIRQ_PENALTY_ISA_USED;
		else
			acpi_irq_penalty[irq] += PIRQ_PENALTY_PCI_USING;
	}
L
Linus Torvalds 已提交
881 882 883 884 885 886 887 888 889 890 891
}

/*
 * 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 已提交
892

L
Linus Torvalds 已提交
893 894 895 896 897 898 899 900 901 902 903
__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 已提交
904

L
Linus Torvalds 已提交
905 906 907 908 909 910 911
__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 已提交
912

L
Linus Torvalds 已提交
913 914 915 916 917 918 919 920
__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 已提交
921
__setup("acpi_irq_balance", acpi_irq_balance_set);
L
Linus Torvalds 已提交
922

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

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

static int __init irqrouter_init_sysfs(void)
{
	int error;


	if (acpi_disabled || acpi_noirq)
940
		return 0;
L
Linus Torvalds 已提交
941 942 943 944 945

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

946
	return error;
L
Len Brown 已提交
947
}
L
Linus Torvalds 已提交
948 949 950

device_initcall(irqrouter_init_sysfs);

L
Len Brown 已提交
951
static int __init acpi_pci_link_init(void)
L
Linus Torvalds 已提交
952 953 954
{

	if (acpi_noirq)
955
		return 0;
L
Linus Torvalds 已提交
956 957 958 959 960

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

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

963
	return 0;
L
Linus Torvalds 已提交
964 965 966
}

subsys_initcall(acpi_pci_link_init);