| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  ioapic.c IOAPIC emulation logic | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright (c) 2004-2005 Fabrice Bellard | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Split the ioapic logic from apic.c | 
					
						
							|  |  |  |  *  Xiantao Zhang <xiantao.zhang@intel.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							| 
									
										
										
										
											2009-07-16 20:47:01 +00:00
										 |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-26 18:17:03 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-17 10:26:34 +02:00
										 |  |  | #include "qapi/error.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-22 16:18:21 +03:00
										 |  |  | #include "monitor/monitor.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-05 17:06:20 +01:00
										 |  |  | #include "hw/i386/pc.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-04 11:04:13 +01:00
										 |  |  | #include "hw/i386/apic.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-05 17:06:20 +01:00
										 |  |  | #include "hw/i386/ioapic.h"
 | 
					
						
							|  |  |  | #include "hw/i386/ioapic_internal.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-03 22:50:32 +03:00
										 |  |  | #include "hw/pci/msi.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-12 07:23:51 +02:00
										 |  |  | #include "hw/qdev-properties.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  | #include "sysemu/kvm.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-12 07:23:57 +02:00
										 |  |  | #include "sysemu/sysemu.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:23 +08:00
										 |  |  | #include "hw/i386/apic-msidef.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:27 +08:00
										 |  |  | #include "hw/i386/x86-iommu.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  | #include "trace.h"
 | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  | #define APIC_DELIVERY_MODE_SHIFT 8
 | 
					
						
							|  |  |  | #define APIC_POLARITY_SHIFT 14
 | 
					
						
							|  |  |  | #define APIC_TRIG_MODE_SHIFT 15
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  | static IOAPICCommonState *ioapics[MAX_IOAPICS]; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:05 +08:00
										 |  |  | /* global variable from ioapic_common.c */ | 
					
						
							|  |  |  | extern int ioapic_no; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  | struct ioapic_entry_info { | 
					
						
							|  |  |  |     /* fields parsed from IOAPIC entries */ | 
					
						
							|  |  |  |     uint8_t masked; | 
					
						
							|  |  |  |     uint8_t trig_mode; | 
					
						
							|  |  |  |     uint16_t dest_idx; | 
					
						
							|  |  |  |     uint8_t dest_mode; | 
					
						
							|  |  |  |     uint8_t delivery_mode; | 
					
						
							|  |  |  |     uint8_t vector; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* MSI message generated from above parsed fields */ | 
					
						
							|  |  |  |     uint32_t addr; | 
					
						
							|  |  |  |     uint32_t data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ioapic_entry_parse(uint64_t entry, struct ioapic_entry_info *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     memset(info, 0, sizeof(*info)); | 
					
						
							|  |  |  |     info->masked = (entry >> IOAPIC_LVT_MASKED_SHIFT) & 1; | 
					
						
							|  |  |  |     info->trig_mode = (entry >> IOAPIC_LVT_TRIGGER_MODE_SHIFT) & 1; | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |      * By default, this would be dest_id[8] + reserved[8]. When IR | 
					
						
							|  |  |  |      * is enabled, this would be interrupt_index[15] + | 
					
						
							|  |  |  |      * interrupt_format[1]. This field never means anything, but | 
					
						
							|  |  |  |      * only used to generate corresponding MSI. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     info->dest_idx = (entry >> IOAPIC_LVT_DEST_IDX_SHIFT) & 0xffff; | 
					
						
							|  |  |  |     info->dest_mode = (entry >> IOAPIC_LVT_DEST_MODE_SHIFT) & 1; | 
					
						
							|  |  |  |     info->delivery_mode = (entry >> IOAPIC_LVT_DELIV_MODE_SHIFT) \ | 
					
						
							|  |  |  |         & IOAPIC_DM_MASK; | 
					
						
							|  |  |  |     if (info->delivery_mode == IOAPIC_DM_EXTINT) { | 
					
						
							|  |  |  |         info->vector = pic_read_irq(isa_pic); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         info->vector = entry & IOAPIC_VECTOR_MASK; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     info->addr = APIC_DEFAULT_ADDRESS | \ | 
					
						
							|  |  |  |         (info->dest_idx << MSI_ADDR_DEST_IDX_SHIFT) | \ | 
					
						
							|  |  |  |         (info->dest_mode << MSI_ADDR_DEST_MODE_SHIFT); | 
					
						
							|  |  |  |     info->data = (info->vector << MSI_DATA_VECTOR_SHIFT) | \ | 
					
						
							|  |  |  |         (info->trig_mode << MSI_DATA_TRIGGER_SHIFT) | \ | 
					
						
							|  |  |  |         (info->delivery_mode << MSI_DATA_DELIVERY_MODE_SHIFT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  | static void ioapic_service(IOAPICCommonState *s) | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:23 +08:00
										 |  |  |     AddressSpace *ioapic_as = PC_MACHINE(qdev_get_machine())->ioapic_as; | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  |     struct ioapic_entry_info info; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     uint8_t i; | 
					
						
							|  |  |  |     uint32_t mask; | 
					
						
							|  |  |  |     uint64_t entry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < IOAPIC_NUM_PINS; i++) { | 
					
						
							|  |  |  |         mask = 1 << i; | 
					
						
							|  |  |  |         if (s->irr & mask) { | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  |             int coalesce = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |             entry = s->ioredtbl[i]; | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  |             ioapic_entry_parse(entry, &info); | 
					
						
							|  |  |  |             if (!info.masked) { | 
					
						
							|  |  |  |                 if (info.trig_mode == IOAPIC_TRIGGER_EDGE) { | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |                     s->irr &= ~mask; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  |                     coalesce = s->ioredtbl[i] & IOAPIC_LVT_REMOTE_IRR; | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  |                     trace_ioapic_set_remote_irr(i); | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |                     s->ioredtbl[i] |= IOAPIC_LVT_REMOTE_IRR; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-31 22:18:05 +08:00
										 |  |  |                 if (coalesce) { | 
					
						
							|  |  |  |                     /* We are level triggered interrupts, and the
 | 
					
						
							|  |  |  |                      * guest should be still working on previous one, | 
					
						
							|  |  |  |                      * so skip it. */ | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  | #ifdef CONFIG_KVM
 | 
					
						
							|  |  |  |                 if (kvm_irqchip_is_split()) { | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  |                     if (info.trig_mode == IOAPIC_TRIGGER_EDGE) { | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  |                         kvm_set_irq(kvm_state, i, 1); | 
					
						
							|  |  |  |                         kvm_set_irq(kvm_state, i, 0); | 
					
						
							|  |  |  |                     } else { | 
					
						
							| 
									
										
										
										
											2016-07-31 22:18:05 +08:00
										 |  |  |                         kvm_set_irq(kvm_state, i, 1); | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  |                     } | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-07-31 22:18:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:23 +08:00
										 |  |  |                 /* No matter whether IR is enabled, we translate
 | 
					
						
							|  |  |  |                  * the IOAPIC message into a MSI one, and its | 
					
						
							|  |  |  |                  * address space will decide whether we need a | 
					
						
							|  |  |  |                  * translation. */ | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  |                 stl_le_phys(ioapic_as, info.addr, info.data); | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  | #define SUCCESSIVE_IRQ_MAX_COUNT 10000
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void delayed_ioapic_service_cb(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     IOAPICCommonState *s = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ioapic_service(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-17 16:32:47 +00:00
										 |  |  | static void ioapic_set_irq(void *opaque, int vector, int level) | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  |     IOAPICCommonState *s = opaque; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* ISA IRQs map to GSI 1-1 except for IRQ0 which maps
 | 
					
						
							|  |  |  |      * to GSI 2.  GSI maps to ioapic 1-1.  This is not | 
					
						
							|  |  |  |      * the cleanest way of doing it but it should work. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 18:03:10 +00:00
										 |  |  |     trace_ioapic_set_irq(vector, level); | 
					
						
							| 
									
										
										
										
											2017-12-29 15:31:03 +08:00
										 |  |  |     ioapic_stat_update_irq(s, vector, level); | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |     if (vector == 0) { | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         vector = 2; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-04 14:03:10 +02:00
										 |  |  |     if (vector < IOAPIC_NUM_PINS) { | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         uint32_t mask = 1 << vector; | 
					
						
							|  |  |  |         uint64_t entry = s->ioredtbl[vector]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         if (((entry >> IOAPIC_LVT_TRIGGER_MODE_SHIFT) & 1) == | 
					
						
							|  |  |  |             IOAPIC_TRIGGER_LEVEL) { | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |             /* level triggered */ | 
					
						
							|  |  |  |             if (level) { | 
					
						
							|  |  |  |                 s->irr |= mask; | 
					
						
							| 
									
										
										
										
											2015-07-30 10:19:24 +02:00
										 |  |  |                 if (!(entry & IOAPIC_LVT_REMOTE_IRR)) { | 
					
						
							|  |  |  |                     ioapic_service(s); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 s->irr &= ~mask; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-04-09 13:18:59 +02:00
										 |  |  |             /* According to the 82093AA manual, we must ignore edge requests
 | 
					
						
							|  |  |  |              * if the input pin is masked. */ | 
					
						
							|  |  |  |             if (level && !(entry & IOAPIC_LVT_MASKED)) { | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |                 s->irr |= mask; | 
					
						
							|  |  |  |                 ioapic_service(s); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  | static void ioapic_update_kvm_routes(IOAPICCommonState *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_KVM
 | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (kvm_irqchip_is_split()) { | 
					
						
							|  |  |  |         for (i = 0; i < IOAPIC_NUM_PINS; i++) { | 
					
						
							|  |  |  |             MSIMessage msg; | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:24 +08:00
										 |  |  |             struct ioapic_entry_info info; | 
					
						
							|  |  |  |             ioapic_entry_parse(s->ioredtbl[i], &info); | 
					
						
							| 
									
										
										
										
											2019-06-02 13:42:13 +02:00
										 |  |  |             if (!info.masked) { | 
					
						
							|  |  |  |                 msg.address = info.addr; | 
					
						
							|  |  |  |                 msg.data = info.data; | 
					
						
							|  |  |  |                 kvm_irqchip_update_msi_route(kvm_state, i, msg, NULL); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         kvm_irqchip_commit_routes(kvm_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:27 +08:00
										 |  |  | #ifdef CONFIG_KVM
 | 
					
						
							|  |  |  | static void ioapic_iec_notifier(void *private, bool global, | 
					
						
							|  |  |  |                                 uint32_t index, uint32_t mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     IOAPICCommonState *s = (IOAPICCommonState *)private; | 
					
						
							|  |  |  |     /* For simplicity, we just update all the routes */ | 
					
						
							|  |  |  |     ioapic_update_kvm_routes(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  | void ioapic_eoi_broadcast(int vector) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  |     IOAPICCommonState *s; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |     uint64_t entry; | 
					
						
							|  |  |  |     int i, n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  |     trace_ioapic_eoi_broadcast(vector); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |     for (i = 0; i < MAX_IOAPICS; i++) { | 
					
						
							|  |  |  |         s = ioapics[i]; | 
					
						
							|  |  |  |         if (!s) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         for (n = 0; n < IOAPIC_NUM_PINS; n++) { | 
					
						
							|  |  |  |             entry = s->ioredtbl[n]; | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if ((entry & IOAPIC_VECTOR_MASK) != vector || | 
					
						
							|  |  |  |                 ((entry >> IOAPIC_LVT_TRIGGER_MODE_SHIFT) & 1) != IOAPIC_TRIGGER_LEVEL) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!(entry & IOAPIC_LVT_REMOTE_IRR)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             trace_ioapic_clear_remote_irr(n, vector); | 
					
						
							|  |  |  |             s->ioredtbl[n] = entry & ~IOAPIC_LVT_REMOTE_IRR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!(entry & IOAPIC_LVT_MASKED) && (s->irr & (1 << n))) { | 
					
						
							| 
									
										
										
										
											2019-06-21 17:21:19 -07:00
										 |  |  |                 ++s->irq_eoi[n]; | 
					
						
							|  |  |  |                 if (s->irq_eoi[n] >= SUCCESSIVE_IRQ_MAX_COUNT) { | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |                     /*
 | 
					
						
							|  |  |  |                      * Real hardware does not deliver the interrupt immediately | 
					
						
							|  |  |  |                      * during eoi broadcast, and this lets a buggy guest make | 
					
						
							|  |  |  |                      * slow progress even if it does not correctly handle a | 
					
						
							|  |  |  |                      * level-triggered interrupt. Emulate this behavior if we | 
					
						
							|  |  |  |                      * detect an interrupt storm. | 
					
						
							|  |  |  |                      */ | 
					
						
							| 
									
										
										
										
											2019-06-21 17:21:19 -07:00
										 |  |  |                     s->irq_eoi[n] = 0; | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |                     timer_mod_anticipate(s->delayed_ioapic_service_timer, | 
					
						
							|  |  |  |                                          qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) + | 
					
						
							|  |  |  |                                          NANOSECONDS_PER_SECOND / 100); | 
					
						
							| 
									
										
										
										
											2019-06-21 17:21:19 -07:00
										 |  |  |                     trace_ioapic_eoi_delayed_reassert(n); | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |                     ioapic_service(s); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2019-06-21 17:21:19 -07:00
										 |  |  |                 s->irq_eoi[n] = 0; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-17 13:11:27 +02:00
										 |  |  | static uint64_t | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  | ioapic_mem_read(void *opaque, hwaddr addr, unsigned int size) | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  |     IOAPICCommonState *s = opaque; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     int index; | 
					
						
							|  |  |  |     uint32_t val = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  |     addr &= 0xff; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (addr) { | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |     case IOAPIC_IOREGSEL: | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         val = s->ioregsel; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case IOAPIC_IOWIN: | 
					
						
							| 
									
										
										
										
											2011-10-17 13:11:29 +02:00
										 |  |  |         if (size != 4) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         switch (s->ioregsel) { | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         case IOAPIC_REG_ID: | 
					
						
							| 
									
										
										
										
											2015-07-30 10:21:00 +02:00
										 |  |  |         case IOAPIC_REG_ARB: | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |             val = s->id << IOAPIC_ID_SHIFT; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case IOAPIC_REG_VER: | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |             val = s->version | | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |                 ((IOAPIC_NUM_PINS - 1) << IOAPIC_VER_ENTRIES_SHIFT); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             index = (s->ioregsel - IOAPIC_REG_REDTBL_BASE) >> 1; | 
					
						
							|  |  |  |             if (index >= 0 && index < IOAPIC_NUM_PINS) { | 
					
						
							|  |  |  |                 if (s->ioregsel & 1) { | 
					
						
							|  |  |  |                     val = s->ioredtbl[index] >> 32; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     val = s->ioredtbl[index] & 0xffffffff; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 18:03:10 +00:00
										 |  |  |     trace_ioapic_mem_read(addr, s->ioregsel, size, val); | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     return val; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-10 18:21:22 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This is to satisfy the hack in Linux kernel. One hack of it is to | 
					
						
							|  |  |  |  * simulate clearing the Remote IRR bit of IOAPIC entry using the | 
					
						
							|  |  |  |  * following: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * "For IO-APIC's with EOI register, we use that to do an explicit EOI. | 
					
						
							|  |  |  |  * Otherwise, we simulate the EOI message manually by changing the trigger | 
					
						
							|  |  |  |  * mode to edge and then back to level, with RTE being masked during | 
					
						
							|  |  |  |  * this." | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * (See linux kernel __eoi_ioapic_pin() comment in commit c0205701) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This is based on the assumption that, Remote IRR bit will be | 
					
						
							|  |  |  |  * cleared by IOAPIC hardware when configured as edge-triggered | 
					
						
							|  |  |  |  * interrupts. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Without this, level-triggered interrupts in IR mode might fail to | 
					
						
							|  |  |  |  * work correctly. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static inline void | 
					
						
							|  |  |  | ioapic_fix_edge_remote_irr(uint64_t *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!(*entry & IOAPIC_LVT_TRIGGER_MODE)) { | 
					
						
							|  |  |  |         /* Edge-triggered interrupts, make sure remote IRR is zero */ | 
					
						
							|  |  |  |         *entry &= ~((uint64_t)IOAPIC_LVT_REMOTE_IRR); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  | ioapic_mem_write(void *opaque, hwaddr addr, uint64_t val, | 
					
						
							| 
									
										
										
										
											2011-10-17 13:11:27 +02:00
										 |  |  |                  unsigned int size) | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-16 19:38:22 +02:00
										 |  |  |     IOAPICCommonState *s = opaque; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     int index; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  |     addr &= 0xff; | 
					
						
							| 
									
										
										
										
											2017-11-02 18:03:10 +00:00
										 |  |  |     trace_ioapic_mem_write(addr, s->ioregsel, size, val); | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (addr) { | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |     case IOAPIC_IOREGSEL: | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         s->ioregsel = val; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case IOAPIC_IOWIN: | 
					
						
							| 
									
										
										
										
											2011-10-17 13:11:29 +02:00
										 |  |  |         if (size != 4) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         switch (s->ioregsel) { | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         case IOAPIC_REG_ID: | 
					
						
							|  |  |  |             s->id = (val >> IOAPIC_ID_SHIFT) & IOAPIC_ID_MASK; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case IOAPIC_REG_VER: | 
					
						
							|  |  |  |         case IOAPIC_REG_ARB: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             index = (s->ioregsel - IOAPIC_REG_REDTBL_BASE) >> 1; | 
					
						
							|  |  |  |             if (index >= 0 && index < IOAPIC_NUM_PINS) { | 
					
						
							| 
									
										
										
										
											2016-05-10 18:21:21 +08:00
										 |  |  |                 uint64_t ro_bits = s->ioredtbl[index] & IOAPIC_RO_BITS; | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |                 if (s->ioregsel & 1) { | 
					
						
							|  |  |  |                     s->ioredtbl[index] &= 0xffffffff; | 
					
						
							|  |  |  |                     s->ioredtbl[index] |= (uint64_t)val << 32; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     s->ioredtbl[index] &= ~0xffffffffULL; | 
					
						
							|  |  |  |                     s->ioredtbl[index] |= val; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2016-05-10 18:21:21 +08:00
										 |  |  |                 /* restore RO bits */ | 
					
						
							|  |  |  |                 s->ioredtbl[index] &= IOAPIC_RW_BITS; | 
					
						
							|  |  |  |                 s->ioredtbl[index] |= ro_bits; | 
					
						
							| 
									
										
										
										
											2019-06-24 08:16:35 -07:00
										 |  |  |                 s->irq_eoi[index] = 0; | 
					
						
							| 
									
										
										
										
											2016-05-10 18:21:22 +08:00
										 |  |  |                 ioapic_fix_edge_remote_irr(&s->ioredtbl[index]); | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |                 ioapic_service(s); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:14 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |     case IOAPIC_EOI: | 
					
						
							|  |  |  |         /* Explicit EOI is only supported for IOAPIC version 0x20 */ | 
					
						
							|  |  |  |         if (size != 4 || s->version != 0x20) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ioapic_eoi_broadcast(val); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-12-17 17:16:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ioapic_update_kvm_routes(s); | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-17 13:11:27 +02:00
										 |  |  | static const MemoryRegionOps ioapic_io_ops = { | 
					
						
							|  |  |  |     .read = ioapic_mem_read, | 
					
						
							|  |  |  |     .write = ioapic_mem_write, | 
					
						
							|  |  |  |     .endianness = DEVICE_NATIVE_ENDIAN, | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:27 +08:00
										 |  |  | static void ioapic_machine_done_notify(Notifier *notifier, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_KVM
 | 
					
						
							|  |  |  |     IOAPICCommonState *s = container_of(notifier, IOAPICCommonState, | 
					
						
							|  |  |  |                                         machine_done); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (kvm_irqchip_is_split()) { | 
					
						
							|  |  |  |         X86IOMMUState *iommu = x86_iommu_get_default(); | 
					
						
							|  |  |  |         if (iommu) { | 
					
						
							|  |  |  |             /* Register this IOAPIC with IOMMU IEC notifier, so that
 | 
					
						
							|  |  |  |              * when there are IR invalidates, we can be notified to | 
					
						
							|  |  |  |              * update kernel IR cache. */ | 
					
						
							|  |  |  |             x86_iommu_iec_register_notifier(iommu, ioapic_iec_notifier, s); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-03 15:18:17 +08:00
										 |  |  | #define IOAPIC_VER_DEF 0x20
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:05 +08:00
										 |  |  | static void ioapic_realize(DeviceState *dev, Error **errp) | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:05 +08:00
										 |  |  |     IOAPICCommonState *s = IOAPIC_COMMON(dev); | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:04 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |     if (s->version != 0x11 && s->version != 0x20) { | 
					
						
							| 
									
										
										
										
											2018-10-17 10:26:34 +02:00
										 |  |  |         error_setg(errp, "IOAPIC only supports version 0x11 or 0x20 " | 
					
						
							|  |  |  |                    "(default: 0x%x).", IOAPIC_VER_DEF); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 21:25:08 -04:00
										 |  |  |     memory_region_init_io(&s->io_memory, OBJECT(s), &ioapic_io_ops, s, | 
					
						
							|  |  |  |                           "ioapic", 0x1000); | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |     s->delayed_ioapic_service_timer = | 
					
						
							|  |  |  |         timer_new_ns(QEMU_CLOCK_VIRTUAL, delayed_ioapic_service_cb, s); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:04 +08:00
										 |  |  |     qdev_init_gpio_in(dev, ioapic_set_irq, IOAPIC_NUM_PINS); | 
					
						
							| 
									
										
										
										
											2011-02-03 22:54:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:05 +08:00
										 |  |  |     ioapics[ioapic_no] = s; | 
					
						
							| 
									
										
										
										
											2016-07-14 13:56:27 +08:00
										 |  |  |     s->machine_done.notify = ioapic_machine_done_notify; | 
					
						
							|  |  |  |     qemu_add_machine_init_done_notifier(&s->machine_done); | 
					
						
							| 
									
										
										
										
											2009-03-12 20:25:12 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-06-19 07:41:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  | static void ioapic_unrealize(DeviceState *dev, Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     IOAPICCommonState *s = IOAPIC_COMMON(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     timer_del(s->delayed_ioapic_service_timer); | 
					
						
							|  |  |  |     timer_free(s->delayed_ioapic_service_timer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  | static Property ioapic_properties[] = { | 
					
						
							| 
									
										
										
										
											2017-02-03 15:18:17 +08:00
										 |  |  |     DEFINE_PROP_UINT8("version", IOAPICCommonState, version, IOAPIC_VER_DEF), | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |     DEFINE_PROP_END_OF_LIST(), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  | static void ioapic_class_init(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     IOAPICCommonClass *k = IOAPIC_COMMON_CLASS(klass); | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-05 18:16:05 +08:00
										 |  |  |     k->realize = ioapic_realize; | 
					
						
							| 
									
										
										
										
											2019-04-02 10:02:15 +02:00
										 |  |  |     k->unrealize = ioapic_unrealize; | 
					
						
							| 
									
										
										
										
											2017-01-09 16:55:53 +08:00
										 |  |  |     /*
 | 
					
						
							|  |  |  |      * If APIC is in kernel, we need to update the kernel cache after | 
					
						
							|  |  |  |      * migration, otherwise first 24 gsi routes will be invalid. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     k->post_load = ioapic_update_kvm_routes; | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     dc->reset = ioapic_reset_common; | 
					
						
							| 
									
										
										
										
											2016-08-01 21:59:19 +08:00
										 |  |  |     dc->props = ioapic_properties; | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 16:19:07 +01:00
										 |  |  | static const TypeInfo ioapic_info = { | 
					
						
							| 
									
										
										
										
											2019-01-04 18:38:31 -08:00
										 |  |  |     .name          = TYPE_IOAPIC, | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     .parent        = TYPE_IOAPIC_COMMON, | 
					
						
							|  |  |  |     .instance_size = sizeof(IOAPICCommonState), | 
					
						
							|  |  |  |     .class_init    = ioapic_class_init, | 
					
						
							| 
									
										
										
										
											2010-06-19 07:41:43 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | static void ioapic_register_types(void) | 
					
						
							| 
									
										
										
										
											2010-06-19 07:41:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     type_register_static(&ioapic_info); | 
					
						
							| 
									
										
										
										
											2010-06-19 07:41:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | type_init(ioapic_register_types) |