gem5  v21.0.1.0
gic_v3_distributor.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2019-2020 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Copyright (c) 2018 Metempsy Technology Consulting
15  * All rights reserved.
16  *
17  * Redistribution and use in source and binary forms, with or without
18  * modification, are permitted provided that the following conditions are
19  * met: redistributions of source code must retain the above copyright
20  * notice, this list of conditions and the following disclaimer;
21  * redistributions in binary form must reproduce the above copyright
22  * notice, this list of conditions and the following disclaimer in the
23  * documentation and/or other materials provided with the distribution;
24  * neither the name of the copyright holders nor the names of its
25  * contributors may be used to endorse or promote products derived from
26  * this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
31  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
32  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
33  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
34  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
35  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
36  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
37  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39  */
40 
42 
43 #include <algorithm>
44 
45 #include "base/intmath.hh"
46 #include "debug/GIC.hh"
47 #include "dev/arm/gic_v3.hh"
50 
51 const AddrRange Gicv3Distributor::GICD_IGROUPR (0x0080, 0x0100);
52 const AddrRange Gicv3Distributor::GICD_ISENABLER (0x0100, 0x0180);
53 const AddrRange Gicv3Distributor::GICD_ICENABLER (0x0180, 0x0200);
54 const AddrRange Gicv3Distributor::GICD_ISPENDR (0x0200, 0x0280);
55 const AddrRange Gicv3Distributor::GICD_ICPENDR (0x0280, 0x0300);
56 const AddrRange Gicv3Distributor::GICD_ISACTIVER (0x0300, 0x0380);
57 const AddrRange Gicv3Distributor::GICD_ICACTIVER (0x0380, 0x0400);
58 const AddrRange Gicv3Distributor::GICD_IPRIORITYR(0x0400, 0x0800);
59 const AddrRange Gicv3Distributor::GICD_ITARGETSR (0x0800, 0x0900);
60 const AddrRange Gicv3Distributor::GICD_ICFGR (0x0c00, 0x0d00);
61 const AddrRange Gicv3Distributor::GICD_IGRPMODR (0x0d00, 0x0d80);
62 const AddrRange Gicv3Distributor::GICD_NSACR (0x0e00, 0x0f00);
63 const AddrRange Gicv3Distributor::GICD_CPENDSGIR (0x0f10, 0x0f20);
64 const AddrRange Gicv3Distributor::GICD_SPENDSGIR (0x0f20, 0x0f30);
65 const AddrRange Gicv3Distributor::GICD_IROUTER (0x6000, 0x7fe0);
66 
68  : gic(gic),
69  itLines(it_lines),
70  ARE(true),
71  EnableGrp1S(0),
72  EnableGrp1NS(0),
73  EnableGrp0(0),
74  irqGroup(it_lines, 0),
75  irqEnabled(it_lines, false),
76  irqPending(it_lines, false),
77  irqPendingIspendr(it_lines, false),
78  irqActive(it_lines, false),
79  irqPriority(it_lines, 0xAA),
80  irqConfig(it_lines, Gicv3::INT_LEVEL_SENSITIVE),
81  irqGrpmod(it_lines, 0),
82  irqNsacr(it_lines, 0),
83  irqAffinityRouting(it_lines, 0),
84  gicdTyper(0),
85  gicdPidr0(0x92),
86  gicdPidr1(0xb4),
87  gicdPidr2(0x3b),
88  gicdPidr3(0),
89  gicdPidr4(0x44)
90 {
91  panic_if(it_lines > Gicv3::INTID_SECURE, "Invalid value for it_lines!");
92  /*
93  * RSS [26] == 1
94  * (The implementation does supports targeted SGIs with affinity
95  * level 0 values of 0 - 255)
96  * No1N [25] == 1
97  * (1 of N SPI interrupts are not supported)
98  * A3V [24] == 1
99  * (Supports nonzero values of Affinity level 3)
100  * IDbits [23:19] == 0xf
101  * (The number of interrupt identifier bits supported, minus one)
102  * DVIS [18] == 0
103  * (The implementation does not support Direct Virtual LPI
104  * injection)
105  * LPIS [17] == 1
106  * (The implementation does not support LPIs)
107  * MBIS [16] == 1
108  * (The implementation supports message-based interrupts
109  * by writing to Distributor registers)
110  * SecurityExtn [10] == X
111  * (The GIC implementation supports two Security states)
112  * CPUNumber [7:5] == 0
113  * (since for us ARE is always 1 [(ARE = 0) == Gicv2 legacy])
114  * ITLinesNumber [4:0] == N
115  * (MaxSPIIntId = 32 (N + 1) - 1)
116  */
117  int max_spi_int_id = itLines - 1;
118  int it_lines_number = divCeil(max_spi_int_id + 1, 32) - 1;
119  gicdTyper = (1 << 26) | (1 << 25) | (1 << 24) | (IDBITS << 19) |
120  (1 << 17) | (1 << 16) |
121  ((gic->getSystem()->haveSecurity() ? 1 : 0) << 10) |
122  (it_lines_number << 0);
123 
124  if (gic->getSystem()->haveSecurity()) {
125  DS = false;
126  } else {
127  DS = true;
128  }
129 }
130 
131 void
133 {
134 }
135 
136 uint64_t
137 Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access)
138 {
139  if (GICD_IGROUPR.contains(addr)) { // Interrupt Group Registers
140  uint64_t val = 0x0;
141 
142  if (!DS && !is_secure_access) {
143  // RAZ/WI for non-secure accesses
144  return 0;
145  }
146 
147  int first_intid = (addr - GICD_IGROUPR.start()) * 8;
148 
149  if (isNotSPI(first_intid)) {
150  return 0;
151  }
152 
153  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
154  i++, int_id++) {
155  val |= irqGroup[int_id] << i;
156  }
157 
158  return val;
159  } else if (GICD_ISENABLER.contains(addr)) {
160  // Interrupt Set-Enable Registers
161  uint64_t val = 0x0;
162  int first_intid = (addr - GICD_ISENABLER.start()) * 8;
163 
164  if (isNotSPI(first_intid)) {
165  return 0;
166  }
167 
168  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
169  i++, int_id++) {
170 
171  if (nsAccessToSecInt(int_id, is_secure_access))
172  {
173  continue;
174  }
175 
176  val |= irqEnabled[int_id] << i;
177  }
178 
179  return val;
180  } else if (GICD_ICENABLER.contains(addr)) {
181  // Interrupt Clear-Enable Registers
182  uint64_t val = 0x0;
183  int first_intid = (addr - GICD_ICENABLER.start()) * 8;
184 
185  if (isNotSPI(first_intid)) {
186  return 0;
187  }
188 
189  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
190  i++, int_id++) {
191 
192  if (nsAccessToSecInt(int_id, is_secure_access))
193  {
194  continue;
195  }
196 
197  val |= (irqEnabled[int_id] << i);
198  }
199 
200  return val;
201  } else if (GICD_ISPENDR.contains(addr)) {
202  // Interrupt Set-Pending Registers
203  uint64_t val = 0x0;
204  int first_intid = (addr - GICD_ISPENDR.start()) * 8;
205 
206  if (isNotSPI(first_intid)) {
207  return 0;
208  }
209 
210  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
211  i++, int_id++) {
212 
213  if (nsAccessToSecInt(int_id, is_secure_access))
214  {
215  if (irqNsacr[int_id] == 0) {
216  // Group 0 or Secure Group 1 interrupts are RAZ/WI
217  continue;
218  }
219  }
220 
221  val |= (irqPending[int_id] << i);
222  }
223 
224  return val;
225  } else if (GICD_ICPENDR.contains(addr)) {
226  // Interrupt Clear-Pending Registers
227  uint64_t val = 0x0;
228  int first_intid = (addr - GICD_ICPENDR.start()) * 8;
229 
230  if (isNotSPI(first_intid)) {
231  return 0;
232  }
233 
234  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
235  i++, int_id++) {
236 
237  if (nsAccessToSecInt(int_id, is_secure_access))
238  {
239  if (irqNsacr[int_id] < 2) {
240  // Group 0 or Secure Group 1 interrupts are RAZ/WI
241  continue;
242  }
243  }
244 
245  val |= (irqPending[int_id] << i);
246  }
247 
248  return val;
249  } else if (GICD_ISACTIVER.contains(addr)) {
250  // Interrupt Set-Active Registers
251  int first_intid = (addr - GICD_ISACTIVER.start()) * 8;
252 
253  if (isNotSPI(first_intid)) {
254  return 0;
255  }
256 
257  uint64_t val = 0x0;
258 
259  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
260  i++, int_id++) {
261 
262  if (nsAccessToSecInt(int_id, is_secure_access))
263  {
264  // Group 0 or Secure Group 1 interrupts are RAZ/WI
265  if (irqNsacr[int_id] < 2) {
266  continue;
267  }
268  }
269 
270  val |= (irqActive[int_id] << i);
271  }
272 
273  return val;
274  } else if (GICD_ICACTIVER.contains(addr)) {
275  // Interrupt Clear-Active Registers
276  int first_intid = (addr - GICD_ICACTIVER.start()) * 8;
277 
278  if (isNotSPI(first_intid)) {
279  return 0;
280  }
281 
282  uint64_t val = 0x0;
283 
284  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
285  i++, int_id++) {
286 
287  if (nsAccessToSecInt(int_id, is_secure_access))
288  {
289  if (irqNsacr[int_id] < 2) {
290  continue;
291  }
292  }
293 
294  val |= (irqActive[int_id] << i);
295  }
296 
297  return val;
298  } else if (GICD_IPRIORITYR.contains(addr)) {
299  // Interrupt Priority Registers
300  uint64_t val = 0x0;
301  int first_intid = addr - GICD_IPRIORITYR.start();
302 
303  if (isNotSPI(first_intid)) {
304  return 0;
305  }
306 
307  for (int i = 0, int_id = first_intid; i < size && int_id < itLines;
308  i++, int_id++) {
309 
310  uint8_t prio = irqPriority[int_id];
311 
312  if (!DS && !is_secure_access) {
313  if (getIntGroup(int_id) != Gicv3::G1NS) {
314  // RAZ/WI for non-secure accesses for secure interrupts
315  continue;
316  } else {
317  // NS view
318  prio = (prio << 1) & 0xff;
319  }
320  }
321 
322  val |= prio << (i * 8);
323  }
324 
325  return val;
326  } else if (GICD_ITARGETSR.contains(addr)) {
327  // Interrupt Processor Targets Registers
328  // ARE always on, RAZ/WI
329  warn("Gicv3Distributor::read(): "
330  "GICD_ITARGETSR is RAZ/WI, legacy not supported!\n");
331  return 0;
332  } else if (GICD_ICFGR.contains(addr)) {
333  // Interrupt Configuration Registers
334  int first_intid = (addr - GICD_ICFGR.start()) * 4;
335 
336  if (isNotSPI(first_intid)) {
337  return 0;
338  }
339 
340  uint64_t val = 0x0;
341 
342  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
343  i = i + 2, int_id++) {
344 
345  if (nsAccessToSecInt(int_id, is_secure_access))
346  {
347  continue;
348  }
349 
350  if (irqConfig[int_id] == Gicv3::INT_EDGE_TRIGGERED) {
351  val |= (0x2 << i);
352  }
353  }
354 
355  return val;
356  } else if (GICD_IGRPMODR.contains(addr)) {
357  // Interrupt Group Modifier Registers
358  if (DS) {
359  // RAZ/WI if security disabled
360  return 0;
361  } else {
362  if (!is_secure_access) {
363  // RAZ/WI for non-secure accesses
364  return 0;
365  } else {
366  int first_intid = (addr - GICD_IGRPMODR.start()) * 8;
367 
368  if (isNotSPI(first_intid)) {
369  return 0;
370  }
371 
372  uint64_t val = 0x0;
373 
374  for (int i = 0, int_id = first_intid;
375  i < 8 * size && int_id < itLines; i++, int_id++) {
376  val |= irqGrpmod[int_id] << i;
377  }
378 
379  return val;
380  }
381  }
382  } else if (GICD_NSACR.contains(addr)) {
383  // Non-secure Access Control Registers
384  // 2 bits per interrupt
385  int first_intid = (addr - GICD_NSACR.start()) * 4;
386 
387  if (isNotSPI(first_intid)) {
388  return 0;
389  }
390 
391  if (DS || (!DS && !is_secure_access)) {
392  return 0;
393  }
394 
395  uint64_t val = 0x0;
396 
397  for (int i = 0, int_id = first_intid;
398  i < 8 * size && int_id < itLines; i = i + 2, int_id++) {
399  val |= irqNsacr[int_id] << i;
400  }
401 
402  return val;
403  } else if (GICD_CPENDSGIR.contains(addr)) { // SGI Clear-Pending Registers
404  // ARE always on, RAZ/WI
405  warn("Gicv3Distributor::read(): "
406  "GICD_CPENDSGIR is RAZ/WI, legacy not supported!\n");
407  return 0x0;
408  } else if (GICD_SPENDSGIR.contains(addr)) { // SGI Set-Pending Registers
409  // ARE always on, RAZ/WI
410  warn("Gicv3Distributor::read(): "
411  "GICD_SPENDSGIR is RAZ/WI, legacy not supported!\n");
412  return 0x0;
413  } else if (GICD_IROUTER.contains(addr)) { // Interrupt Routing Registers
414  // 64 bit registers. 2 or 1 access.
415  int int_id = (addr - GICD_IROUTER.start()) / 8;
416 
417  if (isNotSPI(int_id)) {
418  return 0;
419  }
420 
421  if (nsAccessToSecInt(int_id, is_secure_access))
422  {
423  if (irqNsacr[int_id] < 3) {
424  return 0;
425  }
426  }
427 
428  if (size == 4) {
429  if (addr & 7) { // high half of 64 bit register
430  return irqAffinityRouting[int_id] >> 32;
431  } else { // high low of 64 bit register
432  return irqAffinityRouting[int_id] & 0xFFFFFFFF;
433  }
434  } else {
435  return irqAffinityRouting[int_id];
436  }
437  }
438 
439  switch (addr) {
440  case GICD_CTLR: // Control Register
441  if (!DS) {
442  if (is_secure_access) {
443  // E1NWF [7] RAZ/WI
444  // DS [6] - Disable Security
445  // ARE_NS [5] RAO/WI
446  // ARE_S [4] RAO/WI
447  // EnableGrp1S [2]
448  // EnableGrp1NS [1]
449  // EnableGrp0 [0]
450  return (EnableGrp0 << 0) |
451  (EnableGrp1NS << 1) |
452  (EnableGrp1S << 2) |
453  (1 << 4) |
454  (1 << 5) |
455  (DS << 6);
456  } else {
457  // ARE_NS [4] RAO/WI;
458  // EnableGrp1A [1] is a read-write alias of the Secure
459  // GICD_CTLR.EnableGrp1NS
460  // EnableGrp1 [0] RES0
461  return (1 << 4) | (EnableGrp1NS << 1);
462  }
463  } else {
464  return (DS << 6) | (ARE << 4) |
465  (EnableGrp1NS << 1) | (EnableGrp0 << 0);
466  }
467 
468  case GICD_TYPER: // Interrupt Controller Type Register
469  return gicdTyper;
470 
471  case GICD_IIDR: // Implementer Identification Register
472  //return 0x43b; // ARM JEP106 code (r0p0 GIC-500)
473  return 0;
474 
475  case GICD_TYPER2: // Interrupt Controller Type Register 2
476  return 0; // RES0
477 
478  case GICD_STATUSR: // Error Reporting Status Register
479  // Optional register, RAZ/WI
480  return 0x0;
481 
482  case GICD_PIDR0: // Peripheral ID0 Register
483  return gicdPidr0;
484 
485  case GICD_PIDR1: // Peripheral ID1 Register
486  return gicdPidr1;
487 
488  case GICD_PIDR2: // Peripheral ID2 Register
489  return gicdPidr2;
490 
491  case GICD_PIDR3: // Peripheral ID3 Register
492  return gicdPidr3;
493 
494  case GICD_PIDR4: // Peripheral ID4 Register
495  return gicdPidr4;
496 
497  case GICD_PIDR5: // Peripheral ID5 Register
498  case GICD_PIDR6: // Peripheral ID6 Register
499  case GICD_PIDR7: // Peripheral ID7 Register
500  return 0; // RES0
501 
502  default:
503  panic("Gicv3Distributor::read(): invalid offset %#x\n", addr);
504  break;
505  }
506 }
507 
508 void
509 Gicv3Distributor::write(Addr addr, uint64_t data, size_t size,
510  bool is_secure_access)
511 {
512  if (GICD_IGROUPR.contains(addr)) { // Interrupt Group Registers
513  if (!DS && !is_secure_access) {
514  // RAZ/WI for non-secure accesses
515  return;
516  }
517 
518  int first_intid = (addr - GICD_IGROUPR.start()) * 8;
519 
520  if (isNotSPI(first_intid)) {
521  return;
522  }
523 
524  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
525  i++, int_id++) {
526  irqGroup[int_id] = data & (1 << i) ? 1 : 0;
527  DPRINTF(GIC, "Gicv3Distributor::write(): int_id %d group %d\n",
528  int_id, irqGroup[int_id]);
529  }
530 
531  return;
532  } else if (GICD_ISENABLER.contains(addr)) {
533  // Interrupt Set-Enable Registers
534  int first_intid = (addr - GICD_ISENABLER.start()) * 8;
535 
536  if (isNotSPI(first_intid)) {
537  return;
538  }
539 
540  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
541  i++, int_id++) {
542 
543  if (nsAccessToSecInt(int_id, is_secure_access))
544  {
545  continue;
546  }
547 
548  bool enable = data & (1 << i) ? 1 : 0;
549 
550  if (enable) {
551  if (!irqEnabled[int_id]) {
552  DPRINTF(GIC, "Gicv3Distributor::write(): "
553  "int_id %d enabled\n", int_id);
554  }
555 
556  irqEnabled[int_id] = true;
557  }
558  }
559 
560  return;
561  } else if (GICD_ICENABLER.contains(addr)) {
562  // Interrupt Clear-Enable Registers
563  int first_intid = (addr - GICD_ICENABLER.start()) * 8;
564 
565  if (isNotSPI(first_intid)) {
566  return;
567  }
568 
569  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
570  i++, int_id++) {
571 
572  if (nsAccessToSecInt(int_id, is_secure_access))
573  {
574  continue;
575  }
576 
577  bool disable = data & (1 << i) ? 1 : 0;
578 
579  if (disable) {
580  if (irqEnabled[int_id]) {
581  DPRINTF(GIC, "Gicv3Distributor::write(): "
582  "int_id %d disabled\n", int_id);
583  }
584 
585  irqEnabled[int_id] = false;
586  }
587  }
588 
589  return;
590  } else if (GICD_ISPENDR.contains(addr)) {
591  // Interrupt Set-Pending Registers
592  int first_intid = (addr - GICD_ISPENDR.start()) * 8;
593 
594  if (isNotSPI(first_intid)) {
595  return;
596  }
597 
598  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
599  i++, int_id++) {
600 
601  if (nsAccessToSecInt(int_id, is_secure_access))
602  {
603  if (irqNsacr[int_id] == 0) {
604  // Group 0 or Secure Group 1 interrupts are RAZ/WI
605  continue;
606  }
607  }
608 
609  bool pending = data & (1 << i) ? 1 : 0;
610 
611  if (pending) {
612  DPRINTF(GIC, "Gicv3Distributor::write() (GICD_ISPENDR): "
613  "int_id %d (SPI) pending bit set\n", int_id);
614  irqPending[int_id] = true;
615  irqPendingIspendr[int_id] = true;
616  }
617  }
618 
619  update();
620  return;
621  } else if (GICD_ICPENDR.contains(addr)) {
622  // Interrupt Clear-Pending Registers
623  int first_intid = (addr - GICD_ICPENDR.start()) * 8;
624 
625  if (isNotSPI(first_intid)) {
626  return;
627  }
628 
629  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
630  i++, int_id++) {
631 
632  if (nsAccessToSecInt(int_id, is_secure_access))
633  {
634  if (irqNsacr[int_id] < 2) {
635  // Group 0 or Secure Group 1 interrupts are RAZ/WI
636  continue;
637  }
638  }
639 
640  bool clear = data & (1 << i) ? 1 : 0;
641 
642  if (clear && treatAsEdgeTriggered(int_id)) {
643  irqPending[int_id] = false;
644  clearIrqCpuInterface(int_id);
645  }
646  }
647 
648  update();
649  return;
650  } else if (GICD_ISACTIVER.contains(addr)) {
651  // Interrupt Set-Active Registers
652  int first_intid = (addr - GICD_ISACTIVER.start()) * 8;
653 
654  if (isNotSPI(first_intid)) {
655  return;
656  }
657 
658  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
659  i++, int_id++) {
660 
661  if (nsAccessToSecInt(int_id, is_secure_access))
662  {
663  continue;
664  }
665 
666  bool active = data & (1 << i) ? 1 : 0;
667 
668  if (active) {
669  irqActive[int_id] = 1;
670  }
671  }
672 
673  return;
674  } else if (GICD_ICACTIVER.contains(addr)) {
675  // Interrupt Clear-Active Registers
676  int first_intid = (addr - GICD_ICACTIVER.start()) * 8;
677 
678  if (isNotSPI(first_intid)) {
679  return;
680  }
681 
682  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
683  i++, int_id++) {
684 
685  if (nsAccessToSecInt(int_id, is_secure_access))
686  {
687  continue;
688  }
689 
690  bool clear = data & (1 << i) ? 1 : 0;
691 
692  if (clear) {
693  if (irqActive[int_id]) {
694  DPRINTF(GIC, "Gicv3Distributor::write(): "
695  "int_id %d active cleared\n", int_id);
696  }
697 
698  irqActive[int_id] = false;
699  }
700  }
701 
702  return;
703  } else if (GICD_IPRIORITYR.contains(addr)) {
704  // Interrupt Priority Registers
705  int first_intid = addr - GICD_IPRIORITYR.start();
706 
707  if (isNotSPI(first_intid)) {
708  return;
709  }
710 
711  for (int i = 0, int_id = first_intid; i < size && int_id < itLines;
712  i++, int_id++) {
713  uint8_t prio = bits(data, (i + 1) * 8 - 1, (i * 8));
714 
715  if (!DS && !is_secure_access) {
716  if (getIntGroup(int_id) != Gicv3::G1NS) {
717  // RAZ/WI for non-secure accesses to secure interrupts
718  continue;
719  } else {
720  prio = 0x80 | (prio >> 1);
721  }
722  }
723 
724  irqPriority[int_id] = prio;
725  DPRINTF(GIC, "Gicv3Distributor::write(): int_id %d priority %d\n",
726  int_id, irqPriority[int_id]);
727  }
728 
729  return;
730  } else if (GICD_ITARGETSR.contains(addr)) {
731  // Interrupt Processor Targets Registers
732  // ARE always on, RAZ/WI
733  warn("Gicv3Distributor::write(): "
734  "GICD_ITARGETSR is RAZ/WI, legacy not supported!\n");
735  return;
736  } else if (GICD_ICFGR.contains(addr)) {
737  // Interrupt Configuration Registers
738  // for x = 0 to 15:
739  // GICD_ICFGR[2x] = RES0
740  // GICD_ICFGR[2x + 1] =
741  // 0 level-sensitive
742  // 1 edge-triggered
743  int first_intid = (addr - GICD_ICFGR.start()) * 4;
744 
745  if (isNotSPI(first_intid)) {
746  return;
747  }
748 
749  for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines;
750  i = i + 2, int_id++) {
751 
752  if (nsAccessToSecInt(int_id, is_secure_access)) {
753  continue;
754  }
755 
756  irqConfig[int_id] = data & (0x2 << i) ?
759  DPRINTF(GIC, "Gicv3Distributor::write(): int_id %d config %d\n",
760  int_id, irqConfig[int_id]);
761  }
762 
763  return;
764  } else if (GICD_IGRPMODR.contains(addr)) {
765  // Interrupt Group Modifier Registers
766  if (DS) {
767  return;
768  } else {
769  if (!is_secure_access) {
770  // RAZ/WI for non-secure accesses
771  return;
772  } else {
773  int first_intid = (addr - GICD_IGRPMODR.start()) * 8;
774 
775  if (isNotSPI(first_intid)) {
776  return;
777  }
778 
779  for (int i = 0, int_id = first_intid;
780  i < 8 * size && int_id < itLines; i++, int_id++) {
781  irqGrpmod[int_id] = bits(data, i);
782  }
783 
784  return ;
785  }
786  }
787 
788  } else if (GICD_NSACR.contains(addr)) {
789  // Non-secure Access Control Registers
790  // 2 bits per interrupt
791  int first_intid = (addr - GICD_NSACR.start()) * 4;
792 
793  if (isNotSPI(first_intid)) {
794  return;
795  }
796 
797  if (DS || (!DS && !is_secure_access)) {
798  return;
799  }
800 
801  for (int i = 0, int_id = first_intid;
802  i < 8 * size && int_id < itLines; i = i + 2, int_id++) {
803  irqNsacr[int_id] = (data >> (2 * int_id)) & 0x3;
804  }
805 
806  return;
807  } else if (GICD_IROUTER.contains(addr)) { // Interrupt Routing Registers
808  // 64 bit registers. 2 accesses.
809  int int_id = (addr - GICD_IROUTER.start()) / 8;
810 
811  if (isNotSPI(int_id)) {
812  return;
813  }
814 
815  if (nsAccessToSecInt(int_id, is_secure_access))
816  {
817  if (irqNsacr[int_id] < 3) {
818  // Group 0 or Secure Group 1 interrupts are RAZ/WI
819  return;
820  }
821  }
822 
823  if (size == 4) {
824  if (addr & 7) { // high half of 64 bit register
825  irqAffinityRouting[int_id] =
826  (irqAffinityRouting[int_id] & 0xffffffff) | (data << 32);
827  } else { // low half of 64 bit register
828  irqAffinityRouting[int_id] =
829  (irqAffinityRouting[int_id] & 0xffffffff00000000) |
830  (data & 0xffffffff);
831  }
832  } else {
833  irqAffinityRouting[int_id] = data;
834  }
835 
836  DPRINTF(GIC, "Gicv3Distributor::write(): "
837  "int_id %d GICD_IROUTER %#llx\n",
838  int_id, irqAffinityRouting[int_id]);
839  return;
840  }
841 
842  switch (addr) {
843  case GICD_CTLR: // Control Register
844  if (DS) {
845  /*
846  * E1NWF [7]
847  * 1 of N wakeup functionality not supported, RAZ/WI
848  * DS [6] - RAO/WI
849  * ARE [4]
850  * affinity routing always on, no GICv2 legacy, RAO/WI
851  * EnableGrp1 [1]
852  * EnableGrp0 [0]
853  */
854  if ((data & (1 << 4)) == 0) {
855  warn("Gicv3Distributor::write(): "
856  "setting ARE to 0 is not supported!\n");
857  }
858 
860  EnableGrp0 = data & GICD_CTLR_ENABLEGRP0;
861  DPRINTF(GIC, "Gicv3Distributor::write(): (DS 1)"
862  "EnableGrp1NS %d EnableGrp0 %d\n",
864  } else {
865  if (is_secure_access) {
866  /*
867  * E1NWF [7]
868  * 1 of N wakeup functionality not supported, RAZ/WI
869  * DS [6]
870  * ARE_NS [5]
871  * affinity routing always on, no GICv2 legacy, RAO/WI
872  * ARE_S [4]
873  * affinity routing always on, no GICv2 legacy, RAO/WI
874  * EnableGrp1S [2]
875  * EnableGrp1NS [1]
876  * EnableGrp0 [0]
877  */
878  if ((data & (1 << 5)) == 0) {
879  warn("Gicv3Distributor::write(): "
880  "setting ARE_NS to 0 is not supported!\n");
881  }
882 
883  if ((data & (1 << 4)) == 0) {
884  warn("Gicv3Distributor::write(): "
885  "setting ARE_S to 0 is not supported!\n");
886  }
887 
888  DS = data & GICD_CTLR_DS;
891  EnableGrp0 = data & GICD_CTLR_ENABLEGRP0;
892  DPRINTF(GIC, "Gicv3Distributor::write(): (DS 0 secure)"
893  "DS %d "
894  "EnableGrp1S %d EnableGrp1NS %d EnableGrp0 %d\n",
896 
897  if (data & GICD_CTLR_DS) {
898  EnableGrp1S = 0;
899  }
900  } else {
901  /*
902  * ARE_NS [4] RAO/WI;
903  * EnableGrp1A [1] is a read-write alias of the Secure
904  * GICD_CTLR.EnableGrp1NS
905  * EnableGrp1 [0] RES0
906  */
907  if ((data & (1 << 4)) == 0) {
908  warn("Gicv3Distributor::write(): "
909  "setting ARE_NS to 0 is not supported!\n");
910  }
911 
913  DPRINTF(GIC, "Gicv3Distributor::write(): (DS 0 non-secure)"
914  "EnableGrp1NS %d\n", EnableGrp1NS);
915  }
916  }
917 
918  update();
919 
920  break;
921 
922  case GICD_SGIR: // Error Reporting Status Register
923  // Only if affinity routing is disabled, RES0
924  break;
925 
926  case GICD_SETSPI_NSR: {
927  // Writes to this register have no effect if:
928  // * The value written specifies an invalid SPI.
929  // * The SPI is already pending.
930  // * The value written specifies a Secure SPI, the value is
931  // written by a Non-secure access, and the value of the
932  // corresponding GICD_NSACR<n> register is 0.
933  const uint32_t intid = bits(data, 9, 0);
934  if (isNotSPI(intid) || irqPending[intid] ||
935  (nsAccessToSecInt(intid, is_secure_access) &&
936  irqNsacr[intid] == 0)) {
937  return;
938  } else {
939  // Valid SPI, set interrupt pending
940  sendInt(intid);
941  }
942  break;
943  }
944 
945  case GICD_CLRSPI_NSR: {
946  // Writes to this register have no effect if:
947  // * The value written specifies an invalid SPI.
948  // * The SPI is not pending.
949  // * The value written specifies a Secure SPI, the value is
950  // written by a Non-secure access, and the value of the
951  // corresponding GICD_NSACR<n> register is less than 0b10.
952  const uint32_t intid = bits(data, 9, 0);
953  if (isNotSPI(intid) || !irqPending[intid] ||
954  (nsAccessToSecInt(intid, is_secure_access) &&
955  irqNsacr[intid] < 2)) {
956  return;
957  } else {
958  // Valid SPI, clear interrupt pending
959  deassertSPI(intid);
960  }
961  break;
962  }
963 
964  case GICD_SETSPI_SR: {
965  // Writes to this register have no effect if:
966  // * GICD_CTLR.DS = 1 (WI)
967  // * The value written specifies an invalid SPI.
968  // * The SPI is already pending.
969  // * The value is written by a Non-secure access.
970  const uint32_t intid = bits(data, 9, 0);
971  if (DS || isNotSPI(intid) || irqPending[intid] || !is_secure_access) {
972  return;
973  } else {
974  // Valid SPI, set interrupt pending
975  sendInt(intid);
976  }
977  break;
978  }
979 
980  case GICD_CLRSPI_SR: {
981  // Writes to this register have no effect if:
982  // * GICD_CTLR.DS = 1 (WI)
983  // * The value written specifies an invalid SPI.
984  // * The SPI is not pending.
985  // * The value is written by a Non-secure access.
986  const uint32_t intid = bits(data, 9, 0);
987  if (DS || isNotSPI(intid) || !irqPending[intid] || !is_secure_access) {
988  return;
989  } else {
990  // Valid SPI, clear interrupt pending
991  deassertSPI(intid);
992  }
993  break;
994  }
995 
996  default:
997  panic("Gicv3Distributor::write(): invalid offset %#x\n", addr);
998  break;
999  }
1000 }
1001 
1002 void
1004 {
1005  panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!");
1006  panic_if(int_id > itLines, "Invalid SPI!");
1007  irqPending[int_id] = true;
1008  irqPendingIspendr[int_id] = false;
1009  DPRINTF(GIC, "Gicv3Distributor::sendInt(): "
1010  "int_id %d (SPI) pending bit set\n", int_id);
1011  update();
1012 }
1013 
1014 void
1016 {
1017  // Edge-triggered interrupts remain pending until software
1018  // writes GICD_ICPENDR, GICD_CLRSPI_* or activates them via ICC_IAR
1019  if (isLevelSensitive(int_id)) {
1020  deassertSPI(int_id);
1021  }
1022 }
1023 
1024 void
1026 {
1027  panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!");
1028  panic_if(int_id > itLines, "Invalid SPI!");
1029  irqPending[int_id] = false;
1030  clearIrqCpuInterface(int_id);
1031 
1032  update();
1033 }
1034 
1036 Gicv3Distributor::route(uint32_t int_id)
1037 {
1038  IROUTER affinity_routing = irqAffinityRouting[int_id];
1039  Gicv3Redistributor * target_redistributor = nullptr;
1040 
1041  const Gicv3::GroupId int_group = getIntGroup(int_id);
1042 
1043  if (affinity_routing.IRM) {
1044  // Interrupts routed to any PE defined as a participating node
1045  for (int i = 0; i < gic->getSystem()->threads.size(); i++) {
1046  Gicv3Redistributor * redistributor_i =
1048 
1049  if (redistributor_i->
1050  canBeSelectedFor1toNInterrupt(int_group)) {
1051  target_redistributor = redistributor_i;
1052  break;
1053  }
1054  }
1055  } else {
1056  uint32_t affinity = (affinity_routing.Aff3 << 24) |
1057  (affinity_routing.Aff2 << 16) |
1058  (affinity_routing.Aff1 << 8) |
1059  (affinity_routing.Aff0 << 0);
1060  target_redistributor =
1061  gic->getRedistributorByAffinity(affinity);
1062  }
1063 
1064  if (!target_redistributor) {
1065  // Interrrupts targeting not present cpus must remain pending
1066  return nullptr;
1067  } else {
1068  return target_redistributor->getCPUInterface();
1069  }
1070 }
1071 
1072 void
1074 {
1075  auto cpu_interface = route(int_id);
1076  if (cpu_interface)
1077  cpu_interface->resetHppi(int_id);
1078 }
1079 
1080 void
1082 {
1083  // Find the highest priority pending SPI
1084  for (int int_id = Gicv3::SGI_MAX + Gicv3::PPI_MAX; int_id < itLines;
1085  int_id++) {
1086  Gicv3::GroupId int_group = getIntGroup(int_id);
1087  bool group_enabled = groupEnabled(int_group);
1088 
1089  if (irqPending[int_id] && irqEnabled[int_id] &&
1090  !irqActive[int_id] && group_enabled) {
1091 
1092  // Find the cpu interface where to route the interrupt
1093  Gicv3CPUInterface *target_cpu_interface = route(int_id);
1094 
1095  // Invalid routing
1096  if (!target_cpu_interface) continue;
1097 
1098  if ((irqPriority[int_id] < target_cpu_interface->hppi.prio) ||
1099  (irqPriority[int_id] == target_cpu_interface->hppi.prio &&
1100  int_id < target_cpu_interface->hppi.intid)) {
1101 
1102  target_cpu_interface->hppi.intid = int_id;
1103  target_cpu_interface->hppi.prio = irqPriority[int_id];
1104  target_cpu_interface->hppi.group = int_group;
1105  }
1106  }
1107  }
1108 
1109  // Update all redistributors
1110  for (int i = 0; i < gic->getSystem()->threads.size(); i++) {
1112  }
1113 }
1114 
1116 Gicv3Distributor::intStatus(uint32_t int_id) const
1117 {
1118  panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!");
1119  panic_if(int_id > itLines, "Invalid SPI!");
1120 
1121  if (irqPending[int_id]) {
1122  if (irqActive[int_id]) {
1124  }
1125 
1126  return Gicv3::INT_PENDING;
1127  } else if (irqActive[int_id]) {
1128  return Gicv3::INT_ACTIVE;
1129  } else {
1130  return Gicv3::INT_INACTIVE;
1131  }
1132 }
1133 
1136 {
1137  panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!");
1138  panic_if(int_id > itLines, "Invalid SPI!");
1139 
1140  if (DS) {
1141  if (irqGroup[int_id] == 1) {
1142  return Gicv3::G1NS;
1143  } else {
1144  return Gicv3::G0S;
1145  }
1146  } else {
1147  if (irqGrpmod[int_id] == 0 && irqGroup[int_id] == 0) {
1148  return Gicv3::G0S;
1149  } else if (irqGrpmod[int_id] == 0 && irqGroup[int_id] == 1) {
1150  return Gicv3::G1NS;
1151  } else if (irqGrpmod[int_id] == 1 && irqGroup[int_id] == 0) {
1152  return Gicv3::G1S;
1153  } else if (irqGrpmod[int_id] == 1 && irqGroup[int_id] == 1) {
1154  return Gicv3::G1NS;
1155  }
1156  }
1157 
1158  M5_UNREACHABLE;
1159 }
1160 
1161 void
1163 {
1164  if (treatAsEdgeTriggered(int_id)) {
1165  irqPending[int_id] = false;
1166  }
1167  irqActive[int_id] = true;
1168 }
1169 
1170 void
1172 {
1173  irqActive[int_id] = false;
1174 }
1175 
1176 void
1178 {
1194 }
1195 
1196 void
1198 {
1214 }
Gicv3::INT_EDGE_TRIGGERED
@ INT_EDGE_TRIGGERED
Definition: gic_v3.hh:97
gic_v3_cpu_interface.hh
Gicv3Distributor::GICD_SETSPI_SR
@ GICD_SETSPI_SR
Definition: gic_v3_distributor.hh:77
Gicv3Distributor::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: gic_v3_distributor.cc:1197
Gicv3Distributor::GICD_SETSPI_NSR
@ GICD_SETSPI_NSR
Definition: gic_v3_distributor.hh:73
Gicv3::INTID_SECURE
static const int INTID_SECURE
Definition: gic_v3.hh:71
Gicv3Distributor::isNotSPI
bool isNotSPI(uint32_t int_id) const
Definition: gic_v3_distributor.hh:219
Gicv3Distributor::GICD_ICFGR
static const AddrRange GICD_ICFGR
Definition: gic_v3_distributor.hh:119
Gicv3Distributor::gicdPidr1
uint32_t gicdPidr1
Definition: gic_v3_distributor.hh:166
Gicv3Redistributor::irqPendingIspendr
std::vector< bool > irqPendingIspendr
Definition: gic_v3_redistributor.hh:156
warn
#define warn(...)
Definition: logging.hh:239
Gicv3::G1NS
@ G1NS
Definition: gic_v3.hh:92
gic_v3_redistributor.hh
data
const char data[]
Definition: circlebuf.test.cc:47
Gicv3Distributor::GICD_IPRIORITYR
static const AddrRange GICD_IPRIORITYR
Definition: gic_v3_distributor.hh:115
Gicv3Distributor::GICD_PIDR1
@ GICD_PIDR1
Definition: gic_v3_distributor.hh:85
Gicv3Distributor::irqEnabled
std::vector< bool > irqEnabled
Definition: gic_v3_distributor.hh:154
Gicv3::PPI_MAX
static const int PPI_MAX
Definition: gic_v3.hh:78
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
UNSERIALIZE_CONTAINER
#define UNSERIALIZE_CONTAINER(member)
Definition: serialize.hh:650
Gicv3CPUInterface::hppi
hppi_t hppi
Definition: gic_v3_cpu_interface.hh:160
Gicv3Distributor::GICD_CLRSPI_NSR
@ GICD_CLRSPI_NSR
Definition: gic_v3_distributor.hh:75
Gicv3Distributor::ARE
bool ARE
Definition: gic_v3_distributor.hh:148
Gicv3Distributor::GICD_PIDR6
@ GICD_PIDR6
Definition: gic_v3_distributor.hh:95
Gicv3Distributor::GICD_CTLR_DS
static const uint32_t GICD_CTLR_DS
Definition: gic_v3_distributor.hh:146
Gicv3Distributor::Gicv3Distributor
Gicv3Distributor(Gicv3 *gic, uint32_t it_lines)
Definition: gic_v3_distributor.cc:67
Gicv3::SGI_MAX
static const int SGI_MAX
Definition: gic_v3.hh:76
Gicv3Distributor::GICD_PIDR2
@ GICD_PIDR2
Definition: gic_v3_distributor.hh:87
Gicv3Distributor::GICD_ICACTIVER
static const AddrRange GICD_ICACTIVER
Definition: gic_v3_distributor.hh:113
Gicv3Distributor::itLines
const uint32_t itLines
Definition: gic_v3_distributor.hh:59
Gicv3Distributor::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: gic_v3_distributor.cc:1177
Gicv3Redistributor::irqPending
std::vector< bool > irqPending
Definition: gic_v3_redistributor.hh:155
Gicv3Distributor::route
Gicv3CPUInterface * route(uint32_t int_id)
Definition: gic_v3_distributor.cc:1036
AddrRange::contains
bool contains(const Addr &a) const
Determine if the range contains an address.
Definition: addr_range.hh:435
Gicv3Distributor::GICD_NSACR
static const AddrRange GICD_NSACR
Definition: gic_v3_distributor.hh:123
Gicv3Distributor::intStatus
Gicv3::IntStatus intStatus(uint32_t int_id) const
Definition: gic_v3_distributor.cc:1116
Gicv3Distributor::getIntGroup
Gicv3::GroupId getIntGroup(int int_id) const
Definition: gic_v3_distributor.cc:1135
Gicv3Distributor::clearInt
void clearInt(uint32_t int_id)
Definition: gic_v3_distributor.cc:1015
Gicv3Distributor::irqActive
std::vector< bool > irqActive
Definition: gic_v3_distributor.hh:157
Gicv3::getRedistributorByAffinity
Gicv3Redistributor * getRedistributorByAffinity(uint32_t affinity) const
Definition: gic_v3.cc:237
Gicv3Redistributor::irqActive
std::vector< bool > irqActive
Definition: gic_v3_redistributor.hh:157
Gicv3Distributor::GICD_CTLR_ENABLEGRP1S
static const uint32_t GICD_CTLR_ENABLEGRP1S
Definition: gic_v3_distributor.hh:145
Gicv3Distributor::GICD_IGRPMODR
static const AddrRange GICD_IGRPMODR
Definition: gic_v3_distributor.hh:121
Trace::disable
void disable()
Definition: trace.cc:96
Gicv3Distributor::GICD_ICPENDR
static const AddrRange GICD_ICPENDR
Definition: gic_v3_distributor.hh:109
Gicv3Redistributor::irqGroup
std::vector< uint8_t > irqGroup
Definition: gic_v3_redistributor.hh:153
Gicv3Redistributor::irqPriority
std::vector< uint8_t > irqPriority
Definition: gic_v3_redistributor.hh:158
Gicv3Distributor::GICD_PIDR5
@ GICD_PIDR5
Definition: gic_v3_distributor.hh:93
Gicv3Distributor::irqConfig
std::vector< Gicv3::IntTriggerType > irqConfig
Definition: gic_v3_distributor.hh:159
Gicv3Distributor::GICD_CTLR_ENABLEGRP1NS
static const uint32_t GICD_CTLR_ENABLEGRP1NS
Definition: gic_v3_distributor.hh:143
Gicv3::G0S
@ G0S
Definition: gic_v3.hh:90
Gicv3::INT_ACTIVE_PENDING
@ INT_ACTIVE_PENDING
Definition: gic_v3.hh:85
Gicv3Distributor::GICD_SPENDSGIR
static const AddrRange GICD_SPENDSGIR
Definition: gic_v3_distributor.hh:127
Gicv3Redistributor::irqConfig
std::vector< Gicv3::IntTriggerType > irqConfig
Definition: gic_v3_redistributor.hh:159
Gicv3CPUInterface::hppi_t::group
Gicv3::GroupId group
Definition: gic_v3_cpu_interface.hh:157
Gicv3::INT_LEVEL_SENSITIVE
@ INT_LEVEL_SENSITIVE
Definition: gic_v3.hh:96
divCeil
T divCeil(const T &a, const U &b)
Definition: intmath.hh:114
cp
Definition: cprintf.cc:37
Gicv3Distributor::gic
Gicv3 * gic
Definition: gic_v3_distributor.hh:58
Gicv3Distributor::GICD_PIDR0
@ GICD_PIDR0
Definition: gic_v3_distributor.hh:83
Gicv3Distributor::GICD_CTLR_ENABLEGRP1A
static const uint32_t GICD_CTLR_ENABLEGRP1A
Definition: gic_v3_distributor.hh:144
Gicv3Distributor::irqGroup
std::vector< uint8_t > irqGroup
Definition: gic_v3_distributor.hh:153
ArmSystem::haveSecurity
bool haveSecurity() const
Returns true if this system implements the Security Extensions.
Definition: system.hh:159
AddrRange
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Definition: addr_range.hh:68
Gicv3Distributor::GICD_TYPER2
@ GICD_TYPER2
Definition: gic_v3_distributor.hh:69
Gicv3Distributor::irqPendingIspendr
std::vector< bool > irqPendingIspendr
Definition: gic_v3_distributor.hh:156
Gicv3Redistributor::irqEnabled
std::vector< bool > irqEnabled
Definition: gic_v3_redistributor.hh:154
Gicv3Distributor::GICD_PIDR7
@ GICD_PIDR7
Definition: gic_v3_distributor.hh:97
Gicv3Distributor::update
void update()
Definition: gic_v3_distributor.cc:1081
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
Gicv3Distributor::EnableGrp1S
bool EnableGrp1S
Definition: gic_v3_distributor.hh:150
Gicv3::INT_ACTIVE
@ INT_ACTIVE
Definition: gic_v3.hh:84
Gicv3Distributor::groupEnabled
bool groupEnabled(Gicv3::GroupId group) const
Definition: gic_v3_distributor.hh:184
Gicv3Distributor::GICD_IROUTER
static const AddrRange GICD_IROUTER
Definition: gic_v3_distributor.hh:129
Gicv3Distributor::deactivateIRQ
void deactivateIRQ(uint32_t int_id)
Definition: gic_v3_distributor.cc:1171
Gicv3::IntStatus
IntStatus
Definition: gic_v3.hh:81
Gicv3Distributor::GICD_CLRSPI_SR
@ GICD_CLRSPI_SR
Definition: gic_v3_distributor.hh:79
Gicv3Distributor::gicdPidr0
uint32_t gicdPidr0
Definition: gic_v3_distributor.hh:165
Gicv3Distributor::read
uint64_t read(Addr addr, size_t size, bool is_secure_access)
Definition: gic_v3_distributor.cc:137
Gicv3Distributor::GICD_IIDR
@ GICD_IIDR
Definition: gic_v3_distributor.hh:67
Gicv3
Definition: gic_v3.hh:53
Gicv3Redistributor::getCPUInterface
Gicv3CPUInterface * getCPUInterface() const
Definition: gic_v3_redistributor.hh:202
Gicv3::GroupId
GroupId
Definition: gic_v3.hh:89
X86ISA::val
Bitfield< 63 > val
Definition: misc.hh:769
Gicv3Distributor::DS
bool DS
Definition: gic_v3_distributor.hh:149
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
Gicv3::G1S
@ G1S
Definition: gic_v3.hh:91
Gicv3::getRedistributor
Gicv3Redistributor * getRedistributor(ContextID context_id) const
Definition: gic_v3.hh:145
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
Gicv3Distributor::EnableGrp0
bool EnableGrp0
Definition: gic_v3_distributor.hh:152
Gicv3Distributor::irqGrpmod
std::vector< uint8_t > irqGrpmod
Definition: gic_v3_distributor.hh:160
Gicv3Distributor::gicdTyper
uint32_t gicdTyper
Definition: gic_v3_distributor.hh:164
Gicv3CPUInterface
Definition: gic_v3_cpu_interface.hh:50
Gicv3Distributor::GICD_PIDR4
@ GICD_PIDR4
Definition: gic_v3_distributor.hh:91
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
Gicv3Distributor::treatAsEdgeTriggered
bool treatAsEdgeTriggered(uint32_t int_id) const
This helper is used to check if an interrupt should be treated as edge triggered in the following sce...
Definition: gic_v3_distributor.hh:244
panic_if
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:197
Gicv3Distributor::GICD_ICENABLER
static const AddrRange GICD_ICENABLER
Definition: gic_v3_distributor.hh:105
Gicv3Distributor::init
void init()
Definition: gic_v3_distributor.cc:132
AddrRange::start
Addr start() const
Get the start address of the range.
Definition: addr_range.hh:314
Gicv3Distributor::GICD_SGIR
@ GICD_SGIR
Definition: gic_v3_distributor.hh:81
Gicv3Distributor::GICD_ITARGETSR
static const AddrRange GICD_ITARGETSR
Definition: gic_v3_distributor.hh:117
X86ISA::enable
Bitfield< 11 > enable
Definition: misc.hh:1051
SERIALIZE_CONTAINER
#define SERIALIZE_CONTAINER(member)
Definition: serialize.hh:642
ArmISA::gic
Bitfield< 27, 24 > gic
Definition: miscregs_types.hh:171
Gicv3Distributor::irqPriority
std::vector< uint8_t > irqPriority
Definition: gic_v3_distributor.hh:158
Gicv3Distributor::deassertSPI
void deassertSPI(uint32_t int_id)
Definition: gic_v3_distributor.cc:1025
Gicv3Distributor::nsAccessToSecInt
bool nsAccessToSecInt(uint32_t int_id, bool is_secure_access) const
Definition: gic_v3_distributor.hh:249
Gicv3Redistributor::irqGrpmod
std::vector< uint8_t > irqGrpmod
Definition: gic_v3_redistributor.hh:160
Gicv3CPUInterface::hppi_t::intid
uint32_t intid
Definition: gic_v3_cpu_interface.hh:155
Gicv3Distributor::irqNsacr
std::vector< uint8_t > irqNsacr
Definition: gic_v3_distributor.hh:161
Gicv3::INT_PENDING
@ INT_PENDING
Definition: gic_v3.hh:83
Gicv3Redistributor::update
void update()
Definition: gic_v3_redistributor.cc:801
Gicv3Distributor::irqAffinityRouting
std::vector< IROUTER > irqAffinityRouting
Definition: gic_v3_distributor.hh:162
Gicv3Distributor::GICD_ISENABLER
static const AddrRange GICD_ISENABLER
Definition: gic_v3_distributor.hh:103
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
bits
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition: bitfield.hh:73
Gicv3Distributor::isLevelSensitive
bool isLevelSensitive(uint32_t int_id) const
Definition: gic_v3_distributor.hh:228
Gicv3Distributor::GICD_STATUSR
@ GICD_STATUSR
Definition: gic_v3_distributor.hh:71
Gicv3Distributor::GICD_ISPENDR
static const AddrRange GICD_ISPENDR
Definition: gic_v3_distributor.hh:107
BaseGic::getSystem
ArmSystem * getSystem() const
Definition: base_gic.hh:106
Gicv3Distributor::GICD_CPENDSGIR
static const AddrRange GICD_CPENDSGIR
Definition: gic_v3_distributor.hh:125
Gicv3Distributor::irqPending
std::vector< bool > irqPending
Definition: gic_v3_distributor.hh:155
Gicv3Distributor::GICD_PIDR3
@ GICD_PIDR3
Definition: gic_v3_distributor.hh:89
Gicv3Distributor::gicdPidr4
uint32_t gicdPidr4
Definition: gic_v3_distributor.hh:169
Gicv3Distributor::GICD_CTLR
@ GICD_CTLR
Definition: gic_v3_distributor.hh:63
Gicv3Distributor::clearIrqCpuInterface
void clearIrqCpuInterface(uint32_t int_id)
Definition: gic_v3_distributor.cc:1073
intmath.hh
Gicv3Redistributor::canBeSelectedFor1toNInterrupt
bool canBeSelectedFor1toNInterrupt(Gicv3::GroupId group) const
Definition: gic_v3_redistributor.cc:1027
CheckpointIn
Definition: serialize.hh:68
Gicv3Distributor::GICD_TYPER
@ GICD_TYPER
Definition: gic_v3_distributor.hh:65
Gicv3Distributor::write
void write(Addr addr, uint64_t data, size_t size, bool is_secure_access)
Definition: gic_v3_distributor.cc:509
Gicv3CPUInterface::hppi_t::prio
uint8_t prio
Definition: gic_v3_cpu_interface.hh:156
Gicv3Distributor::gicdPidr2
uint32_t gicdPidr2
Definition: gic_v3_distributor.hh:167
gic_v3_distributor.hh
Gicv3Distributor::sendInt
void sendInt(uint32_t int_id)
Definition: gic_v3_distributor.cc:1003
Gicv3Distributor::GICD_IGROUPR
static const AddrRange GICD_IGROUPR
Definition: gic_v3_distributor.hh:101
Gicv3Distributor::IDBITS
static const uint32_t IDBITS
Definition: gic_v3_distributor.hh:174
gic_v3.hh
Gicv3Redistributor
Definition: gic_v3_redistributor.hh:52
Gicv3Distributor::EnableGrp1NS
bool EnableGrp1NS
Definition: gic_v3_distributor.hh:151
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
Gicv3Redistributor::irqNsacr
std::vector< uint8_t > irqNsacr
Definition: gic_v3_redistributor.hh:161
Gicv3Distributor::GICD_ISACTIVER
static const AddrRange GICD_ISACTIVER
Definition: gic_v3_distributor.hh:111
Gicv3Distributor::gicdPidr3
uint32_t gicdPidr3
Definition: gic_v3_distributor.hh:168
Gicv3::INT_INACTIVE
@ INT_INACTIVE
Definition: gic_v3.hh:82
Gicv3Distributor::activateIRQ
void activateIRQ(uint32_t int_id)
Definition: gic_v3_distributor.cc:1162

Generated on Tue Jun 22 2021 15:28:27 for gem5 by doxygen 1.8.17