Blame view

include/linux/msi.h 10.2 KB
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
1
2
  #ifndef LINUX_MSI_H
  #define LINUX_MSI_H
b50cac55b   Neil Horman   PCI/sysfs: add pe...
3
  #include <linux/kobject.h>
4aa9bc955   Michael Ellerman   MSI: Use a list i...
4
  #include <linux/list.h>
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
5
6
7
8
9
  struct msi_msg {
  	u32	address_lo;	/* low 32 bits of msi message address */
  	u32	address_hi;	/* high 32 bits of msi message address */
  	u32	data;		/* 16 bits of msi message data */
  };
38737d82f   Yijing Wang   PCI/MSI: Add pci_...
10
  extern int pci_msi_ignore_mask;
c54c18790   Satoru Takeuchi   PCI: cleanup MSI ...
11
  /* Helper functions */
1c9db5253   Thomas Gleixner   pci: Convert msi ...
12
  struct irq_data;
39431acb1   Thomas Gleixner   pci: Cleanup the ...
13
  struct msi_desc;
25a98bd4f   Jiang Liu   genirq/MSI: Store...
14
  struct pci_dev;
c09fcc4b2   Marc Zyngier   drivers/base: Add...
15
  struct platform_msi_priv_data;
2366d06eb   Bjorn Helgaas   PCI: Remove "exte...
16
  void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
2366d06eb   Bjorn Helgaas   PCI: Remove "exte...
17
  void get_cached_msi_msg(unsigned int irq, struct msi_msg *msg);
891d4a48f   Jiang Liu   PCI/MSI: Rename _...
18

c09fcc4b2   Marc Zyngier   drivers/base: Add...
19
20
21
22
23
24
25
26
27
28
29
30
  typedef void (*irq_write_msi_msg_t)(struct msi_desc *desc,
  				    struct msi_msg *msg);
  
  /**
   * platform_msi_desc - Platform device specific msi descriptor data
   * @msi_priv_data:	Pointer to platform private data
   * @msi_index:		The index of the MSI descriptor for multi MSI
   */
  struct platform_msi_desc {
  	struct platform_msi_priv_data	*msi_priv_data;
  	u16				msi_index;
  };
fc88419cf   Jiang Liu   genirq/MSI: Reorg...
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
  /**
   * struct msi_desc - Descriptor structure for MSI based interrupts
   * @list:	List head for management
   * @irq:	The base interrupt number
   * @nvec_used:	The number of vectors used
   * @dev:	Pointer to the device which uses this descriptor
   * @msg:	The last set MSI message cached for reuse
   *
   * @masked:	[PCI MSI/X] Mask bits
   * @is_msix:	[PCI MSI/X] True if MSI-X
   * @multiple:	[PCI MSI/X] log2 num of messages allocated
   * @multi_cap:	[PCI MSI/X] log2 num of messages supported
   * @maskbit:	[PCI MSI/X] Mask-Pending bit supported?
   * @is_64:	[PCI MSI/X] Address size: 0=32bit 1=64bit
   * @entry_nr:	[PCI MSI/X] Entry which is described by this descriptor
   * @default_irq:[PCI MSI/X] The default pre-assigned non-MSI irq
   * @mask_pos:	[PCI MSI]   Mask register position
   * @mask_base:	[PCI MSI-X] Mask register base address
c09fcc4b2   Marc Zyngier   drivers/base: Add...
49
   * @platform:	[platform]  Platform device specific msi descriptor data
fc88419cf   Jiang Liu   genirq/MSI: Reorg...
50
   */
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
51
  struct msi_desc {
fc88419cf   Jiang Liu   genirq/MSI: Reorg...
52
53
54
55
56
57
  	/* Shared device/bus type independent data */
  	struct list_head		list;
  	unsigned int			irq;
  	unsigned int			nvec_used;
  	struct device			*dev;
  	struct msi_msg			msg;
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
58

264d9caaa   Matthew Wilcox   PCI MSI: Use mask...
59
  	union {
fc88419cf   Jiang Liu   genirq/MSI: Reorg...
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
  		/* PCI MSI/X specific data */
  		struct {
  			u32 masked;
  			struct {
  				__u8	is_msix		: 1;
  				__u8	multiple	: 3;
  				__u8	multi_cap	: 3;
  				__u8	maskbit		: 1;
  				__u8	is_64		: 1;
  				__u16	entry_nr;
  				unsigned default_irq;
  			} msi_attrib;
  			union {
  				u8	mask_pos;
  				void __iomem *mask_base;
  			};
  		};
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
77

fc88419cf   Jiang Liu   genirq/MSI: Reorg...
78
79
80
81
82
83
84
  		/*
  		 * Non PCI variants add their data structure here. New
  		 * entries need to use a named structure. We want
  		 * proper name spaces for this. The PCI part is
  		 * anonymous for now as it would require an immediate
  		 * tree wide cleanup.
  		 */
c09fcc4b2   Marc Zyngier   drivers/base: Add...
85
  		struct platform_msi_desc platform;
fc88419cf   Jiang Liu   genirq/MSI: Reorg...
86
  	};
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
87
  };
