pci_link.c 25.2 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

	ACPI_FUNCTION_TRACE("acpi_pci_link_check_possible");

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

	return_ACPI_STATUS(AE_CTRL_TERMINATE);
}

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

	ACPI_FUNCTION_TRACE("acpi_pci_link_get_possible");

	if (!link)
		return_VALUE(-EINVAL);

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

L
Len Brown 已提交
187 188 189
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
			  "Found %d possible IRQs\n",
			  link->irq.possible_count));
L
Linus Torvalds 已提交
190 191 192 193 194

	return_VALUE(0);
}

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

	ACPI_FUNCTION_TRACE("acpi_pci_link_check_current");

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

/*
 * Run _CRS and set link->irq.active
 *
 * return value:
 * 0 - success
 * !0 - failure
 */
L
Len Brown 已提交
249
static int acpi_pci_link_get_current(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
250
{
L
Len Brown 已提交
251 252 253
	int result = 0;
	acpi_status status = AE_OK;
	int irq = 0;
L
Linus Torvalds 已提交
254 255 256 257 258 259 260 261 262 263 264 265 266

	ACPI_FUNCTION_TRACE("acpi_pci_link_get_current");

	if (!link || !link->handle)
		return_VALUE(-EINVAL);

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

		if (!link->device->status.enabled) {
			ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Link disabled\n"));
			return_VALUE(0);
		}
	}

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

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

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

	link->irq.active = irq;

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

L
Len Brown 已提交
298
      end:
L
Linus Torvalds 已提交
299 300 301
	return_VALUE(result);
}

L
Len Brown 已提交
302
static int acpi_pci_link_set(struct acpi_pci_link *link, int irq)
L
Linus Torvalds 已提交
303
{
L
Len Brown 已提交
304 305
	int result = 0;
	acpi_status status = AE_OK;
L
Linus Torvalds 已提交
306
	struct {
L
Len Brown 已提交
307 308 309 310
		struct acpi_resource res;
		struct acpi_resource end;
	} *resource;
	struct acpi_buffer buffer = { 0, NULL };
L
Linus Torvalds 已提交
311 312 313 314 315 316

	ACPI_FUNCTION_TRACE("acpi_pci_link_set");

	if (!link || !irq)
		return_VALUE(-EINVAL);

317
	resource = kmalloc(sizeof(*resource) + 1, GFP_ATOMIC);
L
Len Brown 已提交
318
	if (!resource)
L
Linus Torvalds 已提交
319 320
		return_VALUE(-ENOMEM);

L
Len Brown 已提交
321 322
	memset(resource, 0, sizeof(*resource) + 1);
	buffer.length = sizeof(*resource) + 1;
L
Linus Torvalds 已提交
323 324
	buffer.pointer = resource;

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

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

	}
B
Bob Moore 已提交
365
	resource->end.type = ACPI_RESOURCE_TYPE_END_TAG;
L
Linus Torvalds 已提交
366 367 368 369 370 371

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

	/* check for total failure */
	if (ACPI_FAILURE(status)) {
372
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _SRS"));
L
Linus Torvalds 已提交
373 374 375 376 377 378 379
		result = -ENODEV;
		goto end;
	}

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

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

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

      end:
L
Linus Torvalds 已提交
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 463 464 465 466 467
	kfree(resource);
	return_VALUE(result);
}

/* --------------------------------------------------------------------------
                            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 已提交
468 469
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ3 serial */
	PIRQ_PENALTY_ISA_TYPICAL,	/* IRQ4 serial */
L
Linus Torvalds 已提交
470 471 472 473 474 475 476 477 478 479 480
	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 已提交
481
	/* >IRQ15 */
L
Linus Torvalds 已提交
482 483
};

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

	ACPI_FUNCTION_TRACE("acpi_irq_penalty_init");

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

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

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

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

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

	return_VALUE(0);
}

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

