Blame view

drivers/irqchip/irq-gic-v3-its-pci-msi.c 4.69 KB
f130420e5   Marc Zyngier   irqchip/gicv3-its...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
  /*
   * Copyright (C) 2013-2015 ARM Limited, All Rights Reserved.
   * Author: Marc Zyngier <marc.zyngier@arm.com>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   *
   * 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, see <http://www.gnu.org/licenses/>.
   */
723344dd0   Tomasz Nowicki   irqchip/gicv3-its...
17
  #include <linux/acpi_iort.h>
f130420e5   Marc Zyngier   irqchip/gicv3-its...
18
19
20
21
  #include <linux/msi.h>
  #include <linux/of.h>
  #include <linux/of_irq.h>
  #include <linux/of_pci.h>
f130420e5   Marc Zyngier   irqchip/gicv3-its...
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
  static void its_mask_msi_irq(struct irq_data *d)
  {
  	pci_msi_mask_irq(d);
  	irq_chip_mask_parent(d);
  }
  
  static void its_unmask_msi_irq(struct irq_data *d)
  {
  	pci_msi_unmask_irq(d);
  	irq_chip_unmask_parent(d);
  }
  
  static struct irq_chip its_msi_irq_chip = {
  	.name			= "ITS-MSI",
  	.irq_unmask		= its_unmask_msi_irq,
  	.irq_mask		= its_mask_msi_irq,
  	.irq_eoi		= irq_chip_eoi_parent,
  	.irq_write_msi_msg	= pci_msi_domain_write_msg,
  };
  
  struct its_pci_alias {
  	struct pci_dev	*pdev;
f130420e5   Marc Zyngier   irqchip/gicv3-its...
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
  	u32		count;
  };
  
  static int its_pci_msi_vec_count(struct pci_dev *pdev)
  {
  	int msi, msix;
  
  	msi = max(pci_msi_vec_count(pdev), 0);
  	msix = max(pci_msix_vec_count(pdev), 0);
  
  	return max(msi, msix);
  }
  
  static int its_get_pci_alias(struct pci_dev *pdev, u16 alias, void *data)
  {
  	struct its_pci_alias *dev_alias = data;
f130420e5   Marc Zyngier   irqchip/gicv3-its...
60
  	if (pdev != dev_alias->pdev)
791c76d58   Marc Zyngier   irqchip/gic-v3-it...
61
  		dev_alias->count += its_pci_msi_vec_count(pdev);
f130420e5   Marc Zyngier   irqchip/gicv3-its...
62
63
64
65
66
67
68
69
70
  
  	return 0;
  }
  
  static int its_pci_msi_prepare(struct irq_domain *domain, struct device *dev,
  			       int nvec, msi_alloc_info_t *info)
  {
  	struct pci_dev *pdev;
  	struct its_pci_alias dev_alias;
54456db9a   Marc Zyngier   irqchip/gicv3-its...
71
  	struct msi_domain_info *msi_info;
f130420e5   Marc Zyngier   irqchip/gicv3-its...
72
73
74
  
  	if (!dev_is_pci(dev))
  		return -EINVAL;
54456db9a   Marc Zyngier   irqchip/gicv3-its...
75
  	msi_info = msi_get_domain_info(domain->parent);
f130420e5   Marc Zyngier   irqchip/gicv3-its...
76
77
78
79
80
  	pdev = to_pci_dev(dev);
  	dev_alias.pdev = pdev;
  	dev_alias.count = nvec;
  
  	pci_for_each_dma_alias(pdev, its_get_pci_alias, &dev_alias);
54456db9a   Marc Zyngier   irqchip/gicv3-its...
81
  	/* ITS specific DeviceID, as the core ITS ignores dev. */
ccf91e68a   David Daney   irqchip/gic-v3-it...
82
  	info->scratchpad[0].ul = pci_msi_domain_get_msi_rid(domain, pdev);
54456db9a   Marc Zyngier   irqchip/gicv3-its...
83
84
85
  
  	return msi_info->ops->msi_prepare(domain->parent,
  					  dev, dev_alias.count, info);
f130420e5   Marc Zyngier   irqchip/gicv3-its...
86
87
88
89
90
91
92
93
94
95
96
97
  }
  
  static struct msi_domain_ops its_pci_msi_ops = {
  	.msi_prepare	= its_pci_msi_prepare,
  };
  
  static struct msi_domain_info its_pci_msi_domain_info = {
  	.flags	= (MSI_FLAG_USE_DEF_DOM_OPS | MSI_FLAG_USE_DEF_CHIP_OPS |
  		   MSI_FLAG_MULTI_PCI_MSI | MSI_FLAG_PCI_MSIX),
  	.ops	= &its_pci_msi_ops,
  	.chip	= &its_msi_irq_chip,
  };