d31eb3424   Jiang Liu   PCI/MSI: Introduc...
88
  /* Helpers to hide struct msi_desc implementation details */
25a98bd4f   Jiang Liu   genirq/MSI: Store...
89
  #define msi_desc_to_dev(desc)		((desc)->dev)
4a7cc8316   Jiang Liu   genirq/MSI: Move ...
90
  #define dev_to_msi_list(dev)		(&(dev)->msi_list)
d31eb3424   Jiang Liu   PCI/MSI: Introduc...
91
92
93
94
95
96
97
98
99
  #define first_msi_entry(dev)		\
  	list_first_entry(dev_to_msi_list((dev)), struct msi_desc, list)
  #define for_each_msi_entry(desc, dev)	\
  	list_for_each_entry((desc), dev_to_msi_list((dev)), list)
  
  #ifdef CONFIG_PCI_MSI
  #define first_pci_msi_entry(pdev)	first_msi_entry(&(pdev)->dev)
  #define for_each_pci_msi_entry(desc, pdev)	\
  	for_each_msi_entry((desc), &(pdev)->dev)
25a98bd4f   Jiang Liu   genirq/MSI: Store...
100
  struct pci_dev *msi_desc_to_pci_dev(struct msi_desc *desc);
c179c9b97   Jiang Liu   PCI: Add helper f...
101
102
103
104
105
106
  void *msi_desc_to_pci_sysdata(struct msi_desc *desc);
  #else /* CONFIG_PCI_MSI */
  static inline void *msi_desc_to_pci_sysdata(struct msi_desc *desc)
  {
  	return NULL;
  }
d31eb3424   Jiang Liu   PCI/MSI: Introduc...
107
  #endif /* CONFIG_PCI_MSI */
aa48b6f70   Jiang Liu   genirq/MSI: Move ...
108
109
  struct msi_desc *alloc_msi_entry(struct device *dev);
  void free_msi_entry(struct msi_desc *entry);
891d4a48f   Jiang Liu   PCI/MSI: Rename _...
110
  void __pci_read_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
83a18912b   Jiang Liu   PCI/MSI: Rename w...
111
112
  void __pci_write_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
  void pci_write_msi_msg(unsigned int irq, struct msi_msg *msg);
23ed8d57f   Thomas Gleixner   PCI/MSI: Rename m...
113
114
115
116
  u32 __pci_msix_desc_mask_irq(struct msi_desc *desc, u32 flag);
  u32 __pci_msi_desc_mask_irq(struct msi_desc *desc, u32 mask, u32 flag);
  void pci_msi_mask_irq(struct irq_data *data);
  void pci_msi_unmask_irq(struct irq_data *data);
83a18912b   Jiang Liu   PCI/MSI: Rename w...
117
118
119
120
121
122
123
124
125
  /* Conversion helpers. Should be removed after merging */
  static inline void __write_msi_msg(struct msi_desc *entry, struct msi_msg *msg)
  {
  	__pci_write_msi_msg(entry, msg);
  }
  static inline void write_msi_msg(int irq, struct msi_msg *msg)
  {
  	pci_write_msi_msg(irq, msg);
  }