L
Len Brown 已提交
532
static int acpi_pci_link_allocate(struct acpi_pci_link *link)
L
Linus Torvalds 已提交
533
{
L
Len Brown 已提交
534 535
	int irq;
	int i;
L
Linus Torvalds 已提交
536 537 538

	ACPI_FUNCTION_TRACE("acpi_pci_link_allocate");

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

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

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

	link->irq.initialized = 1;

	return_VALUE(0);
}

/*
604
 * acpi_pci_link_allocate_irq
L
Linus Torvalds 已提交
605 606 607 608 609
 * success: return IRQ >= 0
 * failure: return -1
 */

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

618
	ACPI_FUNCTION_TRACE("acpi_pci_link_allocate_irq");
L
Linus Torvalds 已提交
619 620 621

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

L
Len Brown 已提交
626
	link = (struct acpi_pci_link *)acpi_driver_data(device);
L
Linus Torvalds 已提交
627
	if (!link) {
628
		printk(KERN_ERR PREFIX "Invalid link context\n");
L
Linus Torvalds 已提交
629 630 631 632 633
		return_VALUE(-1);
	}

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

I
Ingo Molnar 已提交
638
	mutex_lock(&acpi_link_lock);
639
	if (acpi_pci_link_allocate(link)) {
I
Ingo Molnar 已提交
640
		mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
641
		return_VALUE(-1);
642
	}
L
Len Brown 已提交
643

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

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

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

	ACPI_FUNCTION_TRACE("acpi_pci_link_free_irq");

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

L
Len Brown 已提交
682
	link = (struct acpi_pci_link *)acpi_driver_data(device);
683
	if (!link) {
684
		printk(KERN_ERR PREFIX "Invalid link context\n");
685 686 687
		return_VALUE(-1);
	}

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

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

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

L
Len Brown 已提交
721
static int acpi_pci_link_add(struct acpi_device *device)
L
Linus Torvalds 已提交
722
{
L
Len Brown 已提交
723 724 725 726
	int result = 0;
	struct acpi_pci_link *link = NULL;
	int i = 0;
	int found = 0;
L
Linus Torvalds 已提交
727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743

	ACPI_FUNCTION_TRACE("acpi_pci_link_add");

	if (!device)
		return_VALUE(-EINVAL);

	link = kmalloc(sizeof(struct acpi_pci_link), GFP_KERNEL);
	if (!link)
		return_VALUE(-ENOMEM);
	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 已提交
744
	mutex_lock(&acpi_link_lock);
L
Linus Torvalds 已提交
745 746 747 748 749 750 751 752
	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 已提交
753
	       acpi_device_bid(device));
L
Linus Torvalds 已提交
754 755 756 757
	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 已提交
758
		} else
L
Linus Torvalds 已提交
759 760 761 762 763 764 765 766
			printk(" %d", link->irq.possible[i]);
	}

	printk(")");

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

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

	printk("\n");

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

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

	if (result)
		kfree(link);

	return_VALUE(result);
}

L
Len Brown 已提交
787
static int acpi_pci_link_resume(struct acpi_pci_link *link)
788 789 790 791 792 793 794 795 796
{
	ACPI_FUNCTION_TRACE("acpi_pci_link_resume");

	if (link->refcnt && link->irq.active && link->irq.initialized)
		return_VALUE(acpi_pci_link_set(link, link->irq.active));
	else
		return_VALUE(0);
}

797 798 799 800 801
/*
 * 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 已提交
802
static int irqrouter_resume(struct sys_device *dev)
L
Linus Torvalds 已提交
803
{
L
Len Brown 已提交
804 805
	struct list_head *node = NULL;
	struct acpi_pci_link *link = NULL;
L
Linus Torvalds 已提交
806

807
	ACPI_FUNCTION_TRACE("irqrouter_resume");
L
Linus Torvalds 已提交
808

809 810 811
	/* Make sure SCI is enabled again (Apple firmware bug?) */
	acpi_set_register(ACPI_BITREG_SCI_ENABLE, 1, ACPI_MTX_DO_NOT_LOCK);