54456db9a   Marc Zyngier   irqchip/gicv3-its...
98
99
100
101
  static struct of_device_id its_device_id[] = {
  	{	.compatible	= "arm,gic-v3-its",	},
  	{},
  };
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
102
103
  static int __init its_pci_msi_init_one(struct fwnode_handle *handle,
  				       const char *name)
f130420e5   Marc Zyngier   irqchip/gicv3-its...
104
  {
54456db9a   Marc Zyngier   irqchip/gicv3-its...
105
  	struct irq_domain *parent;
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
  	parent = irq_find_matching_fwnode(handle, DOMAIN_BUS_NEXUS);
  	if (!parent || !msi_get_domain_info(parent)) {
  		pr_err("%s: Unable to locate ITS domain
  ", name);
  		return -ENXIO;
  	}
  
  	if (!pci_msi_create_irq_domain(handle, &its_pci_msi_domain_info,
  				       parent)) {
  		pr_err("%s: Unable to create PCI domain
  ", name);
  		return -ENOMEM;
  	}
  
  	return 0;
  }
  
  static int __init its_pci_of_msi_init(void)
  {
  	struct device_node *np;
54456db9a   Marc Zyngier   irqchip/gicv3-its...
126
127
128
129
  	for (np = of_find_matching_node(NULL, its_device_id); np;
  	     np = of_find_matching_node(np, its_device_id)) {
  		if (!of_property_read_bool(np, "msi-controller"))
  			continue;
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
130
  		if (its_pci_msi_init_one(of_node_to_fwnode(np), np->full_name))
54456db9a   Marc Zyngier   irqchip/gicv3-its...
131
  			continue;
54456db9a   Marc Zyngier   irqchip/gicv3-its...
132
133
134
135
136
137
  
  		pr_info("PCI/MSI: %s domain created
  ", np->full_name);
  	}
  
  	return 0;
f130420e5   Marc Zyngier   irqchip/gicv3-its...
138
  }
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
139

723344dd0   Tomasz Nowicki   irqchip/gicv3-its...
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
  #ifdef CONFIG_ACPI
  
  static int __init
  its_pci_msi_parse_madt(struct acpi_subtable_header *header,
  		       const unsigned long end)
  {
  	struct acpi_madt_generic_translator *its_entry;
  	struct fwnode_handle *dom_handle;
  	const char *node_name;
  	int err = -ENXIO;
  
  	its_entry = (struct acpi_madt_generic_translator *)header;
  	node_name = kasprintf(GFP_KERNEL, "ITS@0x%lx",
  			      (long)its_entry->base_address);
  	dom_handle = iort_find_domain_token(its_entry->translation_id);
  	if (!dom_handle) {
  		pr_err("%s: Unable to locate ITS domain handle
  ", node_name);
  		goto out;
  	}
  
  	err = its_pci_msi_init_one(dom_handle, node_name);
  	if (!err)
  		pr_info("PCI/MSI: %s domain created
  ", node_name);
  
  out:
  	kfree(node_name);
  	return err;
  }
  
  static int __init its_pci_acpi_msi_init(void)
  {
  	acpi_table_parse_madt(ACPI_MADT_TYPE_GENERIC_TRANSLATOR,
  			      its_pci_msi_parse_madt, 0);
  	return 0;
  }
  #else
  static int __init its_pci_acpi_msi_init(void)
  {
  	return 0;
  }
  #endif
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
183
184
185
  static int __init its_pci_msi_init(void)
  {
  	its_pci_of_msi_init();
723344dd0   Tomasz Nowicki   irqchip/gicv3-its...
186
  	its_pci_acpi_msi_init();
db744aaa2   Tomasz Nowicki   irqchip/gicv3-its...
187
188
189
  
  	return 0;
  }
54456db9a   Marc Zyngier   irqchip/gicv3-its...
190
  early_initcall(its_pci_msi_init);