23ed8d57f   Thomas Gleixner   PCI/MSI: Rename m...
126
127
128
129
130
131
132
133
  static inline void mask_msi_irq(struct irq_data *data)
  {
  	pci_msi_mask_irq(data);
  }
  static inline void unmask_msi_irq(struct irq_data *data)
  {
  	pci_msi_unmask_irq(data);
  }
891d4a48f   Jiang Liu   PCI/MSI: Rename _...
134

3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
135
  /*
4287d824f   Thomas Petazzoni   PCI: use weak fun...
136
137
138
   * The arch hooks to setup up msi irqs. Those functions are
   * implemented as weak symbols so that they /can/ be overriden by
   * architecture specific code if needed.
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
139
   */
f7feaca77   Eric W. Biederman   msi: Make MSI use...
140
  int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc);
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
141
  void arch_teardown_msi_irq(unsigned int irq);
2366d06eb   Bjorn Helgaas   PCI: Remove "exte...
142
143
  int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type);
  void arch_teardown_msi_irqs(struct pci_dev *dev);
ac8344c4c   DuanZhenzhong   PCI: Drop "irq" p...
144
  void arch_restore_msi_irqs(struct pci_dev *dev);
4287d824f   Thomas Petazzoni   PCI: use weak fun...
145
146
  
  void default_teardown_msi_irqs(struct pci_dev *dev);
ac8344c4c   DuanZhenzhong   PCI: Drop "irq" p...
147
  void default_restore_msi_irqs(struct pci_dev *dev);
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
148

c2791b806   Yijing Wang   PCI/MSI: Rename "...
149
  struct msi_controller {
0cbdcfcf4   Thierry Reding   PCI: Introduce ne...
150
151
  	struct module *owner;
  	struct device *dev;
0d5a6db3a   Thomas Petazzoni   of: pci: add regi...
152
153
  	struct device_node *of_node;
  	struct list_head list;
0cbdcfcf4   Thierry Reding   PCI: Introduce ne...
154

c2791b806   Yijing Wang   PCI/MSI: Rename "...
155
  	int (*setup_irq)(struct msi_controller *chip, struct pci_dev *dev,
0cbdcfcf4   Thierry Reding   PCI: Introduce ne...
156
  			 struct msi_desc *desc);
339e5b44e   Lucas Stach   PCI: Add msi_cont...
157
158
  	int (*setup_irqs)(struct msi_controller *chip, struct pci_dev *dev,
  			  int nvec, int type);
c2791b806   Yijing Wang   PCI/MSI: Rename "...
159
  	void (*teardown_irq)(struct msi_controller *chip, unsigned int irq);
0cbdcfcf4   Thierry Reding   PCI: Introduce ne...
160
  };
f3cf8bb0d   Jiang Liu   genirq: Add gener...
161
  #ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN
d9109698b   Jiang Liu   genirq: Introduce...
162

aeeb59657   Jiang Liu   genirq: Provide d...
163
  #include <linux/irqhandler.h>
d9109698b   Jiang Liu   genirq: Introduce...
164
  #include <asm/msi.h>
f3cf8bb0d   Jiang Liu   genirq: Add gener...
165
166
167
  struct irq_domain;
  struct irq_chip;
  struct device_node;
be5436c83   Marc Zyngier   irqdomain/msi: Us...
168
  struct fwnode_handle;
