blob: 886ebf6f698d9197de51e2d1e2772b2d25c8ae66 [file] [log] [blame]
Scott Woodb823f982013-04-12 14:08:43 +00001/*
2 * OpenPIC emulation
3 *
4 * Copyright (c) 2004 Jocelyn Mayer
5 * 2011 Alexander Graf
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy
8 * of this software and associated documentation files (the "Software"), to deal
9 * in the Software without restriction, including without limitation the rights
10 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 * copies of the Software, and to permit persons to whom the Software is
12 * furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in
15 * all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
23 * THE SOFTWARE.
24 */
Scott Woodb823f982013-04-12 14:08:43 +000025
Scott Wood5df554a2013-04-12 14:08:46 +000026#include <linux/slab.h>
27#include <linux/mutex.h>
28#include <linux/kvm_host.h>
29#include <linux/errno.h>
30#include <linux/fs.h>
31#include <linux/anon_inodes.h>
32#include <asm/uaccess.h>
33#include <asm/mpic.h>
34#include <asm/kvm_para.h>
35#include <asm/kvm_host.h>
36#include <asm/kvm_ppc.h>
37#include "iodev.h"
38
Scott Woodb823f982013-04-12 14:08:43 +000039#define MAX_CPU 32
40#define MAX_SRC 256
41#define MAX_TMR 4
42#define MAX_IPI 4
43#define MAX_MSI 8
44#define MAX_IRQ (MAX_SRC + MAX_IPI + MAX_TMR)
45#define VID 0x03 /* MPIC version ID */
46
47/* OpenPIC capability flags */
48#define OPENPIC_FLAG_IDR_CRIT (1 << 0)
49#define OPENPIC_FLAG_ILR (2 << 0)
50
51/* OpenPIC address map */
Scott Wood5df554a2013-04-12 14:08:46 +000052#define OPENPIC_REG_SIZE 0x40000
Scott Woodb823f982013-04-12 14:08:43 +000053#define OPENPIC_GLB_REG_START 0x0
54#define OPENPIC_GLB_REG_SIZE 0x10F0
55#define OPENPIC_TMR_REG_START 0x10F0
56#define OPENPIC_TMR_REG_SIZE 0x220
57#define OPENPIC_MSI_REG_START 0x1600
58#define OPENPIC_MSI_REG_SIZE 0x200
Scott Woodf0f5c482013-04-12 14:08:45 +000059#define OPENPIC_SUMMARY_REG_START 0x3800
60#define OPENPIC_SUMMARY_REG_SIZE 0x800
Scott Woodb823f982013-04-12 14:08:43 +000061#define OPENPIC_SRC_REG_START 0x10000
62#define OPENPIC_SRC_REG_SIZE (MAX_SRC * 0x20)
63#define OPENPIC_CPU_REG_START 0x20000
Scott Woodf0f5c482013-04-12 14:08:45 +000064#define OPENPIC_CPU_REG_SIZE (0x100 + ((MAX_CPU - 1) * 0x1000))
Scott Woodb823f982013-04-12 14:08:43 +000065
Scott Woodf0f5c482013-04-12 14:08:45 +000066struct fsl_mpic_info {
Scott Woodb823f982013-04-12 14:08:43 +000067 int max_ext;
Scott Woodf0f5c482013-04-12 14:08:45 +000068};
Scott Woodb823f982013-04-12 14:08:43 +000069
Scott Woodf0f5c482013-04-12 14:08:45 +000070static struct fsl_mpic_info fsl_mpic_20 = {
Scott Woodb823f982013-04-12 14:08:43 +000071 .max_ext = 12,
72};
73
Scott Woodf0f5c482013-04-12 14:08:45 +000074static struct fsl_mpic_info fsl_mpic_42 = {
Scott Woodb823f982013-04-12 14:08:43 +000075 .max_ext = 12,
76};
77
78#define FRR_NIRQ_SHIFT 16
79#define FRR_NCPU_SHIFT 8
80#define FRR_VID_SHIFT 0
81
82#define VID_REVISION_1_2 2
83#define VID_REVISION_1_3 3
84
85#define VIR_GENERIC 0x00000000 /* Generic Vendor ID */
86
87#define GCR_RESET 0x80000000
88#define GCR_MODE_PASS 0x00000000
89#define GCR_MODE_MIXED 0x20000000
90#define GCR_MODE_PROXY 0x60000000
91
92#define TBCR_CI 0x80000000 /* count inhibit */
93#define TCCR_TOG 0x80000000 /* toggles when decrement to zero */
94
95#define IDR_EP_SHIFT 31
96#define IDR_EP_MASK (1 << IDR_EP_SHIFT)
97#define IDR_CI0_SHIFT 30
98#define IDR_CI1_SHIFT 29
99#define IDR_P1_SHIFT 1
100#define IDR_P0_SHIFT 0
101
102#define ILR_INTTGT_MASK 0x000000ff
103#define ILR_INTTGT_INT 0x00
104#define ILR_INTTGT_CINT 0x01 /* critical */
105#define ILR_INTTGT_MCP 0x02 /* machine check */
Scott Wood5df554a2013-04-12 14:08:46 +0000106#define NUM_OUTPUTS 3
Scott Woodb823f982013-04-12 14:08:43 +0000107
Scott Woodb823f982013-04-12 14:08:43 +0000108#define MSIIR_OFFSET 0x140
109#define MSIIR_SRS_SHIFT 29
110#define MSIIR_SRS_MASK (0x7 << MSIIR_SRS_SHIFT)
111#define MSIIR_IBS_SHIFT 24
112#define MSIIR_IBS_MASK (0x1f << MSIIR_IBS_SHIFT)
113
114static int get_current_cpu(void)
115{
Scott Wood5df554a2013-04-12 14:08:46 +0000116#if defined(CONFIG_KVM) && defined(CONFIG_BOOKE)
117 struct kvm_vcpu *vcpu = current->thread.kvm_vcpu;
Scott Woodeb1e4f42013-04-12 14:08:47 +0000118 return vcpu ? vcpu->arch.irq_cpu_id : -1;
Scott Wood5df554a2013-04-12 14:08:46 +0000119#else
120 /* XXX */
121 return -1;
122#endif
Scott Woodb823f982013-04-12 14:08:43 +0000123}
124
Scott Wood5df554a2013-04-12 14:08:46 +0000125static int openpic_cpu_write_internal(void *opaque, gpa_t addr,
126 u32 val, int idx);
127static int openpic_cpu_read_internal(void *opaque, gpa_t addr,
128 u32 *ptr, int idx);
Scott Woodb823f982013-04-12 14:08:43 +0000129
Scott Woodf0f5c482013-04-12 14:08:45 +0000130enum irq_type {
Scott Woodb823f982013-04-12 14:08:43 +0000131 IRQ_TYPE_NORMAL = 0,
132 IRQ_TYPE_FSLINT, /* FSL internal interrupt -- level only */
133 IRQ_TYPE_FSLSPECIAL, /* FSL timer/IPI interrupt, edge, no polarity */
Scott Woodf0f5c482013-04-12 14:08:45 +0000134};
Scott Woodb823f982013-04-12 14:08:43 +0000135
Scott Woodf0f5c482013-04-12 14:08:45 +0000136struct irq_queue {
Scott Woodb823f982013-04-12 14:08:43 +0000137 /* Round up to the nearest 64 IRQs so that the queue length
138 * won't change when moving between 32 and 64 bit hosts.
139 */
140 unsigned long queue[BITS_TO_LONGS((MAX_IRQ + 63) & ~63)];
141 int next;
142 int priority;
Scott Woodf0f5c482013-04-12 14:08:45 +0000143};
Scott Woodb823f982013-04-12 14:08:43 +0000144
Scott Woodf0f5c482013-04-12 14:08:45 +0000145struct irq_source {
Scott Woodb823f982013-04-12 14:08:43 +0000146 uint32_t ivpr; /* IRQ vector/priority register */
147 uint32_t idr; /* IRQ destination register */
148 uint32_t destmask; /* bitmap of CPU destinations */
149 int last_cpu;
Scott Wood5df554a2013-04-12 14:08:46 +0000150 int output; /* IRQ level, e.g. ILR_INTTGT_INT */
Scott Woodb823f982013-04-12 14:08:43 +0000151 int pending; /* TRUE if IRQ is pending */
Scott Woodf0f5c482013-04-12 14:08:45 +0000152 enum irq_type type;
Scott Woodb823f982013-04-12 14:08:43 +0000153 bool level:1; /* level-triggered */
Scott Woodf0f5c482013-04-12 14:08:45 +0000154 bool nomask:1; /* critical interrupts ignore mask on some FSL MPICs */
155};
Scott Woodb823f982013-04-12 14:08:43 +0000156
157#define IVPR_MASK_SHIFT 31
158#define IVPR_MASK_MASK (1 << IVPR_MASK_SHIFT)
159#define IVPR_ACTIVITY_SHIFT 30
160#define IVPR_ACTIVITY_MASK (1 << IVPR_ACTIVITY_SHIFT)
161#define IVPR_MODE_SHIFT 29
162#define IVPR_MODE_MASK (1 << IVPR_MODE_SHIFT)
163#define IVPR_POLARITY_SHIFT 23
164#define IVPR_POLARITY_MASK (1 << IVPR_POLARITY_SHIFT)
165#define IVPR_SENSE_SHIFT 22
166#define IVPR_SENSE_MASK (1 << IVPR_SENSE_SHIFT)
167
168#define IVPR_PRIORITY_MASK (0xF << 16)
169#define IVPR_PRIORITY(_ivprr_) ((int)(((_ivprr_) & IVPR_PRIORITY_MASK) >> 16))
170#define IVPR_VECTOR(opp, _ivprr_) ((_ivprr_) & (opp)->vector_mask)
171
172/* IDR[EP/CI] are only for FSL MPIC prior to v4.0 */
173#define IDR_EP 0x80000000 /* external pin */
174#define IDR_CI 0x40000000 /* critical interrupt */
175
Scott Woodf0f5c482013-04-12 14:08:45 +0000176struct irq_dest {
Scott Wood5df554a2013-04-12 14:08:46 +0000177 struct kvm_vcpu *vcpu;
178
Scott Woodb823f982013-04-12 14:08:43 +0000179 int32_t ctpr; /* CPU current task priority */
Scott Woodf0f5c482013-04-12 14:08:45 +0000180 struct irq_queue raised;
181 struct irq_queue servicing;
Scott Woodb823f982013-04-12 14:08:43 +0000182
183 /* Count of IRQ sources asserting on non-INT outputs */
Scott Wood5df554a2013-04-12 14:08:46 +0000184 uint32_t outputs_active[NUM_OUTPUTS];
Scott Woodf0f5c482013-04-12 14:08:45 +0000185};
Scott Woodb823f982013-04-12 14:08:43 +0000186
Scott Wood398d8782013-04-30 14:57:13 +0000187#define MAX_MMIO_REGIONS 10
188
Scott Woodf0f5c482013-04-12 14:08:45 +0000189struct openpic {
Scott Wood5df554a2013-04-12 14:08:46 +0000190 struct kvm *kvm;
191 struct kvm_device *dev;
192 struct kvm_io_device mmio;
Scott Wood398d8782013-04-30 14:57:13 +0000193 const struct mem_reg *mmio_regions[MAX_MMIO_REGIONS];
194 int num_mmio_regions;
Scott Wood5df554a2013-04-12 14:08:46 +0000195 atomic_t users;
Scott Wood5df554a2013-04-12 14:08:46 +0000196
197 gpa_t reg_base;
198 spinlock_t lock;
199
Scott Woodb823f982013-04-12 14:08:43 +0000200 /* Behavior control */
Scott Woodf0f5c482013-04-12 14:08:45 +0000201 struct fsl_mpic_info *fsl;
Scott Woodb823f982013-04-12 14:08:43 +0000202 uint32_t model;
203 uint32_t flags;
204 uint32_t nb_irqs;
205 uint32_t vid;
206 uint32_t vir; /* Vendor identification register */
207 uint32_t vector_mask;
208 uint32_t tfrr_reset;
209 uint32_t ivpr_reset;
210 uint32_t idr_reset;
211 uint32_t brr1;
212 uint32_t mpic_mode_mask;
213
Scott Woodb823f982013-04-12 14:08:43 +0000214 /* Global registers */
215 uint32_t frr; /* Feature reporting register */
216 uint32_t gcr; /* Global configuration register */
217 uint32_t pir; /* Processor initialization register */
218 uint32_t spve; /* Spurious vector register */
219 uint32_t tfrr; /* Timer frequency reporting register */
220 /* Source registers */
Scott Woodf0f5c482013-04-12 14:08:45 +0000221 struct irq_source src[MAX_IRQ];
Scott Woodb823f982013-04-12 14:08:43 +0000222 /* Local registers per output pin */
Scott Woodf0f5c482013-04-12 14:08:45 +0000223 struct irq_dest dst[MAX_CPU];
Scott Woodb823f982013-04-12 14:08:43 +0000224 uint32_t nb_cpus;
225 /* Timer registers */
226 struct {
227 uint32_t tccr; /* Global timer current count register */
228 uint32_t tbcr; /* Global timer base count register */
229 } timers[MAX_TMR];
230 /* Shared MSI registers */
231 struct {
232 uint32_t msir; /* Shared Message Signaled Interrupt Register */
233 } msi[MAX_MSI];
234 uint32_t max_irq;
235 uint32_t irq_ipi0;
236 uint32_t irq_tim0;
237 uint32_t irq_msi;
Scott Woodf0f5c482013-04-12 14:08:45 +0000238};
Scott Woodb823f982013-04-12 14:08:43 +0000239
Scott Wood5df554a2013-04-12 14:08:46 +0000240
241static void mpic_irq_raise(struct openpic *opp, struct irq_dest *dst,
242 int output)
243{
244 struct kvm_interrupt irq = {
245 .irq = KVM_INTERRUPT_SET_LEVEL,
246 };
247
248 if (!dst->vcpu) {
249 pr_debug("%s: destination cpu %d does not exist\n",
250 __func__, (int)(dst - &opp->dst[0]));
251 return;
252 }
253
Scott Woodeb1e4f42013-04-12 14:08:47 +0000254 pr_debug("%s: cpu %d output %d\n", __func__, dst->vcpu->arch.irq_cpu_id,
Scott Wood5df554a2013-04-12 14:08:46 +0000255 output);
256
257 if (output != ILR_INTTGT_INT) /* TODO */
258 return;
259
260 kvm_vcpu_ioctl_interrupt(dst->vcpu, &irq);
261}
262
263static void mpic_irq_lower(struct openpic *opp, struct irq_dest *dst,
264 int output)
265{
266 if (!dst->vcpu) {
267 pr_debug("%s: destination cpu %d does not exist\n",
268 __func__, (int)(dst - &opp->dst[0]));
269 return;
270 }
271
Scott Woodeb1e4f42013-04-12 14:08:47 +0000272 pr_debug("%s: cpu %d output %d\n", __func__, dst->vcpu->arch.irq_cpu_id,
Scott Wood5df554a2013-04-12 14:08:46 +0000273 output);
274
275 if (output != ILR_INTTGT_INT) /* TODO */
276 return;
277
278 kvmppc_core_dequeue_external(dst->vcpu);
279}
280
Scott Woodf0f5c482013-04-12 14:08:45 +0000281static inline void IRQ_setbit(struct irq_queue *q, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000282{
283 set_bit(n_IRQ, q->queue);
284}
285
Scott Woodf0f5c482013-04-12 14:08:45 +0000286static inline void IRQ_resetbit(struct irq_queue *q, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000287{
288 clear_bit(n_IRQ, q->queue);
289}
290
Scott Woodf0f5c482013-04-12 14:08:45 +0000291static inline int IRQ_testbit(struct irq_queue *q, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000292{
293 return test_bit(n_IRQ, q->queue);
294}
295
Scott Woodf0f5c482013-04-12 14:08:45 +0000296static void IRQ_check(struct openpic *opp, struct irq_queue *q)
Scott Woodb823f982013-04-12 14:08:43 +0000297{
298 int irq = -1;
299 int next = -1;
300 int priority = -1;
301
302 for (;;) {
303 irq = find_next_bit(q->queue, opp->max_irq, irq + 1);
Scott Woodf0f5c482013-04-12 14:08:45 +0000304 if (irq == opp->max_irq)
Scott Woodb823f982013-04-12 14:08:43 +0000305 break;
Scott Woodb823f982013-04-12 14:08:43 +0000306
Scott Woodf0f5c482013-04-12 14:08:45 +0000307 pr_debug("IRQ_check: irq %d set ivpr_pr=%d pr=%d\n",
Scott Woodb823f982013-04-12 14:08:43 +0000308 irq, IVPR_PRIORITY(opp->src[irq].ivpr), priority);
309
310 if (IVPR_PRIORITY(opp->src[irq].ivpr) > priority) {
311 next = irq;
312 priority = IVPR_PRIORITY(opp->src[irq].ivpr);
313 }
314 }
315
316 q->next = next;
317 q->priority = priority;
318}
319
Scott Woodf0f5c482013-04-12 14:08:45 +0000320static int IRQ_get_next(struct openpic *opp, struct irq_queue *q)
Scott Woodb823f982013-04-12 14:08:43 +0000321{
322 /* XXX: optimize */
323 IRQ_check(opp, q);
324
325 return q->next;
326}
327
Scott Woodf0f5c482013-04-12 14:08:45 +0000328static void IRQ_local_pipe(struct openpic *opp, int n_CPU, int n_IRQ,
Scott Woodb823f982013-04-12 14:08:43 +0000329 bool active, bool was_active)
330{
Scott Woodf0f5c482013-04-12 14:08:45 +0000331 struct irq_dest *dst;
332 struct irq_source *src;
Scott Woodb823f982013-04-12 14:08:43 +0000333 int priority;
334
335 dst = &opp->dst[n_CPU];
336 src = &opp->src[n_IRQ];
337
Scott Woodf0f5c482013-04-12 14:08:45 +0000338 pr_debug("%s: IRQ %d active %d was %d\n",
Scott Woodb823f982013-04-12 14:08:43 +0000339 __func__, n_IRQ, active, was_active);
340
Scott Wood5df554a2013-04-12 14:08:46 +0000341 if (src->output != ILR_INTTGT_INT) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000342 pr_debug("%s: output %d irq %d active %d was %d count %d\n",
Scott Woodb823f982013-04-12 14:08:43 +0000343 __func__, src->output, n_IRQ, active, was_active,
344 dst->outputs_active[src->output]);
345
346 /* On Freescale MPIC, critical interrupts ignore priority,
347 * IACK, EOI, etc. Before MPIC v4.1 they also ignore
348 * masking.
349 */
350 if (active) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000351 if (!was_active &&
352 dst->outputs_active[src->output]++ == 0) {
353 pr_debug("%s: Raise OpenPIC output %d cpu %d irq %d\n",
354 __func__, src->output, n_CPU, n_IRQ);
Scott Wood5df554a2013-04-12 14:08:46 +0000355 mpic_irq_raise(opp, dst, src->output);
Scott Woodb823f982013-04-12 14:08:43 +0000356 }
357 } else {
Scott Woodf0f5c482013-04-12 14:08:45 +0000358 if (was_active &&
359 --dst->outputs_active[src->output] == 0) {
360 pr_debug("%s: Lower OpenPIC output %d cpu %d irq %d\n",
361 __func__, src->output, n_CPU, n_IRQ);
Scott Wood5df554a2013-04-12 14:08:46 +0000362 mpic_irq_lower(opp, dst, src->output);
Scott Woodb823f982013-04-12 14:08:43 +0000363 }
364 }
365
366 return;
367 }
368
369 priority = IVPR_PRIORITY(src->ivpr);
370
371 /* Even if the interrupt doesn't have enough priority,
372 * it is still raised, in case ctpr is lowered later.
373 */
Scott Woodf0f5c482013-04-12 14:08:45 +0000374 if (active)
Scott Woodb823f982013-04-12 14:08:43 +0000375 IRQ_setbit(&dst->raised, n_IRQ);
Scott Woodf0f5c482013-04-12 14:08:45 +0000376 else
Scott Woodb823f982013-04-12 14:08:43 +0000377 IRQ_resetbit(&dst->raised, n_IRQ);
Scott Woodb823f982013-04-12 14:08:43 +0000378
379 IRQ_check(opp, &dst->raised);
380
381 if (active && priority <= dst->ctpr) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000382 pr_debug("%s: IRQ %d priority %d too low for ctpr %d on CPU %d\n",
383 __func__, n_IRQ, priority, dst->ctpr, n_CPU);
Scott Woodb823f982013-04-12 14:08:43 +0000384 active = 0;
385 }
386
387 if (active) {
388 if (IRQ_get_next(opp, &dst->servicing) >= 0 &&
389 priority <= dst->servicing.priority) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000390 pr_debug("%s: IRQ %d is hidden by servicing IRQ %d on CPU %d\n",
391 __func__, n_IRQ, dst->servicing.next, n_CPU);
Scott Woodb823f982013-04-12 14:08:43 +0000392 } else {
Scott Woodf0f5c482013-04-12 14:08:45 +0000393 pr_debug("%s: Raise OpenPIC INT output cpu %d irq %d/%d\n",
394 __func__, n_CPU, n_IRQ, dst->raised.next);
Scott Wood5df554a2013-04-12 14:08:46 +0000395 mpic_irq_raise(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +0000396 }
397 } else {
398 IRQ_get_next(opp, &dst->servicing);
399 if (dst->raised.priority > dst->ctpr &&
400 dst->raised.priority > dst->servicing.priority) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000401 pr_debug("%s: IRQ %d inactive, IRQ %d prio %d above %d/%d, CPU %d\n",
402 __func__, n_IRQ, dst->raised.next,
403 dst->raised.priority, dst->ctpr,
404 dst->servicing.priority, n_CPU);
Scott Woodb823f982013-04-12 14:08:43 +0000405 /* IRQ line stays asserted */
406 } else {
Scott Woodf0f5c482013-04-12 14:08:45 +0000407 pr_debug("%s: IRQ %d inactive, current prio %d/%d, CPU %d\n",
408 __func__, n_IRQ, dst->ctpr,
409 dst->servicing.priority, n_CPU);
Scott Wood5df554a2013-04-12 14:08:46 +0000410 mpic_irq_lower(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +0000411 }
412 }
413}
414
415/* update pic state because registers for n_IRQ have changed value */
Scott Woodf0f5c482013-04-12 14:08:45 +0000416static void openpic_update_irq(struct openpic *opp, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000417{
Scott Woodf0f5c482013-04-12 14:08:45 +0000418 struct irq_source *src;
Scott Woodb823f982013-04-12 14:08:43 +0000419 bool active, was_active;
420 int i;
421
422 src = &opp->src[n_IRQ];
423 active = src->pending;
424
425 if ((src->ivpr & IVPR_MASK_MASK) && !src->nomask) {
426 /* Interrupt source is disabled */
Scott Woodf0f5c482013-04-12 14:08:45 +0000427 pr_debug("%s: IRQ %d is disabled\n", __func__, n_IRQ);
Scott Woodb823f982013-04-12 14:08:43 +0000428 active = false;
429 }
430
Scott Woodf0f5c482013-04-12 14:08:45 +0000431 was_active = !!(src->ivpr & IVPR_ACTIVITY_MASK);
Scott Woodb823f982013-04-12 14:08:43 +0000432
433 /*
434 * We don't have a similar check for already-active because
435 * ctpr may have changed and we need to withdraw the interrupt.
436 */
437 if (!active && !was_active) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000438 pr_debug("%s: IRQ %d is already inactive\n", __func__, n_IRQ);
Scott Woodb823f982013-04-12 14:08:43 +0000439 return;
440 }
441
Scott Woodf0f5c482013-04-12 14:08:45 +0000442 if (active)
Scott Woodb823f982013-04-12 14:08:43 +0000443 src->ivpr |= IVPR_ACTIVITY_MASK;
Scott Woodf0f5c482013-04-12 14:08:45 +0000444 else
Scott Woodb823f982013-04-12 14:08:43 +0000445 src->ivpr &= ~IVPR_ACTIVITY_MASK;
Scott Woodb823f982013-04-12 14:08:43 +0000446
447 if (src->destmask == 0) {
448 /* No target */
Scott Woodf0f5c482013-04-12 14:08:45 +0000449 pr_debug("%s: IRQ %d has no target\n", __func__, n_IRQ);
Scott Woodb823f982013-04-12 14:08:43 +0000450 return;
451 }
452
453 if (src->destmask == (1 << src->last_cpu)) {
454 /* Only one CPU is allowed to receive this IRQ */
455 IRQ_local_pipe(opp, src->last_cpu, n_IRQ, active, was_active);
456 } else if (!(src->ivpr & IVPR_MODE_MASK)) {
457 /* Directed delivery mode */
458 for (i = 0; i < opp->nb_cpus; i++) {
459 if (src->destmask & (1 << i)) {
460 IRQ_local_pipe(opp, i, n_IRQ, active,
461 was_active);
462 }
463 }
464 } else {
465 /* Distributed delivery mode */
466 for (i = src->last_cpu + 1; i != src->last_cpu; i++) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000467 if (i == opp->nb_cpus)
Scott Woodb823f982013-04-12 14:08:43 +0000468 i = 0;
Scott Woodf0f5c482013-04-12 14:08:45 +0000469
Scott Woodb823f982013-04-12 14:08:43 +0000470 if (src->destmask & (1 << i)) {
471 IRQ_local_pipe(opp, i, n_IRQ, active,
472 was_active);
473 src->last_cpu = i;
474 break;
475 }
476 }
477 }
478}
479
480static void openpic_set_irq(void *opaque, int n_IRQ, int level)
481{
Scott Woodf0f5c482013-04-12 14:08:45 +0000482 struct openpic *opp = opaque;
483 struct irq_source *src;
Scott Woodb823f982013-04-12 14:08:43 +0000484
485 if (n_IRQ >= MAX_IRQ) {
Scott Wood5df554a2013-04-12 14:08:46 +0000486 WARN_ONCE(1, "%s: IRQ %d out of range\n", __func__, n_IRQ);
487 return;
Scott Woodb823f982013-04-12 14:08:43 +0000488 }
489
490 src = &opp->src[n_IRQ];
Scott Woodf0f5c482013-04-12 14:08:45 +0000491 pr_debug("openpic: set irq %d = %d ivpr=0x%08x\n",
Scott Woodb823f982013-04-12 14:08:43 +0000492 n_IRQ, level, src->ivpr);
493 if (src->level) {
494 /* level-sensitive irq */
495 src->pending = level;
496 openpic_update_irq(opp, n_IRQ);
497 } else {
498 /* edge-sensitive irq */
499 if (level) {
500 src->pending = 1;
501 openpic_update_irq(opp, n_IRQ);
502 }
503
Scott Wood5df554a2013-04-12 14:08:46 +0000504 if (src->output != ILR_INTTGT_INT) {
Scott Woodb823f982013-04-12 14:08:43 +0000505 /* Edge-triggered interrupts shouldn't be used
506 * with non-INT delivery, but just in case,
507 * try to make it do something sane rather than
508 * cause an interrupt storm. This is close to
509 * what you'd probably see happen in real hardware.
510 */
511 src->pending = 0;
512 openpic_update_irq(opp, n_IRQ);
513 }
514 }
515}
516
Scott Wood5df554a2013-04-12 14:08:46 +0000517static void openpic_reset(struct openpic *opp)
Scott Woodb823f982013-04-12 14:08:43 +0000518{
Scott Woodb823f982013-04-12 14:08:43 +0000519 int i;
520
521 opp->gcr = GCR_RESET;
522 /* Initialise controller registers */
523 opp->frr = ((opp->nb_irqs - 1) << FRR_NIRQ_SHIFT) |
Scott Woodb823f982013-04-12 14:08:43 +0000524 (opp->vid << FRR_VID_SHIFT);
525
526 opp->pir = 0;
527 opp->spve = -1 & opp->vector_mask;
528 opp->tfrr = opp->tfrr_reset;
529 /* Initialise IRQ sources */
530 for (i = 0; i < opp->max_irq; i++) {
531 opp->src[i].ivpr = opp->ivpr_reset;
532 opp->src[i].idr = opp->idr_reset;
533
534 switch (opp->src[i].type) {
535 case IRQ_TYPE_NORMAL:
536 opp->src[i].level =
Scott Woodf0f5c482013-04-12 14:08:45 +0000537 !!(opp->ivpr_reset & IVPR_SENSE_MASK);
Scott Woodb823f982013-04-12 14:08:43 +0000538 break;
539
540 case IRQ_TYPE_FSLINT:
541 opp->src[i].ivpr |= IVPR_POLARITY_MASK;
542 break;
543
544 case IRQ_TYPE_FSLSPECIAL:
545 break;
546 }
547 }
548 /* Initialise IRQ destinations */
549 for (i = 0; i < MAX_CPU; i++) {
550 opp->dst[i].ctpr = 15;
Scott Woodf0f5c482013-04-12 14:08:45 +0000551 memset(&opp->dst[i].raised, 0, sizeof(struct irq_queue));
Scott Woodb823f982013-04-12 14:08:43 +0000552 opp->dst[i].raised.next = -1;
Scott Woodf0f5c482013-04-12 14:08:45 +0000553 memset(&opp->dst[i].servicing, 0, sizeof(struct irq_queue));
Scott Woodb823f982013-04-12 14:08:43 +0000554 opp->dst[i].servicing.next = -1;
555 }
556 /* Initialise timers */
557 for (i = 0; i < MAX_TMR; i++) {
558 opp->timers[i].tccr = 0;
559 opp->timers[i].tbcr = TBCR_CI;
560 }
561 /* Go out of RESET state */
562 opp->gcr = 0;
563}
564
Scott Woodf0f5c482013-04-12 14:08:45 +0000565static inline uint32_t read_IRQreg_idr(struct openpic *opp, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000566{
567 return opp->src[n_IRQ].idr;
568}
569
Scott Woodf0f5c482013-04-12 14:08:45 +0000570static inline uint32_t read_IRQreg_ilr(struct openpic *opp, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000571{
Scott Woodf0f5c482013-04-12 14:08:45 +0000572 if (opp->flags & OPENPIC_FLAG_ILR)
Scott Wood5df554a2013-04-12 14:08:46 +0000573 return opp->src[n_IRQ].output;
Scott Woodb823f982013-04-12 14:08:43 +0000574
575 return 0xffffffff;
576}
577
Scott Woodf0f5c482013-04-12 14:08:45 +0000578static inline uint32_t read_IRQreg_ivpr(struct openpic *opp, int n_IRQ)
Scott Woodb823f982013-04-12 14:08:43 +0000579{
580 return opp->src[n_IRQ].ivpr;
581}
582
Scott Woodf0f5c482013-04-12 14:08:45 +0000583static inline void write_IRQreg_idr(struct openpic *opp, int n_IRQ,
584 uint32_t val)
Scott Woodb823f982013-04-12 14:08:43 +0000585{
Scott Woodf0f5c482013-04-12 14:08:45 +0000586 struct irq_source *src = &opp->src[n_IRQ];
Scott Woodb823f982013-04-12 14:08:43 +0000587 uint32_t normal_mask = (1UL << opp->nb_cpus) - 1;
588 uint32_t crit_mask = 0;
589 uint32_t mask = normal_mask;
590 int crit_shift = IDR_EP_SHIFT - opp->nb_cpus;
591 int i;
592
593 if (opp->flags & OPENPIC_FLAG_IDR_CRIT) {
594 crit_mask = mask << crit_shift;
595 mask |= crit_mask | IDR_EP;
596 }
597
598 src->idr = val & mask;
Scott Woodf0f5c482013-04-12 14:08:45 +0000599 pr_debug("Set IDR %d to 0x%08x\n", n_IRQ, src->idr);
Scott Woodb823f982013-04-12 14:08:43 +0000600
601 if (opp->flags & OPENPIC_FLAG_IDR_CRIT) {
602 if (src->idr & crit_mask) {
603 if (src->idr & normal_mask) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000604 pr_debug("%s: IRQ configured for multiple output types, using critical\n",
605 __func__);
Scott Woodb823f982013-04-12 14:08:43 +0000606 }
607
Scott Wood5df554a2013-04-12 14:08:46 +0000608 src->output = ILR_INTTGT_CINT;
Scott Woodb823f982013-04-12 14:08:43 +0000609 src->nomask = true;
610 src->destmask = 0;
611
612 for (i = 0; i < opp->nb_cpus; i++) {
613 int n_ci = IDR_CI0_SHIFT - i;
614
Scott Woodf0f5c482013-04-12 14:08:45 +0000615 if (src->idr & (1UL << n_ci))
Scott Woodb823f982013-04-12 14:08:43 +0000616 src->destmask |= 1UL << i;
Scott Woodb823f982013-04-12 14:08:43 +0000617 }
618 } else {
Scott Wood5df554a2013-04-12 14:08:46 +0000619 src->output = ILR_INTTGT_INT;
Scott Woodb823f982013-04-12 14:08:43 +0000620 src->nomask = false;
621 src->destmask = src->idr & normal_mask;
622 }
623 } else {
624 src->destmask = src->idr;
625 }
626}
627
Scott Woodf0f5c482013-04-12 14:08:45 +0000628static inline void write_IRQreg_ilr(struct openpic *opp, int n_IRQ,
629 uint32_t val)
Scott Woodb823f982013-04-12 14:08:43 +0000630{
631 if (opp->flags & OPENPIC_FLAG_ILR) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000632 struct irq_source *src = &opp->src[n_IRQ];
Scott Woodb823f982013-04-12 14:08:43 +0000633
Scott Wood5df554a2013-04-12 14:08:46 +0000634 src->output = val & ILR_INTTGT_MASK;
Scott Woodf0f5c482013-04-12 14:08:45 +0000635 pr_debug("Set ILR %d to 0x%08x, output %d\n", n_IRQ, src->idr,
Scott Woodb823f982013-04-12 14:08:43 +0000636 src->output);
637
638 /* TODO: on MPIC v4.0 only, set nomask for non-INT */
639 }
640}
641
Scott Woodf0f5c482013-04-12 14:08:45 +0000642static inline void write_IRQreg_ivpr(struct openpic *opp, int n_IRQ,
Scott Woodb823f982013-04-12 14:08:43 +0000643 uint32_t val)
644{
645 uint32_t mask;
646
647 /* NOTE when implementing newer FSL MPIC models: starting with v4.0,
648 * the polarity bit is read-only on internal interrupts.
649 */
650 mask = IVPR_MASK_MASK | IVPR_PRIORITY_MASK | IVPR_SENSE_MASK |
651 IVPR_POLARITY_MASK | opp->vector_mask;
652
653 /* ACTIVITY bit is read-only */
654 opp->src[n_IRQ].ivpr =
655 (opp->src[n_IRQ].ivpr & IVPR_ACTIVITY_MASK) | (val & mask);
656
657 /* For FSL internal interrupts, The sense bit is reserved and zero,
658 * and the interrupt is always level-triggered. Timers and IPIs
659 * have no sense or polarity bits, and are edge-triggered.
660 */
661 switch (opp->src[n_IRQ].type) {
662 case IRQ_TYPE_NORMAL:
663 opp->src[n_IRQ].level =
Scott Woodf0f5c482013-04-12 14:08:45 +0000664 !!(opp->src[n_IRQ].ivpr & IVPR_SENSE_MASK);
Scott Woodb823f982013-04-12 14:08:43 +0000665 break;
666
667 case IRQ_TYPE_FSLINT:
668 opp->src[n_IRQ].ivpr &= ~IVPR_SENSE_MASK;
669 break;
670
671 case IRQ_TYPE_FSLSPECIAL:
672 opp->src[n_IRQ].ivpr &= ~(IVPR_POLARITY_MASK | IVPR_SENSE_MASK);
673 break;
674 }
675
676 openpic_update_irq(opp, n_IRQ);
Scott Woodf0f5c482013-04-12 14:08:45 +0000677 pr_debug("Set IVPR %d to 0x%08x -> 0x%08x\n", n_IRQ, val,
Scott Woodb823f982013-04-12 14:08:43 +0000678 opp->src[n_IRQ].ivpr);
679}
680
Scott Woodf0f5c482013-04-12 14:08:45 +0000681static void openpic_gcr_write(struct openpic *opp, uint64_t val)
Scott Woodb823f982013-04-12 14:08:43 +0000682{
Scott Woodb823f982013-04-12 14:08:43 +0000683 if (val & GCR_RESET) {
Scott Wood5df554a2013-04-12 14:08:46 +0000684 openpic_reset(opp);
Scott Woodb823f982013-04-12 14:08:43 +0000685 return;
686 }
687
688 opp->gcr &= ~opp->mpic_mode_mask;
689 opp->gcr |= val & opp->mpic_mode_mask;
Scott Woodb823f982013-04-12 14:08:43 +0000690}
691
Scott Wood5df554a2013-04-12 14:08:46 +0000692static int openpic_gbl_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +0000693{
Scott Woodf0f5c482013-04-12 14:08:45 +0000694 struct openpic *opp = opaque;
Scott Wood5df554a2013-04-12 14:08:46 +0000695 int err = 0;
Scott Woodb823f982013-04-12 14:08:43 +0000696
Scott Wood5df554a2013-04-12 14:08:46 +0000697 pr_debug("%s: addr %#llx <= %08x\n", __func__, addr, val);
Scott Woodf0f5c482013-04-12 14:08:45 +0000698 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +0000699 return 0;
Scott Woodf0f5c482013-04-12 14:08:45 +0000700
Scott Woodb823f982013-04-12 14:08:43 +0000701 switch (addr) {
Scott Woodf0f5c482013-04-12 14:08:45 +0000702 case 0x00: /* Block Revision Register1 (BRR1) is Readonly */
Scott Woodb823f982013-04-12 14:08:43 +0000703 break;
704 case 0x40:
705 case 0x50:
706 case 0x60:
707 case 0x70:
708 case 0x80:
709 case 0x90:
710 case 0xA0:
711 case 0xB0:
Scott Wood5df554a2013-04-12 14:08:46 +0000712 err = openpic_cpu_write_internal(opp, addr, val,
713 get_current_cpu());
Scott Woodb823f982013-04-12 14:08:43 +0000714 break;
715 case 0x1000: /* FRR */
716 break;
717 case 0x1020: /* GCR */
718 openpic_gcr_write(opp, val);
719 break;
720 case 0x1080: /* VIR */
721 break;
722 case 0x1090: /* PIR */
Scott Wood5df554a2013-04-12 14:08:46 +0000723 /*
724 * This register is used to reset a CPU core --
725 * let userspace handle it.
726 */
727 err = -ENXIO;
Scott Woodb823f982013-04-12 14:08:43 +0000728 break;
729 case 0x10A0: /* IPI_IVPR */
730 case 0x10B0:
731 case 0x10C0:
Scott Woodf0f5c482013-04-12 14:08:45 +0000732 case 0x10D0: {
733 int idx;
734 idx = (addr - 0x10A0) >> 4;
735 write_IRQreg_ivpr(opp, opp->irq_ipi0 + idx, val);
Scott Woodb823f982013-04-12 14:08:43 +0000736 break;
Scott Woodf0f5c482013-04-12 14:08:45 +0000737 }
Scott Woodb823f982013-04-12 14:08:43 +0000738 case 0x10E0: /* SPVE */
739 opp->spve = val & opp->vector_mask;
740 break;
741 default:
742 break;
743 }
Scott Wood5df554a2013-04-12 14:08:46 +0000744
745 return err;
Scott Woodb823f982013-04-12 14:08:43 +0000746}
747
Scott Wood5df554a2013-04-12 14:08:46 +0000748static int openpic_gbl_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +0000749{
Scott Woodf0f5c482013-04-12 14:08:45 +0000750 struct openpic *opp = opaque;
Scott Wood5df554a2013-04-12 14:08:46 +0000751 u32 retval;
752 int err = 0;
Scott Woodb823f982013-04-12 14:08:43 +0000753
Scott Wood5df554a2013-04-12 14:08:46 +0000754 pr_debug("%s: addr %#llx\n", __func__, addr);
Scott Woodb823f982013-04-12 14:08:43 +0000755 retval = 0xFFFFFFFF;
Scott Woodf0f5c482013-04-12 14:08:45 +0000756 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +0000757 goto out;
Scott Woodf0f5c482013-04-12 14:08:45 +0000758
Scott Woodb823f982013-04-12 14:08:43 +0000759 switch (addr) {
760 case 0x1000: /* FRR */
761 retval = opp->frr;
Scott Wood5df554a2013-04-12 14:08:46 +0000762 retval |= (opp->nb_cpus - 1) << FRR_NCPU_SHIFT;
Scott Woodb823f982013-04-12 14:08:43 +0000763 break;
764 case 0x1020: /* GCR */
765 retval = opp->gcr;
766 break;
767 case 0x1080: /* VIR */
768 retval = opp->vir;
769 break;
770 case 0x1090: /* PIR */
771 retval = 0x00000000;
772 break;
773 case 0x00: /* Block Revision Register1 (BRR1) */
774 retval = opp->brr1;
775 break;
776 case 0x40:
777 case 0x50:
778 case 0x60:
779 case 0x70:
780 case 0x80:
781 case 0x90:
782 case 0xA0:
783 case 0xB0:
Scott Wood5df554a2013-04-12 14:08:46 +0000784 err = openpic_cpu_read_internal(opp, addr,
785 &retval, get_current_cpu());
Scott Woodb823f982013-04-12 14:08:43 +0000786 break;
787 case 0x10A0: /* IPI_IVPR */
788 case 0x10B0:
789 case 0x10C0:
790 case 0x10D0:
791 {
792 int idx;
793 idx = (addr - 0x10A0) >> 4;
794 retval = read_IRQreg_ivpr(opp, opp->irq_ipi0 + idx);
795 }
796 break;
797 case 0x10E0: /* SPVE */
798 retval = opp->spve;
799 break;
800 default:
801 break;
802 }
Scott Woodb823f982013-04-12 14:08:43 +0000803
Scott Wood5df554a2013-04-12 14:08:46 +0000804out:
805 pr_debug("%s: => 0x%08x\n", __func__, retval);
806 *ptr = retval;
807 return err;
Scott Woodb823f982013-04-12 14:08:43 +0000808}
809
Scott Wood5df554a2013-04-12 14:08:46 +0000810static int openpic_tmr_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +0000811{
Scott Woodf0f5c482013-04-12 14:08:45 +0000812 struct openpic *opp = opaque;
Scott Woodb823f982013-04-12 14:08:43 +0000813 int idx;
814
815 addr += 0x10f0;
816
Scott Wood5df554a2013-04-12 14:08:46 +0000817 pr_debug("%s: addr %#llx <= %08x\n", __func__, addr, val);
Scott Woodf0f5c482013-04-12 14:08:45 +0000818 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +0000819 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000820
821 if (addr == 0x10f0) {
822 /* TFRR */
823 opp->tfrr = val;
Scott Wood5df554a2013-04-12 14:08:46 +0000824 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000825 }
826
827 idx = (addr >> 6) & 0x3;
828 addr = addr & 0x30;
829
830 switch (addr & 0x30) {
831 case 0x00: /* TCCR */
832 break;
833 case 0x10: /* TBCR */
834 if ((opp->timers[idx].tccr & TCCR_TOG) != 0 &&
835 (val & TBCR_CI) == 0 &&
Scott Woodf0f5c482013-04-12 14:08:45 +0000836 (opp->timers[idx].tbcr & TBCR_CI) != 0)
Scott Woodb823f982013-04-12 14:08:43 +0000837 opp->timers[idx].tccr &= ~TCCR_TOG;
Scott Woodf0f5c482013-04-12 14:08:45 +0000838
Scott Woodb823f982013-04-12 14:08:43 +0000839 opp->timers[idx].tbcr = val;
840 break;
841 case 0x20: /* TVPR */
842 write_IRQreg_ivpr(opp, opp->irq_tim0 + idx, val);
843 break;
844 case 0x30: /* TDR */
845 write_IRQreg_idr(opp, opp->irq_tim0 + idx, val);
846 break;
847 }
Scott Wood5df554a2013-04-12 14:08:46 +0000848
849 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000850}
851
Scott Wood5df554a2013-04-12 14:08:46 +0000852static int openpic_tmr_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +0000853{
Scott Woodf0f5c482013-04-12 14:08:45 +0000854 struct openpic *opp = opaque;
Scott Woodb823f982013-04-12 14:08:43 +0000855 uint32_t retval = -1;
856 int idx;
857
Scott Wood5df554a2013-04-12 14:08:46 +0000858 pr_debug("%s: addr %#llx\n", __func__, addr);
Scott Woodf0f5c482013-04-12 14:08:45 +0000859 if (addr & 0xF)
Scott Woodb823f982013-04-12 14:08:43 +0000860 goto out;
Scott Woodf0f5c482013-04-12 14:08:45 +0000861
Scott Woodb823f982013-04-12 14:08:43 +0000862 idx = (addr >> 6) & 0x3;
863 if (addr == 0x0) {
864 /* TFRR */
865 retval = opp->tfrr;
866 goto out;
867 }
Scott Wood5df554a2013-04-12 14:08:46 +0000868
Scott Woodb823f982013-04-12 14:08:43 +0000869 switch (addr & 0x30) {
870 case 0x00: /* TCCR */
871 retval = opp->timers[idx].tccr;
872 break;
873 case 0x10: /* TBCR */
874 retval = opp->timers[idx].tbcr;
875 break;
876 case 0x20: /* TIPV */
877 retval = read_IRQreg_ivpr(opp, opp->irq_tim0 + idx);
878 break;
879 case 0x30: /* TIDE (TIDR) */
880 retval = read_IRQreg_idr(opp, opp->irq_tim0 + idx);
881 break;
882 }
883
884out:
Scott Woodf0f5c482013-04-12 14:08:45 +0000885 pr_debug("%s: => 0x%08x\n", __func__, retval);
Scott Wood5df554a2013-04-12 14:08:46 +0000886 *ptr = retval;
887 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000888}
889
Scott Wood5df554a2013-04-12 14:08:46 +0000890static int openpic_src_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +0000891{
Scott Woodf0f5c482013-04-12 14:08:45 +0000892 struct openpic *opp = opaque;
Scott Woodb823f982013-04-12 14:08:43 +0000893 int idx;
894
Scott Wood5df554a2013-04-12 14:08:46 +0000895 pr_debug("%s: addr %#llx <= %08x\n", __func__, addr, val);
Scott Woodb823f982013-04-12 14:08:43 +0000896
897 addr = addr & 0xffff;
898 idx = addr >> 5;
899
900 switch (addr & 0x1f) {
901 case 0x00:
902 write_IRQreg_ivpr(opp, idx, val);
903 break;
904 case 0x10:
905 write_IRQreg_idr(opp, idx, val);
906 break;
907 case 0x18:
908 write_IRQreg_ilr(opp, idx, val);
909 break;
910 }
Scott Wood5df554a2013-04-12 14:08:46 +0000911
912 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000913}
914
Scott Wood5df554a2013-04-12 14:08:46 +0000915static int openpic_src_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +0000916{
Scott Woodf0f5c482013-04-12 14:08:45 +0000917 struct openpic *opp = opaque;
Scott Woodb823f982013-04-12 14:08:43 +0000918 uint32_t retval;
919 int idx;
920
Scott Wood5df554a2013-04-12 14:08:46 +0000921 pr_debug("%s: addr %#llx\n", __func__, addr);
Scott Woodb823f982013-04-12 14:08:43 +0000922 retval = 0xFFFFFFFF;
923
924 addr = addr & 0xffff;
925 idx = addr >> 5;
926
927 switch (addr & 0x1f) {
928 case 0x00:
929 retval = read_IRQreg_ivpr(opp, idx);
930 break;
931 case 0x10:
932 retval = read_IRQreg_idr(opp, idx);
933 break;
934 case 0x18:
935 retval = read_IRQreg_ilr(opp, idx);
936 break;
937 }
938
Scott Woodf0f5c482013-04-12 14:08:45 +0000939 pr_debug("%s: => 0x%08x\n", __func__, retval);
Scott Wood5df554a2013-04-12 14:08:46 +0000940 *ptr = retval;
941 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000942}
943
Scott Wood5df554a2013-04-12 14:08:46 +0000944static int openpic_msi_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +0000945{
Scott Woodf0f5c482013-04-12 14:08:45 +0000946 struct openpic *opp = opaque;
Scott Woodb823f982013-04-12 14:08:43 +0000947 int idx = opp->irq_msi;
948 int srs, ibs;
949
Scott Wood5df554a2013-04-12 14:08:46 +0000950 pr_debug("%s: addr %#llx <= 0x%08x\n", __func__, addr, val);
Scott Woodf0f5c482013-04-12 14:08:45 +0000951 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +0000952 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000953
954 switch (addr) {
955 case MSIIR_OFFSET:
956 srs = val >> MSIIR_SRS_SHIFT;
957 idx += srs;
958 ibs = (val & MSIIR_IBS_MASK) >> MSIIR_IBS_SHIFT;
959 opp->msi[srs].msir |= 1 << ibs;
960 openpic_set_irq(opp, idx, 1);
961 break;
962 default:
963 /* most registers are read-only, thus ignored */
964 break;
965 }
Scott Wood5df554a2013-04-12 14:08:46 +0000966
967 return 0;
Scott Woodb823f982013-04-12 14:08:43 +0000968}
969
Scott Wood5df554a2013-04-12 14:08:46 +0000970static int openpic_msi_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +0000971{
Scott Woodf0f5c482013-04-12 14:08:45 +0000972 struct openpic *opp = opaque;
Scott Wood5df554a2013-04-12 14:08:46 +0000973 uint32_t r = 0;
Scott Woodb823f982013-04-12 14:08:43 +0000974 int i, srs;
975
Scott Wood5df554a2013-04-12 14:08:46 +0000976 pr_debug("%s: addr %#llx\n", __func__, addr);
Scott Woodf0f5c482013-04-12 14:08:45 +0000977 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +0000978 return -ENXIO;
Scott Woodb823f982013-04-12 14:08:43 +0000979
980 srs = addr >> 4;
981
982 switch (addr) {
983 case 0x00:
984 case 0x10:
985 case 0x20:
986 case 0x30:
987 case 0x40:
988 case 0x50:
989 case 0x60:
990 case 0x70: /* MSIRs */
991 r = opp->msi[srs].msir;
992 /* Clear on read */
993 opp->msi[srs].msir = 0;
994 openpic_set_irq(opp, opp->irq_msi + srs, 0);
995 break;
996 case 0x120: /* MSISR */
Scott Woodf0f5c482013-04-12 14:08:45 +0000997 for (i = 0; i < MAX_MSI; i++)
Scott Woodb823f982013-04-12 14:08:43 +0000998 r |= (opp->msi[i].msir ? 1 : 0) << i;
Scott Woodb823f982013-04-12 14:08:43 +0000999 break;
1000 }
1001
Scott Wood5df554a2013-04-12 14:08:46 +00001002 pr_debug("%s: => 0x%08x\n", __func__, r);
1003 *ptr = r;
1004 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001005}
1006
Scott Wood5df554a2013-04-12 14:08:46 +00001007static int openpic_summary_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +00001008{
Scott Wood5df554a2013-04-12 14:08:46 +00001009 uint32_t r = 0;
Scott Woodb823f982013-04-12 14:08:43 +00001010
Scott Wood5df554a2013-04-12 14:08:46 +00001011 pr_debug("%s: addr %#llx\n", __func__, addr);
Scott Woodb823f982013-04-12 14:08:43 +00001012
1013 /* TODO: EISR/EIMR */
1014
Scott Wood5df554a2013-04-12 14:08:46 +00001015 *ptr = r;
1016 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001017}
1018
Scott Wood5df554a2013-04-12 14:08:46 +00001019static int openpic_summary_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +00001020{
Scott Wood5df554a2013-04-12 14:08:46 +00001021 pr_debug("%s: addr %#llx <= 0x%08x\n", __func__, addr, val);
Scott Woodb823f982013-04-12 14:08:43 +00001022
1023 /* TODO: EISR/EIMR */
Scott Wood5df554a2013-04-12 14:08:46 +00001024 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001025}
1026
Scott Wood5df554a2013-04-12 14:08:46 +00001027static int openpic_cpu_write_internal(void *opaque, gpa_t addr,
1028 u32 val, int idx)
Scott Woodb823f982013-04-12 14:08:43 +00001029{
Scott Woodf0f5c482013-04-12 14:08:45 +00001030 struct openpic *opp = opaque;
1031 struct irq_source *src;
1032 struct irq_dest *dst;
Scott Woodb823f982013-04-12 14:08:43 +00001033 int s_IRQ, n_IRQ;
1034
Scott Wood5df554a2013-04-12 14:08:46 +00001035 pr_debug("%s: cpu %d addr %#llx <= 0x%08x\n", __func__, idx,
Scott Woodb823f982013-04-12 14:08:43 +00001036 addr, val);
1037
Scott Woodf0f5c482013-04-12 14:08:45 +00001038 if (idx < 0)
Scott Wood5df554a2013-04-12 14:08:46 +00001039 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001040
Scott Woodf0f5c482013-04-12 14:08:45 +00001041 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +00001042 return 0;
Scott Woodf0f5c482013-04-12 14:08:45 +00001043
Scott Woodb823f982013-04-12 14:08:43 +00001044 dst = &opp->dst[idx];
1045 addr &= 0xFF0;
1046 switch (addr) {
1047 case 0x40: /* IPIDR */
1048 case 0x50:
1049 case 0x60:
1050 case 0x70:
1051 idx = (addr - 0x40) >> 4;
1052 /* we use IDE as mask which CPUs to deliver the IPI to still. */
1053 opp->src[opp->irq_ipi0 + idx].destmask |= val;
1054 openpic_set_irq(opp, opp->irq_ipi0 + idx, 1);
1055 openpic_set_irq(opp, opp->irq_ipi0 + idx, 0);
1056 break;
1057 case 0x80: /* CTPR */
1058 dst->ctpr = val & 0x0000000F;
1059
Scott Woodf0f5c482013-04-12 14:08:45 +00001060 pr_debug("%s: set CPU %d ctpr to %d, raised %d servicing %d\n",
Scott Woodb823f982013-04-12 14:08:43 +00001061 __func__, idx, dst->ctpr, dst->raised.priority,
1062 dst->servicing.priority);
1063
1064 if (dst->raised.priority <= dst->ctpr) {
Scott Woodf0f5c482013-04-12 14:08:45 +00001065 pr_debug("%s: Lower OpenPIC INT output cpu %d due to ctpr\n",
1066 __func__, idx);
Scott Wood5df554a2013-04-12 14:08:46 +00001067 mpic_irq_lower(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +00001068 } else if (dst->raised.priority > dst->servicing.priority) {
Scott Woodf0f5c482013-04-12 14:08:45 +00001069 pr_debug("%s: Raise OpenPIC INT output cpu %d irq %d\n",
Scott Woodb823f982013-04-12 14:08:43 +00001070 __func__, idx, dst->raised.next);
Scott Wood5df554a2013-04-12 14:08:46 +00001071 mpic_irq_raise(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +00001072 }
1073
1074 break;
1075 case 0x90: /* WHOAMI */
1076 /* Read-only register */
1077 break;
1078 case 0xA0: /* IACK */
1079 /* Read-only register */
1080 break;
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001081 case 0xB0: { /* EOI */
1082 int notify_eoi;
1083
Scott Woodf0f5c482013-04-12 14:08:45 +00001084 pr_debug("EOI\n");
Scott Woodb823f982013-04-12 14:08:43 +00001085 s_IRQ = IRQ_get_next(opp, &dst->servicing);
1086
1087 if (s_IRQ < 0) {
Scott Woodf0f5c482013-04-12 14:08:45 +00001088 pr_debug("%s: EOI with no interrupt in service\n",
Scott Woodb823f982013-04-12 14:08:43 +00001089 __func__);
1090 break;
1091 }
1092
1093 IRQ_resetbit(&dst->servicing, s_IRQ);
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001094 /* Notify listeners that the IRQ is over */
1095 notify_eoi = s_IRQ;
Scott Woodb823f982013-04-12 14:08:43 +00001096 /* Set up next servicing IRQ */
1097 s_IRQ = IRQ_get_next(opp, &dst->servicing);
1098 /* Check queued interrupts. */
1099 n_IRQ = IRQ_get_next(opp, &dst->raised);
1100 src = &opp->src[n_IRQ];
1101 if (n_IRQ != -1 &&
1102 (s_IRQ == -1 ||
1103 IVPR_PRIORITY(src->ivpr) > dst->servicing.priority)) {
Scott Woodf0f5c482013-04-12 14:08:45 +00001104 pr_debug("Raise OpenPIC INT output cpu %d irq %d\n",
Scott Woodb823f982013-04-12 14:08:43 +00001105 idx, n_IRQ);
Scott Wood5df554a2013-04-12 14:08:46 +00001106 mpic_irq_raise(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +00001107 }
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001108
1109 spin_unlock(&opp->lock);
1110 kvm_notify_acked_irq(opp->kvm, 0, notify_eoi);
1111 spin_lock(&opp->lock);
1112
Scott Woodb823f982013-04-12 14:08:43 +00001113 break;
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001114 }
Scott Woodb823f982013-04-12 14:08:43 +00001115 default:
1116 break;
1117 }
Scott Wood5df554a2013-04-12 14:08:46 +00001118
1119 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001120}
1121
Scott Wood5df554a2013-04-12 14:08:46 +00001122static int openpic_cpu_write(void *opaque, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +00001123{
Scott Wood5df554a2013-04-12 14:08:46 +00001124 struct openpic *opp = opaque;
1125
1126 return openpic_cpu_write_internal(opp, addr, val,
1127 (addr & 0x1f000) >> 12);
Scott Woodb823f982013-04-12 14:08:43 +00001128}
1129
Scott Woodf0f5c482013-04-12 14:08:45 +00001130static uint32_t openpic_iack(struct openpic *opp, struct irq_dest *dst,
1131 int cpu)
Scott Woodb823f982013-04-12 14:08:43 +00001132{
Scott Woodf0f5c482013-04-12 14:08:45 +00001133 struct irq_source *src;
Scott Woodb823f982013-04-12 14:08:43 +00001134 int retval, irq;
1135
Scott Woodf0f5c482013-04-12 14:08:45 +00001136 pr_debug("Lower OpenPIC INT output\n");
Scott Wood5df554a2013-04-12 14:08:46 +00001137 mpic_irq_lower(opp, dst, ILR_INTTGT_INT);
Scott Woodb823f982013-04-12 14:08:43 +00001138
1139 irq = IRQ_get_next(opp, &dst->raised);
Scott Woodf0f5c482013-04-12 14:08:45 +00001140 pr_debug("IACK: irq=%d\n", irq);
Scott Woodb823f982013-04-12 14:08:43 +00001141
Scott Woodf0f5c482013-04-12 14:08:45 +00001142 if (irq == -1)
Scott Woodb823f982013-04-12 14:08:43 +00001143 /* No more interrupt pending */
1144 return opp->spve;
Scott Woodb823f982013-04-12 14:08:43 +00001145
1146 src = &opp->src[irq];
1147 if (!(src->ivpr & IVPR_ACTIVITY_MASK) ||
1148 !(IVPR_PRIORITY(src->ivpr) > dst->ctpr)) {
Scott Woodf0f5c482013-04-12 14:08:45 +00001149 pr_err("%s: bad raised IRQ %d ctpr %d ivpr 0x%08x\n",
Scott Woodb823f982013-04-12 14:08:43 +00001150 __func__, irq, dst->ctpr, src->ivpr);
1151 openpic_update_irq(opp, irq);
1152 retval = opp->spve;
1153 } else {
1154 /* IRQ enter servicing state */
1155 IRQ_setbit(&dst->servicing, irq);
1156 retval = IVPR_VECTOR(opp, src->ivpr);
1157 }
1158
1159 if (!src->level) {
1160 /* edge-sensitive IRQ */
1161 src->ivpr &= ~IVPR_ACTIVITY_MASK;
1162 src->pending = 0;
1163 IRQ_resetbit(&dst->raised, irq);
1164 }
1165
1166 if ((irq >= opp->irq_ipi0) && (irq < (opp->irq_ipi0 + MAX_IPI))) {
1167 src->destmask &= ~(1 << cpu);
1168 if (src->destmask && !src->level) {
1169 /* trigger on CPUs that didn't know about it yet */
1170 openpic_set_irq(opp, irq, 1);
1171 openpic_set_irq(opp, irq, 0);
1172 /* if all CPUs knew about it, set active bit again */
1173 src->ivpr |= IVPR_ACTIVITY_MASK;
1174 }
1175 }
1176
1177 return retval;
1178}
1179
Scott Woodeb1e4f42013-04-12 14:08:47 +00001180void kvmppc_mpic_set_epr(struct kvm_vcpu *vcpu)
1181{
1182 struct openpic *opp = vcpu->arch.mpic;
1183 int cpu = vcpu->arch.irq_cpu_id;
1184 unsigned long flags;
1185
1186 spin_lock_irqsave(&opp->lock, flags);
1187
1188 if ((opp->gcr & opp->mpic_mode_mask) == GCR_MODE_PROXY)
1189 kvmppc_set_epr(vcpu, openpic_iack(opp, &opp->dst[cpu], cpu));
1190
1191 spin_unlock_irqrestore(&opp->lock, flags);
1192}
1193
Scott Wood5df554a2013-04-12 14:08:46 +00001194static int openpic_cpu_read_internal(void *opaque, gpa_t addr,
1195 u32 *ptr, int idx)
Scott Woodb823f982013-04-12 14:08:43 +00001196{
Scott Woodf0f5c482013-04-12 14:08:45 +00001197 struct openpic *opp = opaque;
1198 struct irq_dest *dst;
Scott Woodb823f982013-04-12 14:08:43 +00001199 uint32_t retval;
1200
Scott Wood5df554a2013-04-12 14:08:46 +00001201 pr_debug("%s: cpu %d addr %#llx\n", __func__, idx, addr);
Scott Woodb823f982013-04-12 14:08:43 +00001202 retval = 0xFFFFFFFF;
1203
Scott Woodf0f5c482013-04-12 14:08:45 +00001204 if (idx < 0)
Scott Wood5df554a2013-04-12 14:08:46 +00001205 goto out;
Scott Woodb823f982013-04-12 14:08:43 +00001206
Scott Woodf0f5c482013-04-12 14:08:45 +00001207 if (addr & 0xF)
Scott Wood5df554a2013-04-12 14:08:46 +00001208 goto out;
Scott Woodf0f5c482013-04-12 14:08:45 +00001209
Scott Woodb823f982013-04-12 14:08:43 +00001210 dst = &opp->dst[idx];
1211 addr &= 0xFF0;
1212 switch (addr) {
1213 case 0x80: /* CTPR */
1214 retval = dst->ctpr;
1215 break;
1216 case 0x90: /* WHOAMI */
1217 retval = idx;
1218 break;
1219 case 0xA0: /* IACK */
1220 retval = openpic_iack(opp, dst, idx);
1221 break;
1222 case 0xB0: /* EOI */
1223 retval = 0;
1224 break;
1225 default:
1226 break;
1227 }
Scott Woodf0f5c482013-04-12 14:08:45 +00001228 pr_debug("%s: => 0x%08x\n", __func__, retval);
Scott Woodb823f982013-04-12 14:08:43 +00001229
Scott Wood5df554a2013-04-12 14:08:46 +00001230out:
1231 *ptr = retval;
1232 return 0;
Scott Woodb823f982013-04-12 14:08:43 +00001233}
1234
Scott Wood5df554a2013-04-12 14:08:46 +00001235static int openpic_cpu_read(void *opaque, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +00001236{
Scott Wood5df554a2013-04-12 14:08:46 +00001237 struct openpic *opp = opaque;
1238
1239 return openpic_cpu_read_internal(opp, addr, ptr,
1240 (addr & 0x1f000) >> 12);
Scott Woodb823f982013-04-12 14:08:43 +00001241}
1242
Scott Woodf0f5c482013-04-12 14:08:45 +00001243struct mem_reg {
Scott Wood5df554a2013-04-12 14:08:46 +00001244 int (*read)(void *opaque, gpa_t addr, u32 *ptr);
1245 int (*write)(void *opaque, gpa_t addr, u32 val);
Scott Woodf0f5c482013-04-12 14:08:45 +00001246 gpa_t start_addr;
1247 int size;
1248};
Scott Woodb823f982013-04-12 14:08:43 +00001249
Scott Wood398d8782013-04-30 14:57:13 +00001250static const struct mem_reg openpic_gbl_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001251 .write = openpic_gbl_write,
1252 .read = openpic_gbl_read,
1253 .start_addr = OPENPIC_GLB_REG_START,
1254 .size = OPENPIC_GLB_REG_SIZE,
1255};
1256
Scott Wood398d8782013-04-30 14:57:13 +00001257static const struct mem_reg openpic_tmr_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001258 .write = openpic_tmr_write,
1259 .read = openpic_tmr_read,
1260 .start_addr = OPENPIC_TMR_REG_START,
1261 .size = OPENPIC_TMR_REG_SIZE,
1262};
1263
Scott Wood398d8782013-04-30 14:57:13 +00001264static const struct mem_reg openpic_cpu_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001265 .write = openpic_cpu_write,
1266 .read = openpic_cpu_read,
1267 .start_addr = OPENPIC_CPU_REG_START,
1268 .size = OPENPIC_CPU_REG_SIZE,
1269};
1270
Scott Wood398d8782013-04-30 14:57:13 +00001271static const struct mem_reg openpic_src_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001272 .write = openpic_src_write,
1273 .read = openpic_src_read,
1274 .start_addr = OPENPIC_SRC_REG_START,
1275 .size = OPENPIC_SRC_REG_SIZE,
1276};
1277
Scott Wood398d8782013-04-30 14:57:13 +00001278static const struct mem_reg openpic_msi_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001279 .read = openpic_msi_read,
1280 .write = openpic_msi_write,
1281 .start_addr = OPENPIC_MSI_REG_START,
1282 .size = OPENPIC_MSI_REG_SIZE,
1283};
1284
Scott Wood398d8782013-04-30 14:57:13 +00001285static const struct mem_reg openpic_summary_mmio = {
Scott Wood5df554a2013-04-12 14:08:46 +00001286 .read = openpic_summary_read,
1287 .write = openpic_summary_write,
1288 .start_addr = OPENPIC_SUMMARY_REG_START,
1289 .size = OPENPIC_SUMMARY_REG_SIZE,
1290};
1291
Scott Wood398d8782013-04-30 14:57:13 +00001292static void add_mmio_region(struct openpic *opp, const struct mem_reg *mr)
1293{
1294 if (opp->num_mmio_regions >= MAX_MMIO_REGIONS) {
1295 WARN(1, "kvm mpic: too many mmio regions\n");
1296 return;
1297 }
1298
1299 opp->mmio_regions[opp->num_mmio_regions++] = mr;
1300}
1301
Scott Woodf0f5c482013-04-12 14:08:45 +00001302static void fsl_common_init(struct openpic *opp)
Scott Woodb823f982013-04-12 14:08:43 +00001303{
1304 int i;
1305 int virq = MAX_SRC;
1306
Scott Wood398d8782013-04-30 14:57:13 +00001307 add_mmio_region(opp, &openpic_msi_mmio);
1308 add_mmio_region(opp, &openpic_summary_mmio);
Scott Wood5df554a2013-04-12 14:08:46 +00001309
Scott Woodb823f982013-04-12 14:08:43 +00001310 opp->vid = VID_REVISION_1_2;
1311 opp->vir = VIR_GENERIC;
1312 opp->vector_mask = 0xFFFF;
1313 opp->tfrr_reset = 0;
1314 opp->ivpr_reset = IVPR_MASK_MASK;
1315 opp->idr_reset = 1 << 0;
1316 opp->max_irq = MAX_IRQ;
1317
1318 opp->irq_ipi0 = virq;
1319 virq += MAX_IPI;
1320 opp->irq_tim0 = virq;
1321 virq += MAX_TMR;
1322
Scott Wood5df554a2013-04-12 14:08:46 +00001323 BUG_ON(virq > MAX_IRQ);
Scott Woodb823f982013-04-12 14:08:43 +00001324
1325 opp->irq_msi = 224;
1326
Scott Woodf0f5c482013-04-12 14:08:45 +00001327 for (i = 0; i < opp->fsl->max_ext; i++)
Scott Woodb823f982013-04-12 14:08:43 +00001328 opp->src[i].level = false;
Scott Woodb823f982013-04-12 14:08:43 +00001329
1330 /* Internal interrupts, including message and MSI */
1331 for (i = 16; i < MAX_SRC; i++) {
1332 opp->src[i].type = IRQ_TYPE_FSLINT;
1333 opp->src[i].level = true;
1334 }
1335
1336 /* timers and IPIs */
1337 for (i = MAX_SRC; i < virq; i++) {
1338 opp->src[i].type = IRQ_TYPE_FSLSPECIAL;
1339 opp->src[i].level = false;
1340 }
1341}
1342
Scott Wood5df554a2013-04-12 14:08:46 +00001343static int kvm_mpic_read_internal(struct openpic *opp, gpa_t addr, u32 *ptr)
Scott Woodb823f982013-04-12 14:08:43 +00001344{
Scott Wood398d8782013-04-30 14:57:13 +00001345 int i;
Scott Woodb823f982013-04-12 14:08:43 +00001346
Scott Wood398d8782013-04-30 14:57:13 +00001347 for (i = 0; i < opp->num_mmio_regions; i++) {
1348 const struct mem_reg *mr = opp->mmio_regions[i];
Scott Woodb823f982013-04-12 14:08:43 +00001349
Scott Wood5df554a2013-04-12 14:08:46 +00001350 if (mr->start_addr > addr || addr >= mr->start_addr + mr->size)
1351 continue;
Scott Woodb823f982013-04-12 14:08:43 +00001352
Scott Wood5df554a2013-04-12 14:08:46 +00001353 return mr->read(opp, addr - mr->start_addr, ptr);
Scott Woodb823f982013-04-12 14:08:43 +00001354 }
Scott Wood5df554a2013-04-12 14:08:46 +00001355
1356 return -ENXIO;
Scott Woodb823f982013-04-12 14:08:43 +00001357}
1358
Scott Wood5df554a2013-04-12 14:08:46 +00001359static int kvm_mpic_write_internal(struct openpic *opp, gpa_t addr, u32 val)
Scott Woodb823f982013-04-12 14:08:43 +00001360{
Scott Wood398d8782013-04-30 14:57:13 +00001361 int i;
Scott Woodb823f982013-04-12 14:08:43 +00001362
Scott Wood398d8782013-04-30 14:57:13 +00001363 for (i = 0; i < opp->num_mmio_regions; i++) {
1364 const struct mem_reg *mr = opp->mmio_regions[i];
Scott Wood5df554a2013-04-12 14:08:46 +00001365
1366 if (mr->start_addr > addr || addr >= mr->start_addr + mr->size)
1367 continue;
1368
1369 return mr->write(opp, addr - mr->start_addr, val);
1370 }
1371
1372 return -ENXIO;
1373}
1374
1375static int kvm_mpic_read(struct kvm_io_device *this, gpa_t addr,
1376 int len, void *ptr)
1377{
1378 struct openpic *opp = container_of(this, struct openpic, mmio);
1379 int ret;
1380 union {
1381 u32 val;
1382 u8 bytes[4];
1383 } u;
1384
1385 if (addr & (len - 1)) {
1386 pr_debug("%s: bad alignment %llx/%d\n",
1387 __func__, addr, len);
1388 return -EINVAL;
1389 }
1390
1391 spin_lock_irq(&opp->lock);
1392 ret = kvm_mpic_read_internal(opp, addr - opp->reg_base, &u.val);
1393 spin_unlock_irq(&opp->lock);
1394
1395 /*
1396 * Technically only 32-bit accesses are allowed, but be nice to
1397 * people dumping registers a byte at a time -- it works in real
1398 * hardware (reads only, not writes).
1399 */
1400 if (len == 4) {
1401 *(u32 *)ptr = u.val;
1402 pr_debug("%s: addr %llx ret %d len 4 val %x\n",
1403 __func__, addr, ret, u.val);
1404 } else if (len == 1) {
1405 *(u8 *)ptr = u.bytes[addr & 3];
1406 pr_debug("%s: addr %llx ret %d len 1 val %x\n",
1407 __func__, addr, ret, u.bytes[addr & 3]);
1408 } else {
1409 pr_debug("%s: bad length %d\n", __func__, len);
1410 return -EINVAL;
1411 }
1412
1413 return ret;
1414}
1415
1416static int kvm_mpic_write(struct kvm_io_device *this, gpa_t addr,
1417 int len, const void *ptr)
1418{
1419 struct openpic *opp = container_of(this, struct openpic, mmio);
1420 int ret;
1421
1422 if (len != 4) {
1423 pr_debug("%s: bad length %d\n", __func__, len);
1424 return -EOPNOTSUPP;
1425 }
1426 if (addr & 3) {
1427 pr_debug("%s: bad alignment %llx/%d\n", __func__, addr, len);
1428 return -EOPNOTSUPP;
1429 }
1430
1431 spin_lock_irq(&opp->lock);
1432 ret = kvm_mpic_write_internal(opp, addr - opp->reg_base,
1433 *(const u32 *)ptr);
1434 spin_unlock_irq(&opp->lock);
1435
1436 pr_debug("%s: addr %llx ret %d val %x\n",
1437 __func__, addr, ret, *(const u32 *)ptr);
1438
1439 return ret;
1440}
1441
Scott Wood5df554a2013-04-12 14:08:46 +00001442static const struct kvm_io_device_ops mpic_mmio_ops = {
1443 .read = kvm_mpic_read,
1444 .write = kvm_mpic_write,
Scott Wood5df554a2013-04-12 14:08:46 +00001445};
1446
1447static void map_mmio(struct openpic *opp)
1448{
Scott Wood5df554a2013-04-12 14:08:46 +00001449 kvm_iodevice_init(&opp->mmio, &mpic_mmio_ops);
1450
1451 kvm_io_bus_register_dev(opp->kvm, KVM_MMIO_BUS,
1452 opp->reg_base, OPENPIC_REG_SIZE,
1453 &opp->mmio);
1454}
1455
1456static void unmap_mmio(struct openpic *opp)
1457{
Scott Wood91194912013-04-25 14:11:24 +00001458 kvm_io_bus_unregister_dev(opp->kvm, KVM_MMIO_BUS, &opp->mmio);
Scott Wood5df554a2013-04-12 14:08:46 +00001459}
1460
1461static int set_base_addr(struct openpic *opp, struct kvm_device_attr *attr)
1462{
1463 u64 base;
1464
1465 if (copy_from_user(&base, (u64 __user *)(long)attr->addr, sizeof(u64)))
1466 return -EFAULT;
1467
1468 if (base & 0x3ffff) {
1469 pr_debug("kvm mpic %s: KVM_DEV_MPIC_BASE_ADDR %08llx not aligned\n",
1470 __func__, base);
1471 return -EINVAL;
1472 }
1473
1474 if (base == opp->reg_base)
1475 return 0;
1476
1477 mutex_lock(&opp->kvm->slots_lock);
1478
1479 unmap_mmio(opp);
1480 opp->reg_base = base;
1481
1482 pr_debug("kvm mpic %s: KVM_DEV_MPIC_BASE_ADDR %08llx\n",
1483 __func__, base);
1484
1485 if (base == 0)
1486 goto out;
1487
1488 map_mmio(opp);
1489
1490 mutex_unlock(&opp->kvm->slots_lock);
1491out:
1492 return 0;
1493}
1494
1495#define ATTR_SET 0
1496#define ATTR_GET 1
1497
1498static int access_reg(struct openpic *opp, gpa_t addr, u32 *val, int type)
1499{
1500 int ret;
1501
1502 if (addr & 3)
1503 return -ENXIO;
1504
1505 spin_lock_irq(&opp->lock);
1506
1507 if (type == ATTR_SET)
1508 ret = kvm_mpic_write_internal(opp, addr, *val);
1509 else
1510 ret = kvm_mpic_read_internal(opp, addr, val);
1511
1512 spin_unlock_irq(&opp->lock);
1513
1514 pr_debug("%s: type %d addr %llx val %x\n", __func__, type, addr, *val);
1515
1516 return ret;
1517}
1518
1519static int mpic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr)
1520{
1521 struct openpic *opp = dev->private;
1522 u32 attr32;
1523
1524 switch (attr->group) {
1525 case KVM_DEV_MPIC_GRP_MISC:
1526 switch (attr->attr) {
1527 case KVM_DEV_MPIC_BASE_ADDR:
1528 return set_base_addr(opp, attr);
1529 }
1530
1531 break;
1532
1533 case KVM_DEV_MPIC_GRP_REGISTER:
1534 if (get_user(attr32, (u32 __user *)(long)attr->addr))
1535 return -EFAULT;
1536
1537 return access_reg(opp, attr->attr, &attr32, ATTR_SET);
1538
1539 case KVM_DEV_MPIC_GRP_IRQ_ACTIVE:
1540 if (attr->attr > MAX_SRC)
1541 return -EINVAL;
1542
1543 if (get_user(attr32, (u32 __user *)(long)attr->addr))
1544 return -EFAULT;
1545
1546 if (attr32 != 0 && attr32 != 1)
1547 return -EINVAL;
1548
1549 spin_lock_irq(&opp->lock);
1550 openpic_set_irq(opp, attr->attr, attr32);
1551 spin_unlock_irq(&opp->lock);
1552 return 0;
1553 }
1554
1555 return -ENXIO;
1556}
1557
1558static int mpic_get_attr(struct kvm_device *dev, struct kvm_device_attr *attr)
1559{
1560 struct openpic *opp = dev->private;
1561 u64 attr64;
1562 u32 attr32;
1563 int ret;
1564
1565 switch (attr->group) {
1566 case KVM_DEV_MPIC_GRP_MISC:
1567 switch (attr->attr) {
1568 case KVM_DEV_MPIC_BASE_ADDR:
1569 mutex_lock(&opp->kvm->slots_lock);
1570 attr64 = opp->reg_base;
1571 mutex_unlock(&opp->kvm->slots_lock);
1572
1573 if (copy_to_user((u64 __user *)(long)attr->addr,
1574 &attr64, sizeof(u64)))
1575 return -EFAULT;
1576
1577 return 0;
1578 }
1579
1580 break;
1581
1582 case KVM_DEV_MPIC_GRP_REGISTER:
1583 ret = access_reg(opp, attr->attr, &attr32, ATTR_GET);
1584 if (ret)
1585 return ret;
1586
1587 if (put_user(attr32, (u32 __user *)(long)attr->addr))
1588 return -EFAULT;
1589
1590 return 0;
1591
1592 case KVM_DEV_MPIC_GRP_IRQ_ACTIVE:
1593 if (attr->attr > MAX_SRC)
1594 return -EINVAL;
1595
1596 spin_lock_irq(&opp->lock);
1597 attr32 = opp->src[attr->attr].pending;
1598 spin_unlock_irq(&opp->lock);
1599
1600 if (put_user(attr32, (u32 __user *)(long)attr->addr))
1601 return -EFAULT;
1602
1603 return 0;
1604 }
1605
1606 return -ENXIO;
1607}
1608
1609static int mpic_has_attr(struct kvm_device *dev, struct kvm_device_attr *attr)
1610{
1611 switch (attr->group) {
1612 case KVM_DEV_MPIC_GRP_MISC:
1613 switch (attr->attr) {
1614 case KVM_DEV_MPIC_BASE_ADDR:
1615 return 0;
1616 }
1617
1618 break;
1619
1620 case KVM_DEV_MPIC_GRP_REGISTER:
1621 return 0;
1622
1623 case KVM_DEV_MPIC_GRP_IRQ_ACTIVE:
1624 if (attr->attr > MAX_SRC)
1625 break;
1626
1627 return 0;
1628 }
1629
1630 return -ENXIO;
1631}
1632
1633static void mpic_destroy(struct kvm_device *dev)
1634{
1635 struct openpic *opp = dev->private;
1636
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001637 dev->kvm->arch.mpic = NULL;
Scott Wood5df554a2013-04-12 14:08:46 +00001638 kfree(opp);
1639}
1640
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001641static int mpic_set_default_irq_routing(struct openpic *opp)
1642{
1643 struct kvm_irq_routing_entry *routing;
1644
1645 /* Create a nop default map, so that dereferencing it still works */
1646 routing = kzalloc((sizeof(*routing)), GFP_KERNEL);
1647 if (!routing)
1648 return -ENOMEM;
1649
1650 kvm_set_irq_routing(opp->kvm, routing, 0, 0);
1651
1652 kfree(routing);
1653 return 0;
1654}
1655
Scott Wood5df554a2013-04-12 14:08:46 +00001656static int mpic_create(struct kvm_device *dev, u32 type)
1657{
1658 struct openpic *opp;
1659 int ret;
1660
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001661 /* We only support one MPIC at a time for now */
1662 if (dev->kvm->arch.mpic)
1663 return -EINVAL;
1664
Scott Wood5df554a2013-04-12 14:08:46 +00001665 opp = kzalloc(sizeof(struct openpic), GFP_KERNEL);
1666 if (!opp)
1667 return -ENOMEM;
1668
1669 dev->private = opp;
1670 opp->kvm = dev->kvm;
1671 opp->dev = dev;
1672 opp->model = type;
1673 spin_lock_init(&opp->lock);
1674
Scott Wood398d8782013-04-30 14:57:13 +00001675 add_mmio_region(opp, &openpic_gbl_mmio);
1676 add_mmio_region(opp, &openpic_tmr_mmio);
1677 add_mmio_region(opp, &openpic_src_mmio);
1678 add_mmio_region(opp, &openpic_cpu_mmio);
Scott Woodb823f982013-04-12 14:08:43 +00001679
1680 switch (opp->model) {
Scott Wood5df554a2013-04-12 14:08:46 +00001681 case KVM_DEV_TYPE_FSL_MPIC_20:
Scott Woodb823f982013-04-12 14:08:43 +00001682 opp->fsl = &fsl_mpic_20;
1683 opp->brr1 = 0x00400200;
1684 opp->flags |= OPENPIC_FLAG_IDR_CRIT;
1685 opp->nb_irqs = 80;
1686 opp->mpic_mode_mask = GCR_MODE_MIXED;
1687
1688 fsl_common_init(opp);
Scott Woodb823f982013-04-12 14:08:43 +00001689
1690 break;
1691
Scott Wood5df554a2013-04-12 14:08:46 +00001692 case KVM_DEV_TYPE_FSL_MPIC_42:
Scott Woodb823f982013-04-12 14:08:43 +00001693 opp->fsl = &fsl_mpic_42;
1694 opp->brr1 = 0x00400402;
1695 opp->flags |= OPENPIC_FLAG_ILR;
1696 opp->nb_irqs = 196;
1697 opp->mpic_mode_mask = GCR_MODE_PROXY;
1698
1699 fsl_common_init(opp);
Scott Woodb823f982013-04-12 14:08:43 +00001700
1701 break;
Scott Wood5df554a2013-04-12 14:08:46 +00001702
1703 default:
1704 ret = -ENODEV;
1705 goto err;
Scott Woodb823f982013-04-12 14:08:43 +00001706 }
1707
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001708 ret = mpic_set_default_irq_routing(opp);
1709 if (ret)
1710 goto err;
1711
Scott Wood5df554a2013-04-12 14:08:46 +00001712 openpic_reset(opp);
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001713
1714 smp_wmb();
1715 dev->kvm->arch.mpic = opp;
1716
Scott Woodb823f982013-04-12 14:08:43 +00001717 return 0;
Scott Wood5df554a2013-04-12 14:08:46 +00001718
1719err:
1720 kfree(opp);
1721 return ret;
Scott Woodb823f982013-04-12 14:08:43 +00001722}
Scott Wood5df554a2013-04-12 14:08:46 +00001723
1724struct kvm_device_ops kvm_mpic_ops = {
1725 .name = "kvm-mpic",
1726 .create = mpic_create,
1727 .destroy = mpic_destroy,
1728 .set_attr = mpic_set_attr,
1729 .get_attr = mpic_get_attr,
1730 .has_attr = mpic_has_attr,
1731};
Scott Woodeb1e4f42013-04-12 14:08:47 +00001732
1733int kvmppc_mpic_connect_vcpu(struct kvm_device *dev, struct kvm_vcpu *vcpu,
1734 u32 cpu)
1735{
1736 struct openpic *opp = dev->private;
1737 int ret = 0;
1738
1739 if (dev->ops != &kvm_mpic_ops)
1740 return -EPERM;
1741 if (opp->kvm != vcpu->kvm)
1742 return -EPERM;
1743 if (cpu < 0 || cpu >= MAX_CPU)
1744 return -EPERM;
1745
1746 spin_lock_irq(&opp->lock);
1747
1748 if (opp->dst[cpu].vcpu) {
1749 ret = -EEXIST;
1750 goto out;
1751 }
1752 if (vcpu->arch.irq_type) {
1753 ret = -EBUSY;
1754 goto out;
1755 }
1756
1757 opp->dst[cpu].vcpu = vcpu;
1758 opp->nb_cpus = max(opp->nb_cpus, cpu + 1);
1759
1760 vcpu->arch.mpic = opp;
1761 vcpu->arch.irq_cpu_id = cpu;
1762 vcpu->arch.irq_type = KVMPPC_IRQ_MPIC;
1763
1764 /* This might need to be changed if GCR gets extended */
1765 if (opp->mpic_mode_mask == GCR_MODE_PROXY)
1766 vcpu->arch.epr_flags |= KVMPPC_EPR_KERNEL;
1767
Scott Woodeb1e4f42013-04-12 14:08:47 +00001768out:
1769 spin_unlock_irq(&opp->lock);
1770 return ret;
1771}
1772
1773/*
1774 * This should only happen immediately before the mpic is destroyed,
1775 * so we shouldn't need to worry about anything still trying to
1776 * access the vcpu pointer.
1777 */
1778void kvmppc_mpic_disconnect_vcpu(struct openpic *opp, struct kvm_vcpu *vcpu)
1779{
1780 BUG_ON(!opp->dst[vcpu->arch.irq_cpu_id].vcpu);
1781
1782 opp->dst[vcpu->arch.irq_cpu_id].vcpu = NULL;
Scott Woodeb1e4f42013-04-12 14:08:47 +00001783}
Alexander Grafde9ba2f2013-04-16 17:42:19 +02001784
1785/*
1786 * Return value:
1787 * < 0 Interrupt was ignored (masked or not delivered for other reasons)
1788 * = 0 Interrupt was coalesced (previous irq is still pending)
1789 * > 0 Number of CPUs interrupt was delivered to
1790 */
1791static int mpic_set_irq(struct kvm_kernel_irq_routing_entry *e,
1792 struct kvm *kvm, int irq_source_id, int level,
1793 bool line_status)
1794{
1795 u32 irq = e->irqchip.pin;
1796 struct openpic *opp = kvm->arch.mpic;
1797 unsigned long flags;
1798
1799 spin_lock_irqsave(&opp->lock, flags);
1800 openpic_set_irq(opp, irq, level);
1801 spin_unlock_irqrestore(&opp->lock, flags);
1802
1803 /* All code paths we care about don't check for the return value */
1804 return 0;
1805}
1806
1807int kvm_set_msi(struct kvm_kernel_irq_routing_entry *e,
1808 struct kvm *kvm, int irq_source_id, int level, bool line_status)
1809{
1810 struct openpic *opp = kvm->arch.mpic;
1811 unsigned long flags;
1812
1813 spin_lock_irqsave(&opp->lock, flags);
1814
1815 /*
1816 * XXX We ignore the target address for now, as we only support
1817 * a single MSI bank.
1818 */
1819 openpic_msi_write(kvm->arch.mpic, MSIIR_OFFSET, e->msi.data);
1820 spin_unlock_irqrestore(&opp->lock, flags);
1821
1822 /* All code paths we care about don't check for the return value */
1823 return 0;
1824}
1825
1826int kvm_set_routing_entry(struct kvm_irq_routing_table *rt,
1827 struct kvm_kernel_irq_routing_entry *e,
1828 const struct kvm_irq_routing_entry *ue)
1829{
1830 int r = -EINVAL;
1831
1832 switch (ue->type) {
1833 case KVM_IRQ_ROUTING_IRQCHIP:
1834 e->set = mpic_set_irq;
1835 e->irqchip.irqchip = ue->u.irqchip.irqchip;
1836 e->irqchip.pin = ue->u.irqchip.pin;
1837 if (e->irqchip.pin >= KVM_IRQCHIP_NUM_PINS)
1838 goto out;
1839 rt->chip[ue->u.irqchip.irqchip][e->irqchip.pin] = ue->gsi;
1840 break;
1841 case KVM_IRQ_ROUTING_MSI:
1842 e->set = kvm_set_msi;
1843 e->msi.address_lo = ue->u.msi.address_lo;
1844 e->msi.address_hi = ue->u.msi.address_hi;
1845 e->msi.data = ue->u.msi.data;
1846 break;
1847 default:
1848 goto out;
1849 }
1850
1851 r = 0;
1852out:
1853 return r;
1854}