812
	acpi_in_resume = 1;
L
Linus Torvalds 已提交
813 814 815
	list_for_each(node, &acpi_link.entries) {
		link = list_entry(node, struct acpi_pci_link, node);
		if (!link) {
816
			printk(KERN_ERR PREFIX "Invalid link context\n");
L
Linus Torvalds 已提交
817 818
			continue;
		}
819
		acpi_pci_link_resume(link);
L
Linus Torvalds 已提交
820
	}
821
	acpi_in_resume = 0;
822
	return_VALUE(0);
L
Linus Torvalds 已提交
823 824
}

L
Len Brown 已提交
825
static int acpi_pci_link_remove(struct acpi_device *device, int type)
L
Linus Torvalds 已提交
826 827 828 829 830 831 832 833
{
	struct acpi_pci_link *link = NULL;

	ACPI_FUNCTION_TRACE("acpi_pci_link_remove");

	if (!device || !acpi_driver_data(device))
		return_VALUE(-EINVAL);

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

I
Ingo Molnar 已提交
836
	mutex_lock(&acpi_link_lock);
L
Linus Torvalds 已提交
837
	list_del(&link->node);
I
Ingo Molnar 已提交
838
	mutex_unlock(&acpi_link_lock);
L
Linus Torvalds 已提交
839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855

	kfree(link);

	return_VALUE(0);
}

/*
 * 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 已提交
856
		retval = get_option(&str, &irq);
L
Linus Torvalds 已提交
857 858 859 860 861 862

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

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

L
Linus Torvalds 已提交
864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884
		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 已提交
885
void acpi_penalize_isa_irq(int irq, int active)
L
Linus Torvalds 已提交
886
{
D
David Shaohua Li 已提交
887 888 889 890
	if (active)
		acpi_irq_penalty[irq] += PIRQ_PENALTY_ISA_USED;
	else
		acpi_irq_penalty[irq] += PIRQ_PENALTY_PCI_USING;
L
Linus Torvalds 已提交
891 892 893 894 895 896 897 898 899 900 901
}

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

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

L
Linus Torvalds 已提交
915 916 917 918 919 920 921
__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 已提交
922

L
Linus Torvalds 已提交
923 924 925 926 927 928 929 930
__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 已提交
931
__setup("acpi_irq_balance", acpi_irq_balance_set);
L
Linus Torvalds 已提交
932

933
/* FIXME: we will remove this interface after all drivers call pci_disable_device */
L
Linus Torvalds 已提交
934
static struct sysdev_class irqrouter_sysdev_class = {
L
Len Brown 已提交
935 936
	set_kset_name("irqrouter"),
	.resume = irqrouter_resume,
L
Linus Torvalds 已提交
937 938 939
};

static struct sys_device device_irqrouter = {
L
Len Brown 已提交
940 941
	.id = 0,
	.cls = &irqrouter_sysdev_class,
L
Linus Torvalds 已提交
942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957
};

static int __init irqrouter_init_sysfs(void)
{
	int error;

	ACPI_FUNCTION_TRACE("irqrouter_init_sysfs");

	if (acpi_disabled || acpi_noirq)
		return_VALUE(0);

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

	return_VALUE(error);
L
Len Brown 已提交
958
}
L
Linus Torvalds 已提交
959 960 961

device_initcall(irqrouter_init_sysfs);

L
Len Brown 已提交
962
static int __init acpi_pci_link_init(void)
L
Linus Torvalds 已提交
963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978
{
	ACPI_FUNCTION_TRACE("acpi_pci_link_init");

	if (acpi_noirq)
		return_VALUE(0);

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

	if (acpi_bus_register_driver(&acpi_pci_link_driver) < 0)
		return_VALUE(-ENODEV);

	return_VALUE(0);
}

subsys_initcall(acpi_pci_link_init);