f3cf8bb0d   Jiang Liu   genirq: Add gener...
169
170
171
172
173
174
175
  struct msi_domain_info;
  
  /**
   * struct msi_domain_ops - MSI interrupt domain callbacks
   * @get_hwirq:		Retrieve the resulting hw irq number
   * @msi_init:		Domain specific init function for MSI interrupts
   * @msi_free:		Domain specific function to free a MSI interrupts
d9109698b   Jiang Liu   genirq: Introduce...
176
177
178
179
180
181
182
183
184
185
186
187
   * @msi_check:		Callback for verification of the domain/info/dev data
   * @msi_prepare:	Prepare the allocation of the interrupts in the domain
   * @msi_finish:		Optional callbacl to finalize the allocation
   * @set_desc:		Set the msi descriptor for an interrupt
   * @handle_error:	Optional error handler if the allocation fails
   *
   * @get_hwirq, @msi_init and @msi_free are callbacks used by
   * msi_create_irq_domain() and related interfaces
   *
   * @msi_check, @msi_prepare, @msi_finish, @set_desc and @handle_error
   * are callbacks used by msi_irq_domain_alloc_irqs() and related
   * interfaces which are based on msi_desc.
f3cf8bb0d   Jiang Liu   genirq: Add gener...
188
189
   */
  struct msi_domain_ops {
aeeb59657   Jiang Liu   genirq: Provide d...
190
191
  	irq_hw_number_t	(*get_hwirq)(struct msi_domain_info *info,
  				     msi_alloc_info_t *arg);
f3cf8bb0d   Jiang Liu   genirq: Add gener...
192
193
194
  	int		(*msi_init)(struct irq_domain *domain,
  				    struct msi_domain_info *info,
  				    unsigned int virq, irq_hw_number_t hwirq,
aeeb59657   Jiang Liu   genirq: Provide d...
195
  				    msi_alloc_info_t *arg);
f3cf8bb0d   Jiang Liu   genirq: Add gener...
196
197
198
  	void		(*msi_free)(struct irq_domain *domain,
  				    struct msi_domain_info *info,
  				    unsigned int virq);
d9109698b   Jiang Liu   genirq: Introduce...
199
200
201
202
203
204
205
206
207
208
209
  	int		(*msi_check)(struct irq_domain *domain,
  				     struct msi_domain_info *info,
  				     struct device *dev);
  	int		(*msi_prepare)(struct irq_domain *domain,
  				       struct device *dev, int nvec,
  				       msi_alloc_info_t *arg);
  	void		(*msi_finish)(msi_alloc_info_t *arg, int retval);
  	void		(*set_desc)(msi_alloc_info_t *arg,
  				    struct msi_desc *desc);
  	int		(*handle_error)(struct irq_domain *domain,
  					struct msi_desc *desc, int error);
f3cf8bb0d   Jiang Liu   genirq: Add gener...
210
211
212
213
  };
  
  /**
   * struct msi_domain_info - MSI interrupt domain data
aeeb59657   Jiang Liu   genirq: Provide d...
214
215
216
217
218
219
220
221
   * @flags:		Flags to decribe features and capabilities
   * @ops:		The callback data structure
   * @chip:		Optional: associated interrupt chip
   * @chip_data:		Optional: associated interrupt chip data
   * @handler:		Optional: associated interrupt flow handler
   * @handler_data:	Optional: associated interrupt flow handler data
   * @handler_name:	Optional: associated interrupt flow handler name
   * @data:		Optional: domain specific data
f3cf8bb0d   Jiang Liu   genirq: Add gener...
222
223
   */
  struct msi_domain_info {
aeeb59657   Jiang Liu   genirq: Provide d...
224
  	u32			flags;
f3cf8bb0d   Jiang Liu   genirq: Add gener...
225
226
  	struct msi_domain_ops	*ops;
  	struct irq_chip		*chip;
aeeb59657   Jiang Liu   genirq: Provide d...
227
228
229
230
  	void			*chip_data;
  	irq_flow_handler_t	handler;
  	void			*handler_data;
  	const char		*handler_name;
f3cf8bb0d   Jiang Liu   genirq: Add gener...
231
232
  	void			*data;
  };
aeeb59657   Jiang Liu   genirq: Provide d...
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
  /* Flags for msi_domain_info */
  enum {
  	/*
  	 * Init non implemented ops callbacks with default MSI domain
  	 * callbacks.
  	 */
  	MSI_FLAG_USE_DEF_DOM_OPS	= (1 << 0),
  	/*
  	 * Init non implemented chip callbacks with default MSI chip
  	 * callbacks.
  	 */
  	MSI_FLAG_USE_DEF_CHIP_OPS	= (1 << 1),
  	/* Build identity map between hwirq and irq */
  	MSI_FLAG_IDENTITY_MAP		= (1 << 2),
  	/* Support multiple PCI MSI interrupts */
  	MSI_FLAG_MULTI_PCI_MSI		= (1 << 3),
  	/* Support PCI MSIX interrupts */
  	MSI_FLAG_PCI_MSIX		= (1 << 4),
  };
