summaryrefslogtreecommitdiff
path: root/src/southbridge/intel/common/acpi_pirq_gen.c
blob: 3ff591c1e59176c9fa3bb9983fda715232601368 (plain)
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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
/*
 * This file is part of the coreboot project.
 *
 * Copyright (C) 2017 Arthur Heymans <arthur@aheymans.xyz>
 *
 * 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; version 2 of the License.
 *
 * 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.
 */

#include <arch/acpigen.h>
#include <arch/io.h>
#include <console/console.h>
#include <device/pci_def.h>
#include <string.h>

#include "acpi_pirq_gen.h"

enum emit_type {
	EMIT_NONE,
	EMIT_APIC,
	EMIT_PICM,
};

static size_t enumerate_root_pci_pins(const enum emit_type emit,
				      const char *lpcb_path)
{
	char buffer[DEVICE_PATH_MAX];
	device_t dev;
	pci_pin_t prev_int_pin = PCI_INT_NONE;
	u8 prev_pci_dev = 0;
	size_t num_devs = 0;

	for (dev = all_devices; dev; dev = dev->next) {
		u8 pci_dev;
		u8 int_pin;
		pirq_t pirq;

		if (dev->path.type != DEVICE_PATH_PCI ||
		    dev->bus->secondary != 0)
			continue;

		pci_dev = PCI_SLOT(dev->path.pci.devfn);
		int_pin = pci_read_config8(dev, PCI_INTERRUPT_PIN);

		if (int_pin == PCI_INT_NONE || int_pin > PCI_INT_D)
			continue;

		pirq = intel_common_map_pirq(dev, int_pin);
		if (emit == EMIT_NONE)  /* Only print on the first pass */
			printk(BIOS_SPEW, "ACPI_PIRQ_GEN: %s: pin=%d pirq=%d\n",
				dev_path(dev), int_pin, pirq);

		if (pirq == PIRQ_NONE)
			continue;

		/* Avoid duplicate entries */
		if (prev_pci_dev == pci_dev && prev_int_pin == int_pin) {
			continue;
		} else {
			prev_int_pin = int_pin;
			prev_pci_dev = pci_dev;
		}
		if (emit != EMIT_NONE) {
			acpigen_write_package(4);
			acpigen_write_dword((pci_dev << 16) | 0xffff);
			acpigen_write_dword(int_pin - PCI_INT_A);
			if (emit == EMIT_APIC) {
				acpigen_write_dword(0);
				acpigen_write_dword(16 + (pirq - PIRQ_A));
			} else {
				snprintf(buffer, sizeof(buffer),
					"%s.LNK%c",
					lpcb_path, 'A' + pirq - PIRQ_A);
				acpigen_emit_namestring(buffer);
				acpigen_write_dword(0);
			}
			acpigen_pop_len();
		}
		num_devs++;
	}
	return num_devs;
}

void intel_acpi_gen_def_acpi_pirq(device_t dev)
{
	const char *lpcb_path = acpi_device_path(dev);
	const size_t num_devs = enumerate_root_pci_pins(EMIT_NONE, lpcb_path);

	if (!lpcb_path)
		die("ACPI_PIRQ_GEN: Missing LPCB ACPI path\n");

	acpigen_write_scope("\\_SB.PCI0");
	acpigen_write_method("_PRT", 0);
	acpigen_write_if();
	acpigen_emit_namestring("PICM");
	acpigen_emit_byte(RETURN_OP);
	acpigen_write_package(num_devs);
	enumerate_root_pci_pins(EMIT_APIC, lpcb_path);
	acpigen_pop_len(); /* package */
	acpigen_pop_len(); /* if PICM */
	acpigen_write_else();
	acpigen_emit_byte(RETURN_OP);
	acpigen_write_package(num_devs);
	enumerate_root_pci_pins(EMIT_PICM, lpcb_path);
	acpigen_pop_len(); /* package */
	acpigen_pop_len(); /* else PICM */
	acpigen_pop_len(); /* _PRT */
	acpigen_pop_len(); /* \_SB */
}