2 * Driver for ePAPR Embedded Hypervisor PIC
4 * Copyright 2008-2011 Freescale Semiconductor, Inc.
6 * Author: Ashish Kalra <ashish.kalra@freescale.com>
8 * This file is licensed under the terms of the GNU General Public License
9 * version 2. This program is licensed "as is" without any warranty of any
10 * kind, whether express or implied.
13 #include <linux/types.h>
14 #include <linux/kernel.h>
15 #include <linux/init.h>
16 #include <linux/irq.h>
17 #include <linux/smp.h>
18 #include <linux/interrupt.h>
19 #include <linux/slab.h>
20 #include <linux/spinlock.h>
22 #include <linux/of_address.h>
27 #include <asm/machdep.h>
28 #include <asm/ehv_pic.h>
29 #include <asm/fsl_hcalls.h>
31 static struct ehv_pic
*global_ehv_pic
;
32 static DEFINE_SPINLOCK(ehv_pic_lock
);
34 static u32 hwirq_intspec
[NR_EHV_PIC_INTS
];
35 static u32 __iomem
*mpic_percpu_base_vaddr
;
37 #define IRQ_TYPE_MPIC_DIRECT 4
38 #define MPIC_EOI 0x00B0
41 * Linux descriptor level callbacks
44 void ehv_pic_unmask_irq(struct irq_data
*d
)
46 unsigned int src
= virq_to_hw(d
->irq
);
48 ev_int_set_mask(src
, 0);
51 void ehv_pic_mask_irq(struct irq_data
*d
)
53 unsigned int src
= virq_to_hw(d
->irq
);
55 ev_int_set_mask(src
, 1);
58 void ehv_pic_end_irq(struct irq_data
*d
)
60 unsigned int src
= virq_to_hw(d
->irq
);
65 void ehv_pic_direct_end_irq(struct irq_data
*d
)
67 out_be32(mpic_percpu_base_vaddr
+ MPIC_EOI
/ 4, 0);
70 int ehv_pic_set_affinity(struct irq_data
*d
, const struct cpumask
*dest
,
73 unsigned int src
= virq_to_hw(d
->irq
);
74 unsigned int config
, prio
, cpu_dest
;
75 int cpuid
= irq_choose_cpu(dest
);
78 spin_lock_irqsave(&ehv_pic_lock
, flags
);
79 ev_int_get_config(src
, &config
, &prio
, &cpu_dest
);
80 ev_int_set_config(src
, config
, prio
, cpuid
);
81 spin_unlock_irqrestore(&ehv_pic_lock
, flags
);
83 return IRQ_SET_MASK_OK
;
86 static unsigned int ehv_pic_type_to_vecpri(unsigned int type
)
88 /* Now convert sense value */
90 switch (type
& IRQ_TYPE_SENSE_MASK
) {
91 case IRQ_TYPE_EDGE_RISING
:
92 return EHV_PIC_INFO(VECPRI_SENSE_EDGE
) |
93 EHV_PIC_INFO(VECPRI_POLARITY_POSITIVE
);
95 case IRQ_TYPE_EDGE_FALLING
:
96 case IRQ_TYPE_EDGE_BOTH
:
97 return EHV_PIC_INFO(VECPRI_SENSE_EDGE
) |
98 EHV_PIC_INFO(VECPRI_POLARITY_NEGATIVE
);
100 case IRQ_TYPE_LEVEL_HIGH
:
101 return EHV_PIC_INFO(VECPRI_SENSE_LEVEL
) |
102 EHV_PIC_INFO(VECPRI_POLARITY_POSITIVE
);
104 case IRQ_TYPE_LEVEL_LOW
:
106 return EHV_PIC_INFO(VECPRI_SENSE_LEVEL
) |
107 EHV_PIC_INFO(VECPRI_POLARITY_NEGATIVE
);
111 int ehv_pic_set_irq_type(struct irq_data
*d
, unsigned int flow_type
)
113 unsigned int src
= virq_to_hw(d
->irq
);
114 unsigned int vecpri
, vold
, vnew
, prio
, cpu_dest
;
117 if (flow_type
== IRQ_TYPE_NONE
)
118 flow_type
= IRQ_TYPE_LEVEL_LOW
;
120 irqd_set_trigger_type(d
, flow_type
);
122 vecpri
= ehv_pic_type_to_vecpri(flow_type
);
124 spin_lock_irqsave(&ehv_pic_lock
, flags
);
125 ev_int_get_config(src
, &vold
, &prio
, &cpu_dest
);
126 vnew
= vold
& ~(EHV_PIC_INFO(VECPRI_POLARITY_MASK
) |
127 EHV_PIC_INFO(VECPRI_SENSE_MASK
));
131 * TODO : Add specific interface call for platform to set
132 * individual interrupt priorities.
133 * platform currently using static/default priority for all ints
138 ev_int_set_config(src
, vecpri
, prio
, cpu_dest
);
140 spin_unlock_irqrestore(&ehv_pic_lock
, flags
);
141 return IRQ_SET_MASK_OK_NOCOPY
;
144 static struct irq_chip ehv_pic_irq_chip
= {
145 .irq_mask
= ehv_pic_mask_irq
,
146 .irq_unmask
= ehv_pic_unmask_irq
,
147 .irq_eoi
= ehv_pic_end_irq
,
148 .irq_set_type
= ehv_pic_set_irq_type
,
151 static struct irq_chip ehv_pic_direct_eoi_irq_chip
= {
152 .irq_mask
= ehv_pic_mask_irq
,
153 .irq_unmask
= ehv_pic_unmask_irq
,
154 .irq_eoi
= ehv_pic_direct_end_irq
,
155 .irq_set_type
= ehv_pic_set_irq_type
,
158 /* Return an interrupt vector or NO_IRQ if no interrupt is pending. */
159 unsigned int ehv_pic_get_irq(void)
163 BUG_ON(global_ehv_pic
== NULL
);
165 if (global_ehv_pic
->coreint_flag
)
166 irq
= mfspr(SPRN_EPR
); /* if core int mode */
168 ev_int_iack(0, &irq
); /* legacy mode */
170 if (irq
== 0xFFFF) /* 0xFFFF --> no irq is pending */
174 * this will also setup revmap[] in the slow path for the first
175 * time, next calls will always use fast path by indexing revmap
177 return irq_linear_revmap(global_ehv_pic
->irqhost
, irq
);
180 static int ehv_pic_host_match(struct irq_domain
*h
, struct device_node
*node
,
181 enum irq_domain_bus_token bus_token
)
183 /* Exact match, unless ehv_pic node is NULL */
184 return h
->of_node
== NULL
|| h
->of_node
== node
;
187 static int ehv_pic_host_map(struct irq_domain
*h
, unsigned int virq
,
190 struct ehv_pic
*ehv_pic
= h
->host_data
;
191 struct irq_chip
*chip
;
194 chip
= &ehv_pic
->hc_irq
;
196 if (mpic_percpu_base_vaddr
)
197 if (hwirq_intspec
[hw
] & IRQ_TYPE_MPIC_DIRECT
)
198 chip
= &ehv_pic_direct_eoi_irq_chip
;
200 irq_set_chip_data(virq
, chip
);
202 * using handle_fasteoi_irq as our irq handler, this will
203 * only call the eoi callback and suitable for the MPIC
204 * controller which set ISR/IPR automatically and clear the
205 * highest priority active interrupt in ISR/IPR when we do
208 irq_set_chip_and_handler(virq
, chip
, handle_fasteoi_irq
);
210 /* Set default irq type */
211 irq_set_irq_type(virq
, IRQ_TYPE_NONE
);
216 static int ehv_pic_host_xlate(struct irq_domain
*h
, struct device_node
*ct
,
217 const u32
*intspec
, unsigned int intsize
,
218 irq_hw_number_t
*out_hwirq
, unsigned int *out_flags
)
222 * interrupt sense values coming from the guest device tree
223 * interrupt specifiers can have four possible sense and
224 * level encoding information and they need to
225 * be translated between firmware type & linux type.
228 static unsigned char map_of_senses_to_linux_irqtype
[4] = {
229 IRQ_TYPE_EDGE_FALLING
,
230 IRQ_TYPE_EDGE_RISING
,
235 *out_hwirq
= intspec
[0];
237 hwirq_intspec
[intspec
[0]] = intspec
[1];
238 *out_flags
= map_of_senses_to_linux_irqtype
[intspec
[1] &
239 ~IRQ_TYPE_MPIC_DIRECT
];
241 *out_flags
= IRQ_TYPE_NONE
;
247 static const struct irq_domain_ops ehv_pic_host_ops
= {
248 .match
= ehv_pic_host_match
,
249 .map
= ehv_pic_host_map
,
250 .xlate
= ehv_pic_host_xlate
,
253 void __init
ehv_pic_init(void)
255 struct device_node
*np
, *np2
;
256 struct ehv_pic
*ehv_pic
;
257 int coreint_flag
= 1;
259 np
= of_find_compatible_node(NULL
, NULL
, "epapr,hv-pic");
261 pr_err("ehv_pic_init: could not find epapr,hv-pic node\n");
265 if (!of_find_property(np
, "has-external-proxy", NULL
))
268 ehv_pic
= kzalloc(sizeof(struct ehv_pic
), GFP_KERNEL
);
274 ehv_pic
->irqhost
= irq_domain_add_linear(np
, NR_EHV_PIC_INTS
,
275 &ehv_pic_host_ops
, ehv_pic
);
276 if (!ehv_pic
->irqhost
) {
282 np2
= of_find_compatible_node(NULL
, NULL
, "fsl,hv-mpic-per-cpu");
284 mpic_percpu_base_vaddr
= of_iomap(np2
, 0);
285 if (!mpic_percpu_base_vaddr
)
286 pr_err("ehv_pic_init: of_iomap failed\n");
291 ehv_pic
->hc_irq
= ehv_pic_irq_chip
;
292 ehv_pic
->hc_irq
.irq_set_affinity
= ehv_pic_set_affinity
;
293 ehv_pic
->coreint_flag
= coreint_flag
;
295 global_ehv_pic
= ehv_pic
;
296 irq_set_default_host(global_ehv_pic
->irqhost
);