f3cf8bb0d   Jiang Liu   genirq: Add gener...
252
253
  int msi_domain_set_affinity(struct irq_data *data, const struct cpumask *mask,
  			    bool force);
be5436c83   Marc Zyngier   irqdomain/msi: Us...
254
  struct irq_domain *msi_create_irq_domain(struct fwnode_handle *fwnode,
f3cf8bb0d   Jiang Liu   genirq: Add gener...
255
256
  					 struct msi_domain_info *info,
  					 struct irq_domain *parent);
d9109698b   Jiang Liu   genirq: Introduce...
257
258
259
  int msi_domain_alloc_irqs(struct irq_domain *domain, struct device *dev,
  			  int nvec);
  void msi_domain_free_irqs(struct irq_domain *domain, struct device *dev);
f3cf8bb0d   Jiang Liu   genirq: Add gener...
260
  struct msi_domain_info *msi_get_domain_info(struct irq_domain *domain);
be5436c83   Marc Zyngier   irqdomain/msi: Us...
261
  struct irq_domain *platform_msi_create_irq_domain(struct fwnode_handle *fwnode,
c09fcc4b2   Marc Zyngier   drivers/base: Add...
262
263
264
265
266
  						  struct msi_domain_info *info,
  						  struct irq_domain *parent);
  int platform_msi_domain_alloc_irqs(struct device *dev, unsigned int nvec,
  				   irq_write_msi_msg_t write_msi_msg);
  void platform_msi_domain_free_irqs(struct device *dev);
f3cf8bb0d   Jiang Liu   genirq: Add gener...
267
  #endif /* CONFIG_GENERIC_MSI_IRQ_DOMAIN */
3878eaefb   Jiang Liu   PCI/MSI: Enhance ...
268
269
  #ifdef CONFIG_PCI_MSI_IRQ_DOMAIN
  void pci_msi_domain_write_msg(struct irq_data *irq_data, struct msi_msg *msg);
be5436c83   Marc Zyngier   irqdomain/msi: Us...
270
  struct irq_domain *pci_msi_create_irq_domain(struct fwnode_handle *fwnode,
3878eaefb   Jiang Liu   PCI/MSI: Enhance ...
271
272
273
274
275
  					     struct msi_domain_info *info,
  					     struct irq_domain *parent);
  int pci_msi_domain_alloc_irqs(struct irq_domain *domain, struct pci_dev *dev,
  			      int nvec, int type);
  void pci_msi_domain_free_irqs(struct irq_domain *domain, struct pci_dev *dev);
be5436c83   Marc Zyngier   irqdomain/msi: Us...
276
  struct irq_domain *pci_msi_create_default_irq_domain(struct fwnode_handle *fwnode,
8e047adae   Jiang Liu   PCI/MSI: Provide ...
277
  		 struct msi_domain_info *info, struct irq_domain *parent);
3878eaefb   Jiang Liu   PCI/MSI: Enhance ...
278
279
280
281
  irq_hw_number_t pci_msi_domain_calc_hwirq(struct pci_dev *dev,
  					  struct msi_desc *desc);
  int pci_msi_domain_check_cap(struct irq_domain *domain,
  			     struct msi_domain_info *info, struct device *dev);
b6eec9b71   David Daney   PCI/MSI: Add help...
282
  u32 pci_msi_domain_get_msi_rid(struct irq_domain *domain, struct pci_dev *pdev);
54fa97eeb   Marc Zyngier   PCI/MSI: Allow th...
283
284
285
286
287
288
  struct irq_domain *pci_msi_get_device_domain(struct pci_dev *pdev);
  #else
  static inline struct irq_domain *pci_msi_get_device_domain(struct pci_dev *pdev)
  {
  	return NULL;
  }
3878eaefb   Jiang Liu   PCI/MSI: Enhance ...
289
  #endif /* CONFIG_PCI_MSI_IRQ_DOMAIN */
3b7d1921f   Eric W. Biederman   [PATCH] msi: refa...
290
  #endif /* LINUX_MSI_H */