xref: /qemu/include/hw/ppc/openpic.h (revision db1015e92e04835c9eb50c29625fe566d1202dbd)
12a6a4076SMarkus Armbruster #ifndef OPENPIC_H
22a6a4076SMarkus Armbruster #define OPENPIC_H
3b7169916Saurel32 
4f7bd7941SMark Cave-Ayland #include "hw/sysbus.h"
52e5b09fdSMarkus Armbruster #include "hw/core/cpu.h"
6*db1015e9SEduardo Habkost #include "qom/object.h"
78935a442SScott Wood 
8f7bd7941SMark Cave-Ayland #define MAX_CPU     32
9f7bd7941SMark Cave-Ayland #define MAX_MSI     8
10f7bd7941SMark Cave-Ayland #define VID         0x03 /* MPIC version ID */
11e1766344SAndreas Färber 
12b7169916Saurel32 /* OpenPIC have 5 outputs per CPU connected and one IRQ out single output */
13b7169916Saurel32 enum {
14b7169916Saurel32     OPENPIC_OUTPUT_INT = 0, /* IRQ                       */
15b7169916Saurel32     OPENPIC_OUTPUT_CINT,    /* critical IRQ              */
16b7169916Saurel32     OPENPIC_OUTPUT_MCK,     /* Machine check event       */
17b7169916Saurel32     OPENPIC_OUTPUT_DEBUG,   /* Inconditional debug event */
18b7169916Saurel32     OPENPIC_OUTPUT_RESET,   /* Core reset event          */
19b7169916Saurel32     OPENPIC_OUTPUT_NB,
20b7169916Saurel32 };
21b7169916Saurel32 
229929301eSGreg Kurz typedef struct IrqLines { qemu_irq irq[OPENPIC_OUTPUT_NB]; } IrqLines;
239929301eSGreg Kurz 
24d0b72631SAlexander Graf #define OPENPIC_MODEL_RAVEN       0
25d0b72631SAlexander Graf #define OPENPIC_MODEL_FSL_MPIC_20 1
26e0dfe5b1SScott Wood #define OPENPIC_MODEL_FSL_MPIC_42 2
2758b62835SBenjamin Herrenschmidt #define OPENPIC_MODEL_KEYLARGO    3
285861a338SAlexander Graf 
298935a442SScott Wood #define OPENPIC_MAX_SRC     256
308935a442SScott Wood #define OPENPIC_MAX_TMR     4
318935a442SScott Wood #define OPENPIC_MAX_IPI     4
328935a442SScott Wood #define OPENPIC_MAX_IRQ     (OPENPIC_MAX_SRC + OPENPIC_MAX_IPI + \
338935a442SScott Wood                              OPENPIC_MAX_TMR)
348935a442SScott Wood 
35f7bd7941SMark Cave-Ayland /* Raven */
36f7bd7941SMark Cave-Ayland #define RAVEN_MAX_CPU      2
37f7bd7941SMark Cave-Ayland #define RAVEN_MAX_EXT     48
38f7bd7941SMark Cave-Ayland #define RAVEN_MAX_IRQ     64
39f7bd7941SMark Cave-Ayland #define RAVEN_MAX_TMR      OPENPIC_MAX_TMR
40f7bd7941SMark Cave-Ayland #define RAVEN_MAX_IPI      OPENPIC_MAX_IPI
41f7bd7941SMark Cave-Ayland 
42f7bd7941SMark Cave-Ayland /* KeyLargo */
43f7bd7941SMark Cave-Ayland #define KEYLARGO_MAX_CPU  4
44f7bd7941SMark Cave-Ayland #define KEYLARGO_MAX_EXT  64
45f7bd7941SMark Cave-Ayland #define KEYLARGO_MAX_IPI  4
46f7bd7941SMark Cave-Ayland #define KEYLARGO_MAX_IRQ  (64 + KEYLARGO_MAX_IPI)
47f7bd7941SMark Cave-Ayland #define KEYLARGO_MAX_TMR  0
48f7bd7941SMark Cave-Ayland #define KEYLARGO_IPI_IRQ  (KEYLARGO_MAX_EXT) /* First IPI IRQ */
49f7bd7941SMark Cave-Ayland /* Timers don't exist but this makes the code happy... */
50f7bd7941SMark Cave-Ayland #define KEYLARGO_TMR_IRQ  (KEYLARGO_IPI_IRQ + KEYLARGO_MAX_IPI)
51f7bd7941SMark Cave-Ayland 
52f7bd7941SMark Cave-Ayland /* Interrupt definitions */
53f7bd7941SMark Cave-Ayland #define RAVEN_FE_IRQ     (RAVEN_MAX_EXT)     /* Internal functional IRQ */
54f7bd7941SMark Cave-Ayland #define RAVEN_ERR_IRQ    (RAVEN_MAX_EXT + 1) /* Error IRQ */
55f7bd7941SMark Cave-Ayland #define RAVEN_TMR_IRQ    (RAVEN_MAX_EXT + 2) /* First timer IRQ */
56f7bd7941SMark Cave-Ayland #define RAVEN_IPI_IRQ    (RAVEN_TMR_IRQ + RAVEN_MAX_TMR) /* First IPI IRQ */
57f7bd7941SMark Cave-Ayland /* First doorbell IRQ */
58f7bd7941SMark Cave-Ayland #define RAVEN_DBL_IRQ    (RAVEN_IPI_IRQ + (RAVEN_MAX_CPU * RAVEN_MAX_IPI))
59f7bd7941SMark Cave-Ayland 
60f7bd7941SMark Cave-Ayland typedef struct FslMpicInfo {
61f7bd7941SMark Cave-Ayland     int max_ext;
62f7bd7941SMark Cave-Ayland } FslMpicInfo;
63f7bd7941SMark Cave-Ayland 
64f7bd7941SMark Cave-Ayland typedef enum IRQType {
65f7bd7941SMark Cave-Ayland     IRQ_TYPE_NORMAL = 0,
66f7bd7941SMark Cave-Ayland     IRQ_TYPE_FSLINT,        /* FSL internal interrupt -- level only */
67f7bd7941SMark Cave-Ayland     IRQ_TYPE_FSLSPECIAL,    /* FSL timer/IPI interrupt, edge, no polarity */
68f7bd7941SMark Cave-Ayland } IRQType;
69f7bd7941SMark Cave-Ayland 
70f7bd7941SMark Cave-Ayland /* Round up to the nearest 64 IRQs so that the queue length
71f7bd7941SMark Cave-Ayland  * won't change when moving between 32 and 64 bit hosts.
72f7bd7941SMark Cave-Ayland  */
73f7bd7941SMark Cave-Ayland #define IRQQUEUE_SIZE_BITS ((OPENPIC_MAX_IRQ + 63) & ~63)
74f7bd7941SMark Cave-Ayland 
75f7bd7941SMark Cave-Ayland typedef struct IRQQueue {
76f7bd7941SMark Cave-Ayland     unsigned long *queue;
77f7bd7941SMark Cave-Ayland     int32_t queue_size; /* Only used for VMSTATE_BITMAP */
78f7bd7941SMark Cave-Ayland     int next;
79f7bd7941SMark Cave-Ayland     int priority;
80f7bd7941SMark Cave-Ayland } IRQQueue;
81f7bd7941SMark Cave-Ayland 
82f7bd7941SMark Cave-Ayland typedef struct IRQSource {
83f7bd7941SMark Cave-Ayland     uint32_t ivpr;  /* IRQ vector/priority register */
84f7bd7941SMark Cave-Ayland     uint32_t idr;   /* IRQ destination register */
85f7bd7941SMark Cave-Ayland     uint32_t destmask; /* bitmap of CPU destinations */
86f7bd7941SMark Cave-Ayland     int last_cpu;
87f7bd7941SMark Cave-Ayland     int output;     /* IRQ level, e.g. OPENPIC_OUTPUT_INT */
88f7bd7941SMark Cave-Ayland     int pending;    /* TRUE if IRQ is pending */
89f7bd7941SMark Cave-Ayland     IRQType type;
90f7bd7941SMark Cave-Ayland     bool level:1;   /* level-triggered */
91f7bd7941SMark Cave-Ayland     bool nomask:1;  /* critical interrupts ignore mask on some FSL MPICs */
92f7bd7941SMark Cave-Ayland } IRQSource;
93f7bd7941SMark Cave-Ayland 
94f7bd7941SMark Cave-Ayland #define IVPR_MASK_SHIFT       31
95f7bd7941SMark Cave-Ayland #define IVPR_MASK_MASK        (1U << IVPR_MASK_SHIFT)
96f7bd7941SMark Cave-Ayland #define IVPR_ACTIVITY_SHIFT   30
97f7bd7941SMark Cave-Ayland #define IVPR_ACTIVITY_MASK    (1U << IVPR_ACTIVITY_SHIFT)
98f7bd7941SMark Cave-Ayland #define IVPR_MODE_SHIFT       29
99f7bd7941SMark Cave-Ayland #define IVPR_MODE_MASK        (1U << IVPR_MODE_SHIFT)
100f7bd7941SMark Cave-Ayland #define IVPR_POLARITY_SHIFT   23
101f7bd7941SMark Cave-Ayland #define IVPR_POLARITY_MASK    (1U << IVPR_POLARITY_SHIFT)
102f7bd7941SMark Cave-Ayland #define IVPR_SENSE_SHIFT      22
103f7bd7941SMark Cave-Ayland #define IVPR_SENSE_MASK       (1U << IVPR_SENSE_SHIFT)
104f7bd7941SMark Cave-Ayland 
105f7bd7941SMark Cave-Ayland #define IVPR_PRIORITY_MASK     (0xFU << 16)
106f7bd7941SMark Cave-Ayland #define IVPR_PRIORITY(_ivprr_) ((int)(((_ivprr_) & IVPR_PRIORITY_MASK) >> 16))
107f7bd7941SMark Cave-Ayland #define IVPR_VECTOR(opp, _ivprr_) ((_ivprr_) & (opp)->vector_mask)
108f7bd7941SMark Cave-Ayland 
109f7bd7941SMark Cave-Ayland /* IDR[EP/CI] are only for FSL MPIC prior to v4.0 */
110f7bd7941SMark Cave-Ayland #define IDR_EP      0x80000000  /* external pin */
111f7bd7941SMark Cave-Ayland #define IDR_CI      0x40000000  /* critical interrupt */
112f7bd7941SMark Cave-Ayland 
113f7bd7941SMark Cave-Ayland typedef struct OpenPICTimer {
114f7bd7941SMark Cave-Ayland     uint32_t tccr;  /* Global timer current count register */
115f7bd7941SMark Cave-Ayland     uint32_t tbcr;  /* Global timer base count register */
116f7bd7941SMark Cave-Ayland     int                   n_IRQ;
117f7bd7941SMark Cave-Ayland     bool                  qemu_timer_active; /* Is the qemu_timer is running? */
118f7bd7941SMark Cave-Ayland     struct QEMUTimer     *qemu_timer;
119f7bd7941SMark Cave-Ayland     struct OpenPICState  *opp;          /* Device timer is part of. */
120f7bd7941SMark Cave-Ayland     /* The QEMU_CLOCK_VIRTUAL time (in ns) corresponding to the last
121f7bd7941SMark Cave-Ayland        current_count written or read, only defined if qemu_timer_active. */
122f7bd7941SMark Cave-Ayland     uint64_t              origin_time;
123f7bd7941SMark Cave-Ayland } OpenPICTimer;
124f7bd7941SMark Cave-Ayland 
125f7bd7941SMark Cave-Ayland typedef struct OpenPICMSI {
126f7bd7941SMark Cave-Ayland     uint32_t msir;   /* Shared Message Signaled Interrupt Register */
127f7bd7941SMark Cave-Ayland } OpenPICMSI;
128f7bd7941SMark Cave-Ayland 
129f7bd7941SMark Cave-Ayland typedef struct IRQDest {
130f7bd7941SMark Cave-Ayland     int32_t ctpr; /* CPU current task priority */
131f7bd7941SMark Cave-Ayland     IRQQueue raised;
132f7bd7941SMark Cave-Ayland     IRQQueue servicing;
133f7bd7941SMark Cave-Ayland     qemu_irq *irqs;
134f7bd7941SMark Cave-Ayland 
135f7bd7941SMark Cave-Ayland     /* Count of IRQ sources asserting on non-INT outputs */
136f7bd7941SMark Cave-Ayland     uint32_t outputs_active[OPENPIC_OUTPUT_NB];
137f7bd7941SMark Cave-Ayland } IRQDest;
138f7bd7941SMark Cave-Ayland 
139f7bd7941SMark Cave-Ayland #define TYPE_OPENPIC "openpic"
140*db1015e9SEduardo Habkost typedef struct OpenPICState OpenPICState;
141f7bd7941SMark Cave-Ayland #define OPENPIC(obj) OBJECT_CHECK(OpenPICState, (obj), TYPE_OPENPIC)
142f7bd7941SMark Cave-Ayland 
143*db1015e9SEduardo Habkost struct OpenPICState {
144f7bd7941SMark Cave-Ayland     /*< private >*/
145f7bd7941SMark Cave-Ayland     SysBusDevice parent_obj;
146f7bd7941SMark Cave-Ayland     /*< public >*/
147f7bd7941SMark Cave-Ayland 
148f7bd7941SMark Cave-Ayland     MemoryRegion mem;
149f7bd7941SMark Cave-Ayland 
150f7bd7941SMark Cave-Ayland     /* Behavior control */
151f7bd7941SMark Cave-Ayland     FslMpicInfo *fsl;
152f7bd7941SMark Cave-Ayland     uint32_t model;
153f7bd7941SMark Cave-Ayland     uint32_t flags;
154f7bd7941SMark Cave-Ayland     uint32_t nb_irqs;
155f7bd7941SMark Cave-Ayland     uint32_t vid;
156f7bd7941SMark Cave-Ayland     uint32_t vir; /* Vendor identification register */
157f7bd7941SMark Cave-Ayland     uint32_t vector_mask;
158f7bd7941SMark Cave-Ayland     uint32_t tfrr_reset;
159f7bd7941SMark Cave-Ayland     uint32_t ivpr_reset;
160f7bd7941SMark Cave-Ayland     uint32_t idr_reset;
161f7bd7941SMark Cave-Ayland     uint32_t brr1;
162f7bd7941SMark Cave-Ayland     uint32_t mpic_mode_mask;
163f7bd7941SMark Cave-Ayland 
164f7bd7941SMark Cave-Ayland     /* Sub-regions */
165f7bd7941SMark Cave-Ayland     MemoryRegion sub_io_mem[6];
166f7bd7941SMark Cave-Ayland 
167f7bd7941SMark Cave-Ayland     /* Global registers */
168f7bd7941SMark Cave-Ayland     uint32_t frr; /* Feature reporting register */
169f7bd7941SMark Cave-Ayland     uint32_t gcr; /* Global configuration register  */
170f7bd7941SMark Cave-Ayland     uint32_t pir; /* Processor initialization register */
171f7bd7941SMark Cave-Ayland     uint32_t spve; /* Spurious vector register */
172f7bd7941SMark Cave-Ayland     uint32_t tfrr; /* Timer frequency reporting register */
173f7bd7941SMark Cave-Ayland     /* Source registers */
174f7bd7941SMark Cave-Ayland     IRQSource src[OPENPIC_MAX_IRQ];
175f7bd7941SMark Cave-Ayland     /* Local registers per output pin */
176f7bd7941SMark Cave-Ayland     IRQDest dst[MAX_CPU];
177f7bd7941SMark Cave-Ayland     uint32_t nb_cpus;
178f7bd7941SMark Cave-Ayland     /* Timer registers */
179f7bd7941SMark Cave-Ayland     OpenPICTimer timers[OPENPIC_MAX_TMR];
180f7bd7941SMark Cave-Ayland     uint32_t max_tmr;
181f7bd7941SMark Cave-Ayland 
182f7bd7941SMark Cave-Ayland     /* Shared MSI registers */
183f7bd7941SMark Cave-Ayland     OpenPICMSI msi[MAX_MSI];
184f7bd7941SMark Cave-Ayland     uint32_t max_irq;
185f7bd7941SMark Cave-Ayland     uint32_t irq_ipi0;
186f7bd7941SMark Cave-Ayland     uint32_t irq_tim0;
187f7bd7941SMark Cave-Ayland     uint32_t irq_msi;
188*db1015e9SEduardo Habkost };
189f7bd7941SMark Cave-Ayland 
1902a6a4076SMarkus Armbruster #endif /* OPENPIC_H */
191