VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/DevAPIC.cpp@ 31808

Last change on this file since 31808 was 31592, checked in by vboxsync, 14 years ago

space

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 110.3 KB
Line 
1#ifdef VBOX
2/* $Id: DevAPIC.cpp 31592 2010-08-12 00:51:40Z vboxsync $ */
3/** @file
4 * Advanced Programmable Interrupt Controller (APIC) Device and
5 * I/O Advanced Programmable Interrupt Controller (IO-APIC) Device.
6 */
7
8/*
9 * Copyright (C) 2006-2007 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 * --------------------------------------------------------------------
19 *
20 * This code is based on:
21 *
22 * apic.c revision 1.5 @@OSETODO
23 */
24
25/*******************************************************************************
26* Header Files *
27*******************************************************************************/
28#define LOG_GROUP LOG_GROUP_DEV_APIC
29#include <VBox/pdmdev.h>
30
31#include <VBox/log.h>
32#include <VBox/stam.h>
33#include <iprt/assert.h>
34#include <iprt/asm.h>
35
36#include "Builtins2.h"
37#include "vl_vbox.h"
38
39#define MSR_IA32_APICBASE 0x1b
40#define MSR_IA32_APICBASE_BSP (1<<8)
41#define MSR_IA32_APICBASE_ENABLE (1<<11)
42#define MSR_IA32_APICBASE_X2ENABLE (1<<10)
43#define MSR_IA32_APICBASE_BASE (0xfffff<<12)
44
45#ifndef EINVAL
46# define EINVAL 1
47#endif
48
49#ifdef _MSC_VER
50# pragma warning(disable:4244)
51#endif
52
53/** The current saved state version.*/
54#define APIC_SAVED_STATE_VERSION 3
55/** The saved state version used by VirtualBox v3 and earlier.
56 * This does not include the config. */
57#define APIC_SAVED_STATE_VERSION_VBOX_30 2
58/** Some ancient version... */
59#define APIC_SAVED_STATE_VERSION_ANCIENT 1
60
61/* version 0x14: Pentium 4, Xeon; LVT count depends on that */
62#define APIC_HW_VERSION 0x14
63
64/** @def APIC_LOCK
65 * Acquires the PDM lock. */
66#define APIC_LOCK(pThis, rcBusy) \
67 do { \
68 int rc2 = PDMCritSectEnter((pThis)->CTX_SUFF(pCritSect), (rcBusy)); \
69 if (rc2 != VINF_SUCCESS) \
70 return rc2; \
71 } while (0)
72
73/** @def APIC_LOCK_VOID
74 * Acquires the PDM lock and does not expect failure (i.e. ring-3 only!). */
75#define APIC_LOCK_VOID(pThis, rcBusy) \
76 do { \
77 int rc2 = PDMCritSectEnter((pThis)->CTX_SUFF(pCritSect), (rcBusy)); \
78 AssertLogRelRCReturnVoid(rc2); \
79 } while (0)
80
81/** @def APIC_UNLOCK
82 * Releases the PDM lock. */
83#define APIC_UNLOCK(pThis) \
84 PDMCritSectLeave((pThis)->CTX_SUFF(pCritSect))
85
86/** @def IOAPIC_LOCK
87 * Acquires the PDM lock. */
88#define IOAPIC_LOCK(pThis, rc) \
89 do { \
90 int rc2 = (pThis)->CTX_SUFF(pIoApicHlp)->pfnLock((pThis)->CTX_SUFF(pDevIns), rc); \
91 if (rc2 != VINF_SUCCESS) \
92 return rc2; \
93 } while (0)
94
95/** @def IOAPIC_UNLOCK
96 * Releases the PDM lock. */
97#define IOAPIC_UNLOCK(pThis) (pThis)->CTX_SUFF(pIoApicHlp)->pfnUnlock((pThis)->CTX_SUFF(pDevIns))
98
99
100#define foreach_apic(dev, mask, code) \
101 do { \
102 uint32_t i; \
103 APICState *apic = (dev)->CTX_SUFF(paLapics); \
104 for (i = 0; i < (dev)->cCpus; i++) \
105 { \
106 if (mask & (1 << (apic->id))) \
107 { \
108 code; \
109 } \
110 apic++; \
111 } \
112 } while (0)
113
114# define set_bit(pvBitmap, iBit) ASMBitSet(pvBitmap, iBit)
115# define reset_bit(pvBitmap, iBit) ASMBitClear(pvBitmap, iBit)
116# define fls_bit(value) (ASMBitLastSetU32(value) - 1)
117# define ffs_bit(value) (ASMBitFirstSetU32(value) - 1)
118
119#endif /* VBOX */
120
121/*
122 * APIC support
123 *
124 * Copyright (c) 2004-2005 Fabrice Bellard
125 *
126 * This library is free software; you can redistribute it and/or
127 * modify it under the terms of the GNU Lesser General Public
128 * License as published by the Free Software Foundation; either
129 * version 2 of the License, or (at your option) any later version.
130 *
131 * This library is distributed in the hope that it will be useful,
132 * but WITHOUT ANY WARRANTY; without even the implied warranty of
133 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
134 * Lesser General Public License for more details.
135 *
136 * You should have received a copy of the GNU Lesser General Public
137 * License along with this library; if not, write to the Free Software
138 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
139 */
140#ifndef VBOX
141#include "vl.h"
142#endif
143
144#define DEBUG_APIC
145#define DEBUG_IOAPIC
146
147/* APIC Local Vector Table */
148#define APIC_LVT_TIMER 0
149#define APIC_LVT_THERMAL 1
150#define APIC_LVT_PERFORM 2
151#define APIC_LVT_LINT0 3
152#define APIC_LVT_LINT1 4
153#define APIC_LVT_ERROR 5
154#define APIC_LVT_NB 6
155
156/* APIC delivery modes */
157#define APIC_DM_FIXED 0
158#define APIC_DM_LOWPRI 1
159#define APIC_DM_SMI 2
160#define APIC_DM_NMI 4
161#define APIC_DM_INIT 5
162#define APIC_DM_SIPI 6
163#define APIC_DM_EXTINT 7
164
165/* APIC destination mode */
166#define APIC_DESTMODE_FLAT 0xf
167#define APIC_DESTMODE_CLUSTER 0x0
168
169#define APIC_TRIGGER_EDGE 0
170#define APIC_TRIGGER_LEVEL 1
171
172#define APIC_LVT_TIMER_PERIODIC (1<<17)
173#define APIC_LVT_MASKED (1<<16)
174#define APIC_LVT_LEVEL_TRIGGER (1<<15)
175#define APIC_LVT_REMOTE_IRR (1<<14)
176#define APIC_INPUT_POLARITY (1<<13)
177#define APIC_SEND_PENDING (1<<12)
178
179#define IOAPIC_NUM_PINS 0x18
180
181#define ESR_ILLEGAL_ADDRESS (1 << 7)
182
183#define APIC_SV_ENABLE (1 << 8)
184
185#ifdef VBOX
186#define APIC_MAX_PATCH_ATTEMPTS 100
187
188typedef uint32_t PhysApicId;
189typedef uint32_t LogApicId;
190#endif
191
192typedef struct APICState {
193#ifndef VBOX
194 CPUState *cpu_env;
195#endif /* !VBOX */
196 uint32_t apicbase;
197#ifdef VBOX
198 /* Task priority register (interrupt level) */
199 uint32_t tpr;
200 /* Logical APIC id - user programmable */
201 LogApicId id;
202 /* Physical APIC id - not visible to user, constant */
203 PhysApicId phys_id;
204 /** @todo: is it logical or physical? Not really used anyway now. */
205 PhysApicId arb_id;
206#else
207 uint8_t tpr;
208 uint8_t id;
209 uint8_t arb_id;
210#endif
211 uint32_t spurious_vec;
212 uint8_t log_dest;
213 uint8_t dest_mode;
214 uint32_t isr[8]; /* in service register */
215 uint32_t tmr[8]; /* trigger mode register */
216 uint32_t irr[8]; /* interrupt request register */
217 uint32_t lvt[APIC_LVT_NB];
218 uint32_t esr; /* error register */
219 uint32_t icr[2];
220 uint32_t divide_conf;
221 int count_shift;
222 uint32_t initial_count;
223#ifdef VBOX
224 uint32_t Alignment0;
225#endif
226#ifndef VBOX
227 int64_t initial_count_load_time, next_time;
228 QEMUTimer *timer;
229 struct APICState *next_apic;
230#else
231 /** The time stamp of the initial_count load, i.e. when it was started. */
232 uint64_t initial_count_load_time;
233 /** The time stamp of the next timer callback. */
234 uint64_t next_time;
235 /** The APIC timer - R3 Ptr. */
236 PTMTIMERR3 pTimerR3;
237 /** The APIC timer - R0 Ptr. */
238 PTMTIMERR0 pTimerR0;
239 /** The APIC timer - RC Ptr. */
240 PTMTIMERRC pTimerRC;
241 /** Whether the timer is armed or not */
242 bool fTimerArmed;
243 /** Alignment */
244 bool afAlignment[3];
245 /** Timer description timer. */
246 R3PTRTYPE(char *) pszDesc;
247# ifdef VBOX_WITH_STATISTICS
248# if HC_ARCH_BITS == 32
249 uint32_t u32Alignment0;
250# endif
251 STAMCOUNTER StatTimerSetInitialCount;
252 STAMCOUNTER StatTimerSetInitialCountArm;
253 STAMCOUNTER StatTimerSetInitialCountDisarm;
254 STAMCOUNTER StatTimerSetLvt;
255 STAMCOUNTER StatTimerSetLvtClearPeriodic;
256 STAMCOUNTER StatTimerSetLvtPostponed;
257 STAMCOUNTER StatTimerSetLvtArmed;
258 STAMCOUNTER StatTimerSetLvtArm;
259 STAMCOUNTER StatTimerSetLvtArmRetries;
260 STAMCOUNTER StatTimerSetLvtNoRelevantChange;
261# endif
262#endif /* VBOX */
263} APICState;
264#ifdef VBOX
265AssertCompileMemberAlignment(APICState, initial_count_load_time, 8);
266# ifdef VBOX_WITH_STATISTICS
267AssertCompileMemberAlignment(APICState, StatTimerSetInitialCount, 8);
268# endif
269#endif
270
271struct IOAPICState {
272 uint8_t id;
273 uint8_t ioregsel;
274
275 uint32_t irr;
276 uint64_t ioredtbl[IOAPIC_NUM_PINS];
277
278#ifdef VBOX
279 /** The device instance - R3 Ptr. */
280 PPDMDEVINSR3 pDevInsR3;
281 /** The IOAPIC helpers - R3 Ptr. */
282 PCPDMIOAPICHLPR3 pIoApicHlpR3;
283
284 /** The device instance - R0 Ptr. */
285 PPDMDEVINSR0 pDevInsR0;
286 /** The IOAPIC helpers - R0 Ptr. */
287 PCPDMIOAPICHLPR0 pIoApicHlpR0;
288
289 /** The device instance - RC Ptr. */
290 PPDMDEVINSRC pDevInsRC;
291 /** The IOAPIC helpers - RC Ptr. */
292 PCPDMIOAPICHLPRC pIoApicHlpRC;
293
294# ifdef VBOX_WITH_STATISTICS
295 STAMCOUNTER StatMMIOReadGC;
296 STAMCOUNTER StatMMIOReadHC;
297 STAMCOUNTER StatMMIOWriteGC;
298 STAMCOUNTER StatMMIOWriteHC;
299 STAMCOUNTER StatSetIrqGC;
300 STAMCOUNTER StatSetIrqHC;
301# endif
302#endif /* VBOX */
303};
304
305#ifdef VBOX
306typedef struct IOAPICState IOAPICState;
307
308typedef struct
309{
310 /** The device instance - R3 Ptr. */
311 PPDMDEVINSR3 pDevInsR3;
312 /** The APIC helpers - R3 Ptr. */
313 PCPDMAPICHLPR3 pApicHlpR3;
314 /** LAPICs states - R3 Ptr */
315 R3PTRTYPE(APICState *) paLapicsR3;
316 /** The critical section - R3 Ptr. */
317 R3PTRTYPE(PPDMCRITSECT) pCritSectR3;
318
319 /** The device instance - R0 Ptr. */
320 PPDMDEVINSR0 pDevInsR0;
321 /** The APIC helpers - R0 Ptr. */
322 PCPDMAPICHLPR0 pApicHlpR0;
323 /** LAPICs states - R0 Ptr */
324 R0PTRTYPE(APICState *) paLapicsR0;
325 /** The critical section - R3 Ptr. */
326 R0PTRTYPE(PPDMCRITSECT) pCritSectR0;
327
328 /** The device instance - RC Ptr. */
329 PPDMDEVINSRC pDevInsRC;
330 /** The APIC helpers - RC Ptr. */
331 PCPDMAPICHLPRC pApicHlpRC;
332 /** LAPICs states - RC Ptr */
333 RCPTRTYPE(APICState *) paLapicsRC;
334 /** The critical section - R3 Ptr. */
335 RCPTRTYPE(PPDMCRITSECT) pCritSectRC;
336
337 /** APIC specification version in this virtual hardware configuration. */
338 PDMAPICVERSION enmVersion;
339
340 /** Number of attempts made to optimize TPR accesses. */
341 uint32_t cTPRPatchAttempts;
342
343 /** Number of CPUs on the system (same as LAPIC count). */
344 uint32_t cCpus;
345 /** Whether we've got an IO APIC or not. */
346 bool fIoApic;
347 /** Alignment padding. */
348 bool afPadding[3];
349
350# ifdef VBOX_WITH_STATISTICS
351 STAMCOUNTER StatMMIOReadGC;
352 STAMCOUNTER StatMMIOReadHC;
353 STAMCOUNTER StatMMIOWriteGC;
354 STAMCOUNTER StatMMIOWriteHC;
355 STAMCOUNTER StatClearedActiveIrq;
356# endif
357} APICDeviceInfo;
358# ifdef VBOX_WITH_STATISTICS
359AssertCompileMemberAlignment(APICDeviceInfo, StatMMIOReadGC, 8);
360# endif
361#endif /* VBOX */
362
363#ifndef VBOX_DEVICE_STRUCT_TESTCASE
364
365#ifndef VBOX
366static int apic_io_memory;
367static APICState *first_local_apic = NULL;
368static int last_apic_id = 0;
369#endif /* !VBOX */
370
371
372#ifdef VBOX
373/*******************************************************************************
374* Internal Functions *
375*******************************************************************************/
376RT_C_DECLS_BEGIN
377PDMBOTHCBDECL(int) apicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
378PDMBOTHCBDECL(int) apicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
379PDMBOTHCBDECL(int) apicGetInterrupt(PPDMDEVINS pDevIns);
380PDMBOTHCBDECL(bool) apicHasPendingIrq(PPDMDEVINS pDevIns);
381PDMBOTHCBDECL(void) apicSetBase(PPDMDEVINS pDevIns, uint64_t val);
382PDMBOTHCBDECL(uint64_t) apicGetBase(PPDMDEVINS pDevIns);
383PDMBOTHCBDECL(void) apicSetTPR(PPDMDEVINS pDevIns, VMCPUID idCpu, uint8_t val);
384PDMBOTHCBDECL(uint8_t) apicGetTPR(PPDMDEVINS pDevIns, VMCPUID idCpu);
385PDMBOTHCBDECL(int) apicBusDeliverCallback(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode,
386 uint8_t u8DeliveryMode, uint8_t iVector, uint8_t u8Polarity,
387 uint8_t u8TriggerMode);
388PDMBOTHCBDECL(int) apicLocalInterrupt(PPDMDEVINS pDevIns, uint8_t u8Pin, uint8_t u8Level);
389PDMBOTHCBDECL(int) apicWriteMSR(PPDMDEVINS pDevIns, VMCPUID iCpu, uint32_t u32Reg, uint64_t u64Value);
390PDMBOTHCBDECL(int) apicReadMSR(PPDMDEVINS pDevIns, VMCPUID iCpu, uint32_t u32Reg, uint64_t *pu64Value);
391PDMBOTHCBDECL(int) ioapicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
392PDMBOTHCBDECL(int) ioapicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
393PDMBOTHCBDECL(void) ioapicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel);
394
395static void apic_update_tpr(APICDeviceInfo *dev, APICState* s, uint32_t val);
396RT_C_DECLS_END
397
398static void apic_eoi(APICDeviceInfo *dev, APICState* s); /* */
399static uint32_t apic_get_delivery_bitmask(APICDeviceInfo* dev, uint8_t dest, uint8_t dest_mode);
400static int apic_deliver(APICDeviceInfo* dev, APICState *s,
401 uint8_t dest, uint8_t dest_mode,
402 uint8_t delivery_mode, uint8_t vector_num,
403 uint8_t polarity, uint8_t trigger_mode);
404static int apic_get_arb_pri(APICState *s);
405static int apic_get_ppr(APICState *s);
406static uint32_t apic_get_current_count(APICDeviceInfo* dev, APICState *s);
407static void apicTimerSetInitialCount(APICDeviceInfo *dev, APICState *s, uint32_t initial_count);
408static void apicTimerSetLvt(APICDeviceInfo *dev, APICState *pThis, uint32_t fNew);
409static void apicSendInitIpi(APICDeviceInfo* dev, APICState *s);
410
411#endif /* VBOX */
412
413static void apic_init_ipi(APICDeviceInfo* dev, APICState *s);
414static void apic_set_irq(APICDeviceInfo* dev, APICState *s, int vector_num, int trigger_mode);
415static bool apic_update_irq(APICDeviceInfo* dev, APICState *s);
416
417
418#ifdef VBOX
419
420DECLINLINE(APICState*) getLapicById(APICDeviceInfo* dev, VMCPUID id)
421{
422 AssertFatalMsg(id < dev->cCpus, ("CPU id %d out of range\n", id));
423 return &dev->CTX_SUFF(paLapics)[id];
424}
425
426DECLINLINE(APICState*) getLapic(APICDeviceInfo* dev)
427{
428 /* LAPIC's array is indexed by CPU id */
429 VMCPUID id = dev->CTX_SUFF(pApicHlp)->pfnGetCpuId(dev->CTX_SUFF(pDevIns));
430 return getLapicById(dev, id);
431}
432
433DECLINLINE(VMCPUID) getCpuFromLapic(APICDeviceInfo* dev, APICState *s)
434{
435 /* for now we assume LAPIC physical id == CPU id */
436 return VMCPUID(s->phys_id);
437}
438
439DECLINLINE(void) cpuSetInterrupt(APICDeviceInfo* dev, APICState *s, PDMAPICIRQ enmType = PDMAPICIRQ_HARDWARE)
440{
441 LogFlow(("apic: setting interrupt flag for cpu %d\n", getCpuFromLapic(dev, s)));
442 dev->CTX_SUFF(pApicHlp)->pfnSetInterruptFF(dev->CTX_SUFF(pDevIns), enmType,
443 getCpuFromLapic(dev, s));
444}
445
446DECLINLINE(void) cpuClearInterrupt(APICDeviceInfo* dev, APICState *s, PDMAPICIRQ enmType = PDMAPICIRQ_HARDWARE)
447{
448 LogFlow(("apic: clear interrupt flag\n"));
449 dev->CTX_SUFF(pApicHlp)->pfnClearInterruptFF(dev->CTX_SUFF(pDevIns), enmType,
450 getCpuFromLapic(dev, s));
451}
452
453# ifdef IN_RING3
454
455DECLINLINE(void) cpuSendSipi(APICDeviceInfo* dev, APICState *s, int vector)
456{
457 Log2(("apic: send SIPI vector=%d\n", vector));
458
459 dev->pApicHlpR3->pfnSendSipi(dev->pDevInsR3,
460 getCpuFromLapic(dev, s),
461 vector);
462}
463
464DECLINLINE(void) cpuSendInitIpi(APICDeviceInfo* dev, APICState *s)
465{
466 Log2(("apic: send init IPI\n"));
467
468 dev->pApicHlpR3->pfnSendInitIpi(dev->pDevInsR3,
469 getCpuFromLapic(dev, s));
470}
471
472# endif /* IN_RING3 */
473
474DECLINLINE(uint32_t) getApicEnableBits(APICDeviceInfo* dev)
475{
476 switch (dev->enmVersion)
477 {
478 case PDMAPICVERSION_NONE:
479 return 0;
480 case PDMAPICVERSION_APIC:
481 return MSR_IA32_APICBASE_ENABLE;
482 case PDMAPICVERSION_X2APIC:
483 return MSR_IA32_APICBASE_ENABLE | MSR_IA32_APICBASE_X2ENABLE ;
484 default:
485 AssertMsgFailed(("Unsuported APIC version %d\n", dev->enmVersion));
486 return 0;
487 }
488}
489
490DECLINLINE(PDMAPICVERSION) getApicMode(APICState *apic)
491{
492 switch (((apic->apicbase) >> 10) & 0x3)
493 {
494 case 0:
495 return PDMAPICVERSION_NONE;
496 case 1:
497 default:
498 /* Invalid */
499 return PDMAPICVERSION_NONE;
500 case 2:
501 return PDMAPICVERSION_APIC;
502 case 3:
503 return PDMAPICVERSION_X2APIC;
504 }
505}
506
507#endif /* VBOX */
508
509#ifndef VBOX
510static void apic_bus_deliver(uint32_t deliver_bitmask, uint8_t delivery_mode,
511 uint8_t vector_num, uint8_t polarity,
512 uint8_t trigger_mode)
513{
514 APICState *apic_iter;
515#else /* VBOX */
516static int apic_bus_deliver(APICDeviceInfo* dev,
517 uint32_t deliver_bitmask, uint8_t delivery_mode,
518 uint8_t vector_num, uint8_t polarity,
519 uint8_t trigger_mode)
520{
521#endif /* VBOX */
522
523 LogFlow(("apic_bus_deliver mask=%x mode=%x vector=%x polarity=%x trigger_mode=%x\n", deliver_bitmask, delivery_mode, vector_num, polarity, trigger_mode));
524 switch (delivery_mode) {
525 case APIC_DM_LOWPRI:
526 {
527 int d = -1;
528 if (deliver_bitmask)
529 d = ffs_bit(deliver_bitmask);
530 if (d >= 0)
531 {
532 APICState* apic = getLapicById(dev, d);
533 apic_set_irq(dev, apic, vector_num, trigger_mode);
534 }
535 return VINF_SUCCESS;
536 }
537 case APIC_DM_FIXED:
538 /* XXX: arbitration */
539 break;
540
541 case APIC_DM_SMI:
542 foreach_apic(dev, deliver_bitmask,
543 cpuSetInterrupt(dev, apic, PDMAPICIRQ_SMI));
544 return VINF_SUCCESS;
545
546 case APIC_DM_NMI:
547 foreach_apic(dev, deliver_bitmask,
548 cpuSetInterrupt(dev, apic, PDMAPICIRQ_NMI));
549 return VINF_SUCCESS;
550
551 case APIC_DM_INIT:
552 /* normal INIT IPI sent to processors */
553#ifdef VBOX
554#ifdef IN_RING3
555 foreach_apic(dev, deliver_bitmask,
556 apicSendInitIpi(dev, apic));
557 return VINF_SUCCESS;
558#else
559 /* We shall send init IPI only in R3, R0 calls should be
560 rescheduled to R3 */
561 return VINF_IOM_HC_MMIO_READ_WRITE;
562#endif /* IN_RING3 */
563
564#else
565 for (apic_iter = first_local_apic; apic_iter != NULL;
566 apic_iter = apic_iter->next_apic) {
567 apic_init_ipi(apic_iter);
568 }
569#endif
570
571 case APIC_DM_EXTINT:
572 /* handled in I/O APIC code */
573 break;
574
575 default:
576 return VINF_SUCCESS;
577 }
578
579#ifdef VBOX
580 foreach_apic(dev, deliver_bitmask,
581 apic_set_irq (dev, apic, vector_num, trigger_mode));
582 return VINF_SUCCESS;
583#else /* VBOX */
584 for (apic_iter = first_local_apic; apic_iter != NULL;
585 apic_iter = apic_iter->next_apic) {
586 if (deliver_bitmask & (1 << apic_iter->id))
587 apic_set_irq(apic_iter, vector_num, trigger_mode);
588 }
589#endif /* VBOX */
590}
591
592#ifndef VBOX
593void cpu_set_apic_base(CPUState *env, uint64_t val)
594{
595 APICState *s = env->apic_state;
596#ifdef DEBUG_APIC
597 Log(("cpu_set_apic_base: %016llx\n", val));
598#endif
599
600 s->apicbase = (val & 0xfffff000) |
601 (s->apicbase & (MSR_IA32_APICBASE_BSP | MSR_IA32_APICBASE_ENABLE));
602 /* if disabled, cannot be enabled again */
603 if (!(val & MSR_IA32_APICBASE_ENABLE)) {
604 s->apicbase &= ~MSR_IA32_APICBASE_ENABLE;
605 env->cpuid_features &= ~CPUID_APIC;
606 s->spurious_vec &= ~APIC_SV_ENABLE;
607 }
608}
609#else /* VBOX */
610PDMBOTHCBDECL(void) apicSetBase(PPDMDEVINS pDevIns, uint64_t val)
611{
612 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
613 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
614 APICState *s = getLapic(dev); /** @todo fix interface */
615 Log(("cpu_set_apic_base: %016RX64\n", val));
616
617 /** @todo: do we need to lock here ? */
618 /* APIC_LOCK_VOID(dev, VERR_INTERNAL_ERROR); */
619 /** @todo If this change is valid immediately, then we should change the MMIO registration! */
620 /* We cannot change if this CPU is BSP or not by writing to MSR - it's hardwired */
621 PDMAPICVERSION oldMode = getApicMode(s);
622 s->apicbase =
623 (val & 0xfffff000) | /* base */
624 (val & getApicEnableBits(dev)) | /* mode */
625 (s->apicbase & MSR_IA32_APICBASE_BSP) /* keep BSP bit */;
626 PDMAPICVERSION newMode = getApicMode(s);
627
628 if (oldMode != newMode)
629 {
630 switch (newMode)
631 {
632 case PDMAPICVERSION_NONE:
633 {
634 s->spurious_vec &= ~APIC_SV_ENABLE;
635 /* Clear any pending APIC interrupt action flag. */
636 cpuClearInterrupt(dev, s);
637 /** @todo: why do we do that? */
638 dev->CTX_SUFF(pApicHlp)->pfnChangeFeature(pDevIns, PDMAPICVERSION_NONE);
639 break;
640 }
641 case PDMAPICVERSION_APIC:
642 /** @todo: map MMIO ranges, if needed */
643 break;
644 case PDMAPICVERSION_X2APIC:
645 /** @todo: unmap MMIO ranges of this APIC, according to the spec */
646 break;
647 default:
648 break;
649 }
650 }
651 /* APIC_UNLOCK(dev); */
652}
653#endif /* VBOX */
654
655#ifndef VBOX
656
657uint64_t cpu_get_apic_base(CPUState *env)
658{
659 APICState *s = env->apic_state;
660#ifdef DEBUG_APIC
661 Log(("cpu_get_apic_base: %016llx\n", (uint64_t)s->apicbase));
662#endif
663 return s->apicbase;
664}
665
666void cpu_set_apic_tpr(CPUX86State *env, uint8_t val)
667{
668 APICState *s = env->apic_state;
669 s->tpr = (val & 0x0f) << 4;
670 apic_update_irq(s);
671}
672
673uint8_t cpu_get_apic_tpr(CPUX86State *env)
674{
675 APICState *s = env->apic_state;
676 return s->tpr >> 4;
677}
678
679static int fls_bit(int value)
680{
681 unsigned int ret = 0;
682
683#ifdef HOST_I386
684 __asm__ __volatile__ ("bsr %1, %0\n" : "+r" (ret) : "rm" (value));
685 return ret;
686#else
687 if (value > 0xffff)
688 value >>= 16, ret = 16;
689 if (value > 0xff)
690 value >>= 8, ret += 8;
691 if (value > 0xf)
692 value >>= 4, ret += 4;
693 if (value > 0x3)
694 value >>= 2, ret += 2;
695 return ret + (value >> 1);
696#endif
697}
698
699static inline void set_bit(uint32_t *tab, int index)
700{
701 int i, mask;
702 i = index >> 5;
703 mask = 1 << (index & 0x1f);
704 tab[i] |= mask;
705}
706
707static inline void reset_bit(uint32_t *tab, int index)
708{
709 int i, mask;
710 i = index >> 5;
711 mask = 1 << (index & 0x1f);
712 tab[i] &= ~mask;
713}
714
715
716#else /* VBOX */
717
718PDMBOTHCBDECL(uint64_t) apicGetBase(PPDMDEVINS pDevIns)
719{
720 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
721 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
722 APICState *s = getLapic(dev); /** @todo fix interface */
723 LogFlow(("apicGetBase: %016llx\n", (uint64_t)s->apicbase));
724 return s->apicbase;
725}
726
727PDMBOTHCBDECL(void) apicSetTPR(PPDMDEVINS pDevIns, VMCPUID idCpu, uint8_t val)
728{
729 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
730 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
731 APICState *s = getLapicById(dev, idCpu);
732 LogFlow(("apicSetTPR: val=%#x (trp %#x -> %#x)\n", val, s->tpr, val));
733 apic_update_tpr(dev, s, val);
734}
735
736PDMBOTHCBDECL(uint8_t) apicGetTPR(PPDMDEVINS pDevIns, VMCPUID idCpu)
737{
738 /* We don't perform any locking here as that would cause a lot of contention for VT-x/AMD-V. */
739 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
740 APICState *s = getLapicById(dev, idCpu);
741 Log2(("apicGetTPR: returns %#x\n", s->tpr));
742 return s->tpr;
743}
744
745/**
746 * x2APIC MSR write interface.
747 *
748 * @returns VBox status code.
749 *
750 * @param pDevIns The device instance.
751 * @param idCpu The ID of the virtual CPU and thereby APIC index.
752 * @param u32Reg Register to write (ecx).
753 * @param u64Value The value to write (eax:edx / rax).
754 *
755 */
756PDMBOTHCBDECL(int) apicWriteMSR(PPDMDEVINS pDevIns, VMCPUID idCpu, uint32_t u32Reg, uint64_t u64Value)
757{
758 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
759 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
760 int rc = VINF_SUCCESS;
761
762 if (dev->enmVersion < PDMAPICVERSION_X2APIC)
763 return VERR_EM_INTERPRETER;
764
765 APICState *pThis = getLapicById(dev, idCpu);
766
767 uint32_t index = (u32Reg - MSR_IA32_APIC_START) & 0xff;
768 switch (index)
769 {
770 case 0x02:
771 pThis->id = (u64Value >> 24);
772 break;
773 case 0x03:
774 break;
775 case 0x08:
776 apic_update_tpr(dev, pThis, u64Value);
777 break;
778 case 0x09: case 0x0a:
779 Log(("apicWriteMSR: write to read-only register %d ignored\n", index));
780 break;
781 case 0x0b: /* EOI */
782 apic_eoi(dev, pThis);
783 break;
784 case 0x0d:
785 pThis->log_dest = u64Value >> 24;
786 break;
787 case 0x0e:
788 pThis->dest_mode = u64Value >> 28;
789 break;
790 case 0x0f:
791 pThis->spurious_vec = u64Value & 0x1ff;
792 apic_update_irq(dev, pThis);
793 break;
794 case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
795 case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
796 case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
797 case 0x28:
798 Log(("apicWriteMSR: write to read-only register %d ignored\n", index));
799 break;
800
801 case 0x30:
802 /* Here one of the differences with regular APIC: ICR is single 64-bit register */
803 pThis->icr[0] = (uint32_t)u64Value;
804 pThis->icr[1] = (uint32_t)(u64Value >> 32);
805 rc = apic_deliver(dev, pThis, (pThis->icr[1] >> 24) & 0xff, (pThis->icr[0] >> 11) & 1,
806 (pThis->icr[0] >> 8) & 7, (pThis->icr[0] & 0xff),
807 (pThis->icr[0] >> 14) & 1, (pThis->icr[0] >> 15) & 1);
808 break;
809 case 0x32 + APIC_LVT_TIMER:
810 AssertCompile(APIC_LVT_TIMER == 0);
811 apicTimerSetLvt(dev, pThis, u64Value);
812 break;
813
814 case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
815 pThis->lvt[index - 0x32] = u64Value;
816 break;
817 case 0x38:
818 apicTimerSetInitialCount(dev, pThis, u64Value);
819 break;
820 case 0x39:
821 Log(("apicWriteMSR: write to read-only register %d ignored\n", index));
822 break;
823 case 0x3e:
824 {
825 int v;
826 pThis->divide_conf = u64Value & 0xb;
827 v = (pThis->divide_conf & 3) | ((pThis->divide_conf >> 1) & 4);
828 pThis->count_shift = (v + 1) & 7;
829 break;
830 }
831 case 0x3f:
832 {
833 /* Self IPI, see x2APIC book 2.4.5 */
834 int vector = u64Value & 0xff;
835 rc = apic_bus_deliver(dev,
836 1 << getLapicById(dev, idCpu)->id /* Self */,
837 0 /* Delivery mode - fixed */,
838 vector,
839 0 /* Polarity - conform to the bus */,
840 0 /* Trigger mode - edge */);
841 break;
842 }
843 default:
844 AssertMsgFailed(("apicWriteMSR: unknown index %x\n", index));
845 pThis->esr |= ESR_ILLEGAL_ADDRESS;
846 break;
847 }
848
849 return rc;
850}
851
852/**
853 * x2APIC MSR read interface.
854 *
855 * @returns VBox status code.
856 *
857 * @param pDevIns The device instance.
858 * @param idCpu The ID of the virtual CPU and thereby APIC index.
859 * @param u32Reg Register to write (ecx).
860 * @param pu64Value Where to return the value (eax:edx / rax).
861 */
862PDMBOTHCBDECL(int) apicReadMSR(PPDMDEVINS pDevIns, VMCPUID idCpu, uint32_t u32Reg, uint64_t *pu64Value)
863{
864 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
865 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
866
867 if (dev->enmVersion < PDMAPICVERSION_X2APIC)
868 return VERR_EM_INTERPRETER;
869
870 uint32_t index = (u32Reg - MSR_IA32_APIC_START) & 0xff;
871 APICState* apic = getLapicById(dev, idCpu);
872 uint64_t val = 0;
873
874 switch (index)
875 {
876 case 0x02: /* id */
877 val = apic->id << 24;
878 break;
879 case 0x03: /* version */
880 val = APIC_HW_VERSION |
881 ((APIC_LVT_NB - 1) << 16) /* Max LVT index */ |
882 (0 << 24) /* Support for EOI broadcast supression */;
883 break;
884 case 0x08:
885 val = apic->tpr;
886 break;
887 case 0x09:
888 val = apic_get_arb_pri(apic);
889 break;
890 case 0x0a:
891 /* ppr */
892 val = apic_get_ppr(apic);
893 break;
894 case 0x0b:
895 val = 0;
896 break;
897 case 0x0d:
898 val = (uint64_t)apic->log_dest << 24;
899 break;
900 case 0x0e:
901 /* Bottom 28 bits are always 1 */
902 val = ((uint64_t)apic->dest_mode << 28) | 0xfffffff;
903 break;
904 case 0x0f:
905 val = apic->spurious_vec;
906 break;
907 case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
908 val = apic->isr[index & 7];
909 break;
910 case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
911 val = apic->tmr[index & 7];
912 break;
913 case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
914 val = apic->irr[index & 7];
915 break;
916 case 0x28:
917 val = apic->esr;
918 break;
919 case 0x30:
920 /* Here one of the differences with regular APIC: ICR is single 64-bit register */
921 val = ((uint64_t)apic->icr[1] << 32) | apic->icr[0];
922 break;
923 case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
924 val = apic->lvt[index - 0x32];
925 break;
926 case 0x38:
927 val = apic->initial_count;
928 break;
929 case 0x39:
930 val = apic_get_current_count(dev, apic);
931 break;
932 case 0x3e:
933 val = apic->divide_conf;
934 break;
935 case 0x3f:
936 /* Self IPI register is write only */
937 Log(("apicReadMSR: read from write-only register %d ignored\n", index));
938 break;
939 case 0x2f:
940 /**
941 * Correctable machine check exception vector, @todo: implement me!
942 */
943 default:
944 AssertMsgFailed(("apicReadMSR: unknown index %x\n", index));
945 /**
946 * @todo: according to spec when APIC writes to ESR it msut raise error interrupt,
947 * i.e. LVT[5]
948 */
949 apic->esr |= ESR_ILLEGAL_ADDRESS;
950 val = 0;
951 break;
952 }
953 *pu64Value = val;
954 return VINF_SUCCESS;
955}
956
957/**
958 * More or less private interface between IOAPIC, only PDM is responsible
959 * for connecting the two devices.
960 */
961PDMBOTHCBDECL(int) apicBusDeliverCallback(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode,
962 uint8_t u8DeliveryMode, uint8_t iVector, uint8_t u8Polarity,
963 uint8_t u8TriggerMode)
964{
965 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
966 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
967 LogFlow(("apicBusDeliverCallback: pDevIns=%p u8Dest=%#x u8DestMode=%#x u8DeliveryMode=%#x iVector=%#x u8Polarity=%#x u8TriggerMode=%#x\n",
968 pDevIns, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode));
969 return apic_bus_deliver(dev, apic_get_delivery_bitmask(dev, u8Dest, u8DestMode),
970 u8DeliveryMode, iVector, u8Polarity, u8TriggerMode);
971}
972
973/**
974 * Local interrupt delivery, for devices attached to the CPU's LINT0/LINT1 pin.
975 * Normally used for 8259A PIC and NMI.
976 */
977PDMBOTHCBDECL(int) apicLocalInterrupt(PPDMDEVINS pDevIns, uint8_t u8Pin, uint8_t u8Level)
978{
979 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
980 APICState *s = getLapicById(dev, 0);
981
982 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
983 LogFlow(("apicLocalInterrupt: pDevIns=%p u8Pin=%x u8Level=%x\n", pDevIns, u8Pin, u8Level));
984
985 /* If LAPIC is disabled, go straight to the CPU. */
986 if (!(s->spurious_vec & APIC_SV_ENABLE))
987 {
988 LogFlow(("apicLocalInterrupt: LAPIC disabled, delivering directly to CPU core.\n"));
989 if (u8Level)
990 cpuSetInterrupt(dev, s, PDMAPICIRQ_EXTINT);
991 else
992 cpuClearInterrupt(dev, s, PDMAPICIRQ_EXTINT);
993
994 return VINF_SUCCESS;
995 }
996
997 /* If LAPIC is enabled, interrupts are subject to LVT programming. */
998
999 /* There are only two local interrupt pins. */
1000 AssertMsgReturn(u8Pin <= 1, ("Invalid LAPIC pin %d\n", u8Pin), VERR_INVALID_PARAMETER);
1001
1002 /* NB: We currently only deliver local interrupts to the first CPU. In theory they
1003 * should be delivered to all CPUs and it is the guest's responsibility to ensure
1004 * no more than one CPU has the interrupt unmasked.
1005 */
1006 uint32_t u32Lvec;
1007
1008 u32Lvec = s->lvt[APIC_LVT_LINT0 + u8Pin]; /* Fetch corresponding LVT entry. */
1009 /* Drop int if entry is masked. May not be correct for level-triggered interrupts. */
1010 if (!(u32Lvec & APIC_LVT_MASKED))
1011 { uint8_t u8Delivery;
1012 PDMAPICIRQ enmType;
1013
1014 u8Delivery = (u32Lvec >> 8) & 7;
1015 switch (u8Delivery)
1016 {
1017 case APIC_DM_EXTINT:
1018 Assert(u8Pin == 0); /* PIC should be wired to LINT0. */
1019 enmType = PDMAPICIRQ_EXTINT;
1020 /* ExtINT can be both set and cleared, NMI/SMI/INIT can only be set. */
1021 LogFlow(("apicLocalInterrupt: %s ExtINT interrupt\n", u8Level ? "setting" : "clearing"));
1022 if (u8Level)
1023 cpuSetInterrupt(dev, s, enmType);
1024 else
1025 cpuClearInterrupt(dev, s, enmType);
1026 return VINF_SUCCESS;
1027 case APIC_DM_NMI:
1028 /* External NMI should be wired to LINT1, but Linux sometimes programs
1029 * LVT0 to NMI delivery mode as well.
1030 */
1031 enmType = PDMAPICIRQ_NMI;
1032 /* Currently delivering NMIs through here causes problems with NMI watchdogs
1033 * on certain Linux kernels, e.g. 64-bit CentOS 5.3. Disable NMIs for now.
1034 */
1035 return VINF_SUCCESS;
1036 case APIC_DM_SMI:
1037 enmType = PDMAPICIRQ_SMI;
1038 break;
1039 case APIC_DM_FIXED:
1040 {
1041 /** @todo implement APIC_DM_FIXED! */
1042 static unsigned s_c = 0;
1043 if (s_c++ < 5)
1044 LogRel(("delivery type APIC_DM_FIXED not implemented. u8Pin=%d u8Level=%d\n", u8Pin, u8Level));
1045 return VINF_SUCCESS;
1046 }
1047 case APIC_DM_INIT:
1048 /** @todo implement APIC_DM_INIT? */
1049 default:
1050 {
1051 static unsigned s_c = 0;
1052 if (s_c++ < 100)
1053 AssertLogRelMsgFailed(("delivery type %d not implemented. u8Pin=%d u8Level=%d\n", u8Delivery, u8Pin, u8Level));
1054 return VERR_INTERNAL_ERROR_4;
1055 }
1056 }
1057 LogFlow(("apicLocalInterrupt: setting local interrupt type %d\n", enmType));
1058 cpuSetInterrupt(dev, s, enmType);
1059 }
1060 return VINF_SUCCESS;
1061}
1062
1063#endif /* VBOX */
1064
1065/* return -1 if no bit is set */
1066static int get_highest_priority_int(uint32_t *tab)
1067{
1068 int i;
1069 for(i = 7; i >= 0; i--) {
1070 if (tab[i] != 0) {
1071 return i * 32 + fls_bit(tab[i]);
1072 }
1073 }
1074 return -1;
1075}
1076
1077static int apic_get_ppr(APICState *s)
1078{
1079 int tpr, isrv, ppr;
1080
1081 tpr = (s->tpr >> 4);
1082 isrv = get_highest_priority_int(s->isr);
1083 if (isrv < 0)
1084 isrv = 0;
1085 isrv >>= 4;
1086 if (tpr >= isrv)
1087 ppr = s->tpr;
1088 else
1089 ppr = isrv << 4;
1090 return ppr;
1091}
1092
1093static int apic_get_ppr_zero_tpr(APICState *s)
1094{
1095 int isrv;
1096
1097 isrv = get_highest_priority_int(s->isr);
1098 if (isrv < 0)
1099 isrv = 0;
1100 return isrv;
1101}
1102
1103static int apic_get_arb_pri(APICState *s)
1104{
1105 /* XXX: arbitration */
1106 return 0;
1107}
1108
1109/* signal the CPU if an irq is pending */
1110static bool apic_update_irq(APICDeviceInfo *dev, APICState* s)
1111{
1112 int irrv, ppr;
1113 if (!(s->spurious_vec & APIC_SV_ENABLE))
1114#ifdef VBOX
1115 {
1116 /* Clear any pending APIC interrupt action flag. */
1117 cpuClearInterrupt(dev, s);
1118 return false;
1119 }
1120#else
1121 return false;
1122#endif /* VBOX */
1123 irrv = get_highest_priority_int(s->irr);
1124 if (irrv < 0)
1125 return false;
1126 ppr = apic_get_ppr(s);
1127 if (ppr && (irrv & 0xf0) <= (ppr & 0xf0))
1128 return false;
1129#ifndef VBOX
1130 cpu_interrupt(s->cpu_env, CPU_INTERRUPT_HARD);
1131#else
1132 cpuSetInterrupt(dev, s);
1133 return true;
1134#endif
1135}
1136
1137#ifdef VBOX
1138
1139/* Check if the APIC has a pending interrupt/if a TPR change would active one. */
1140PDMBOTHCBDECL(bool) apicHasPendingIrq(PPDMDEVINS pDevIns)
1141{
1142 int irrv, ppr;
1143 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
1144 if (!dev)
1145 return false;
1146
1147 /* We don't perform any locking here as that would cause a lot of contention for VT-x/AMD-V. */
1148
1149 APICState *s = getLapic(dev); /** @todo fix interface */
1150
1151 /*
1152 * All our callbacks now come from single IOAPIC, thus locking
1153 * seems to be excessive now (@todo: check)
1154 */
1155 irrv = get_highest_priority_int(s->irr);
1156 if (irrv < 0)
1157 return false;
1158
1159 ppr = apic_get_ppr_zero_tpr(s);
1160
1161 if (ppr && (irrv & 0xf0) <= (ppr & 0xf0))
1162 return false;
1163
1164 return true;
1165}
1166
1167static void apic_update_tpr(APICDeviceInfo *dev, APICState* s, uint32_t val)
1168{
1169 bool fIrqIsActive = false;
1170 bool fIrqWasActive = false;
1171
1172 fIrqWasActive = apic_update_irq(dev, s);
1173 s->tpr = val;
1174 fIrqIsActive = apic_update_irq(dev, s);
1175
1176 /* If an interrupt is pending and now masked, then clear the FF flag. */
1177 if (fIrqWasActive && !fIrqIsActive)
1178 {
1179 Log(("apic_update_tpr: deactivate interrupt that was masked by the TPR update (%x)\n", val));
1180 STAM_COUNTER_INC(&dev->StatClearedActiveIrq);
1181 cpuClearInterrupt(dev, s);
1182 }
1183}
1184#endif
1185
1186static void apic_set_irq(APICDeviceInfo *dev, APICState* s, int vector_num, int trigger_mode)
1187{
1188 LogFlow(("CPU%d: apic_set_irq vector=%x, trigger_mode=%x\n", s->phys_id, vector_num, trigger_mode));
1189 set_bit(s->irr, vector_num);
1190 if (trigger_mode)
1191 set_bit(s->tmr, vector_num);
1192 else
1193 reset_bit(s->tmr, vector_num);
1194 apic_update_irq(dev, s);
1195}
1196
1197static void apic_eoi(APICDeviceInfo *dev, APICState* s)
1198{
1199 int isrv;
1200 isrv = get_highest_priority_int(s->isr);
1201 if (isrv < 0)
1202 return;
1203 reset_bit(s->isr, isrv);
1204 LogFlow(("CPU%d: apic_eoi isrv=%x\n", s->phys_id, isrv));
1205 /* XXX: send the EOI packet to the APIC bus to allow the I/O APIC to
1206 set the remote IRR bit for level triggered interrupts. */
1207 apic_update_irq(dev, s);
1208}
1209
1210#ifndef VBOX
1211static uint32_t apic_get_delivery_bitmask(uint8_t dest, uint8_t dest_mode)
1212#else /* VBOX */
1213static uint32_t apic_get_delivery_bitmask(APICDeviceInfo *dev, uint8_t dest, uint8_t dest_mode)
1214#endif /* VBOX */
1215{
1216 uint32_t mask = 0;
1217
1218 if (dest_mode == 0)
1219 {
1220 if (dest == 0xff)
1221 mask = 0xff;
1222 else
1223 mask = 1 << dest;
1224 }
1225 else
1226 {
1227 APICState *apic = dev->CTX_SUFF(paLapics);
1228 uint32_t i;
1229
1230 /* XXX: cluster mode */
1231 for(i = 0; i < dev->cCpus; i++)
1232 {
1233 if (apic->dest_mode == APIC_DESTMODE_FLAT)
1234 {
1235 if (dest & apic->log_dest)
1236 mask |= (1 << i);
1237 }
1238 else if (apic->dest_mode == APIC_DESTMODE_CLUSTER)
1239 {
1240 if ((dest & 0xf0) == (apic->log_dest & 0xf0)
1241 &&
1242 (dest & apic->log_dest & 0x0f))
1243 {
1244 mask |= (1 << i);
1245 }
1246 }
1247 apic++;
1248 }
1249 }
1250
1251 return mask;
1252}
1253
1254#ifdef IN_RING3
1255static void apic_init_ipi(APICDeviceInfo* dev, APICState *s)
1256{
1257 int i;
1258
1259 for(i = 0; i < APIC_LVT_NB; i++)
1260 s->lvt[i] = 1 << 16; /* mask LVT */
1261 s->tpr = 0;
1262 s->spurious_vec = 0xff;
1263 s->log_dest = 0;
1264 s->dest_mode = 0xff;
1265 memset(s->isr, 0, sizeof(s->isr));
1266 memset(s->tmr, 0, sizeof(s->tmr));
1267 memset(s->irr, 0, sizeof(s->irr));
1268 s->esr = 0;
1269 memset(s->icr, 0, sizeof(s->icr));
1270 s->divide_conf = 0;
1271 s->count_shift = 0;
1272 s->initial_count = 0;
1273 s->initial_count_load_time = 0;
1274 s->next_time = 0;
1275}
1276
1277
1278#ifdef VBOX
1279static void apicSendInitIpi(APICDeviceInfo* dev, APICState *s)
1280{
1281 apic_init_ipi(dev, s);
1282 cpuSendInitIpi(dev, s);
1283}
1284#endif
1285
1286/* send a SIPI message to the CPU to start it */
1287static void apic_startup(APICDeviceInfo* dev, APICState *s, int vector_num)
1288{
1289#ifndef VBOX
1290 CPUState *env = s->cpu_env;
1291 if (!env->halted)
1292 return;
1293 env->eip = 0;
1294 cpu_x86_load_seg_cache(env, R_CS, vector_num << 8, vector_num << 12,
1295 0xffff, 0);
1296 env->halted = 0;
1297#else
1298 Log(("[SMP] apic_startup: %d on CPUs %d\n", vector_num, s->phys_id));
1299 cpuSendSipi(dev, s, vector_num);
1300#endif
1301}
1302#endif /* IN_RING3 */
1303
1304static int apic_deliver(APICDeviceInfo* dev, APICState *s,
1305 uint8_t dest, uint8_t dest_mode,
1306 uint8_t delivery_mode, uint8_t vector_num,
1307 uint8_t polarity, uint8_t trigger_mode)
1308{
1309 uint32_t deliver_bitmask = 0;
1310 int dest_shorthand = (s->icr[0] >> 18) & 3;
1311#ifndef VBOX
1312 APICState *apic_iter;
1313#endif /* !VBOX */
1314
1315 LogFlow(("apic_deliver dest=%x dest_mode=%x dest_shorthand=%x delivery_mode=%x vector_num=%x polarity=%x trigger_mode=%x\n", dest, dest_mode, dest_shorthand, delivery_mode, vector_num, polarity, trigger_mode));
1316
1317 switch (dest_shorthand) {
1318 case 0:
1319#ifndef VBOX
1320 deliver_bitmask = apic_get_delivery_bitmask(dest, dest_mode);
1321#else /* VBOX */
1322 deliver_bitmask = apic_get_delivery_bitmask(dev, dest, dest_mode);
1323#endif /* !VBOX */
1324 break;
1325 case 1:
1326 deliver_bitmask = (1 << s->id);
1327 break;
1328 case 2:
1329 deliver_bitmask = 0xffffffff;
1330 break;
1331 case 3:
1332 deliver_bitmask = 0xffffffff & ~(1 << s->id);
1333 break;
1334 }
1335
1336 switch (delivery_mode) {
1337 case APIC_DM_INIT:
1338 {
1339 int trig_mode = (s->icr[0] >> 15) & 1;
1340 int level = (s->icr[0] >> 14) & 1;
1341 if (level == 0 && trig_mode == 1) {
1342 foreach_apic(dev, deliver_bitmask,
1343 apic->arb_id = apic->id);
1344#ifndef VBOX
1345 return;
1346#else
1347 Log(("CPU%d: APIC_DM_INIT arbitration id(s) set\n", s->phys_id));
1348 return VINF_SUCCESS;
1349#endif
1350 }
1351 }
1352 break;
1353
1354 case APIC_DM_SIPI:
1355#ifndef VBOX
1356 for (apic_iter = first_local_apic; apic_iter != NULL;
1357 apic_iter = apic_iter->next_apic) {
1358 if (deliver_bitmask & (1 << apic_iter->id)) {
1359 /* XXX: SMP support */
1360 /* apic_startup(apic_iter); */
1361 }
1362 }
1363 return;
1364#else
1365# ifdef IN_RING3
1366 foreach_apic(dev, deliver_bitmask,
1367 apic_startup(dev, apic, vector_num));
1368 return VINF_SUCCESS;
1369# else
1370 /* We shall send SIPI only in R3, R0 calls should be
1371 rescheduled to R3 */
1372 return VINF_IOM_HC_MMIO_WRITE;
1373# endif
1374#endif /* !VBOX */
1375 }
1376
1377#ifndef VBOX
1378 apic_bus_deliver(deliver_bitmask, delivery_mode, vector_num, polarity,
1379 trigger_mode);
1380#else /* VBOX */
1381 return apic_bus_deliver(dev, deliver_bitmask, delivery_mode, vector_num,
1382 polarity, trigger_mode);
1383#endif /* VBOX */
1384}
1385
1386
1387PDMBOTHCBDECL(int) apicGetInterrupt(PPDMDEVINS pDevIns)
1388{
1389 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
1390 /* if the APIC is not installed or enabled, we let the 8259 handle the
1391 IRQs */
1392 if (!dev)
1393 {
1394 Log(("apic_get_interrupt: returns -1 (!s)\n"));
1395 return -1;
1396 }
1397
1398 Assert(PDMCritSectIsOwner(dev->CTX_SUFF(pCritSect)));
1399
1400 APICState *s = getLapic(dev); /** @todo fix interface */
1401 int intno;
1402
1403 if (!(s->spurious_vec & APIC_SV_ENABLE)) {
1404 Log(("CPU%d: apic_get_interrupt: returns -1 (APIC_SV_ENABLE)\n", s->phys_id));
1405 return -1;
1406 }
1407
1408 /* XXX: spurious IRQ handling */
1409 intno = get_highest_priority_int(s->irr);
1410 if (intno < 0) {
1411 Log(("CPU%d: apic_get_interrupt: returns -1 (irr)\n", s->phys_id));
1412 return -1;
1413 }
1414 if (s->tpr && (uint32_t)intno <= s->tpr) {
1415 Log(("apic_get_interrupt: returns %d (sp)\n", s->spurious_vec & 0xff));
1416 return s->spurious_vec & 0xff;
1417 }
1418 reset_bit(s->irr, intno);
1419 set_bit(s->isr, intno);
1420 apic_update_irq(dev, s);
1421 LogFlow(("CPU%d: apic_get_interrupt: returns %d\n", s->phys_id, intno));
1422 return intno;
1423}
1424
1425static uint32_t apic_get_current_count(APICDeviceInfo *dev, APICState *s)
1426{
1427 int64_t d;
1428 uint32_t val;
1429#ifndef VBOX
1430 d = (qemu_get_clock(vm_clock) - s->initial_count_load_time) >>
1431 s->count_shift;
1432#else /* VBOX */
1433 d = (TMTimerGet(s->CTX_SUFF(pTimer)) - s->initial_count_load_time) >>
1434 s->count_shift;
1435#endif /* VBOX */
1436 if (s->lvt[APIC_LVT_TIMER] & APIC_LVT_TIMER_PERIODIC) {
1437 /* periodic */
1438 val = s->initial_count - (d % ((uint64_t)s->initial_count + 1));
1439 } else {
1440 if (d >= s->initial_count)
1441 val = 0;
1442 else
1443 val = s->initial_count - d;
1444 }
1445 return val;
1446}
1447
1448#ifndef VBOX /* we've replaced all the code working the APIC timer. */
1449
1450static void apic_timer_update(APICDeviceInfo* dev, APICState *s, int64_t current_time)
1451{
1452 int64_t next_time, d;
1453
1454 if (!(s->lvt[APIC_LVT_TIMER] & APIC_LVT_MASKED)) {
1455 d = (current_time - s->initial_count_load_time) >>
1456 s->count_shift;
1457 if (s->lvt[APIC_LVT_TIMER] & APIC_LVT_TIMER_PERIODIC) {
1458 d = ((d / ((uint64_t)s->initial_count + 1)) + 1) * ((uint64_t)s->initial_count + 1);
1459 } else {
1460 if (d >= s->initial_count)
1461 goto no_timer;
1462 d = (uint64_t)s->initial_count + 1;
1463 }
1464 next_time = s->initial_count_load_time + (d << s->count_shift);
1465# ifndef VBOX
1466 qemu_mod_timer(s->timer, next_time);
1467# else
1468 TMTimerSet(s->CTX_SUFF(pTimer), next_time);
1469 s->fTimerArmed = true;
1470# endif
1471 s->next_time = next_time;
1472 } else {
1473 no_timer:
1474# ifndef VBOX
1475 qemu_del_timer(s->timer);
1476# else
1477 TMTimerStop(s->CTX_SUFF(pTimer));
1478 s->fTimerArmed = false;
1479# endif
1480 }
1481}
1482
1483static void apic_timer(void *opaque)
1484{
1485 APICState *s = opaque;
1486
1487 if (!(s->lvt[APIC_LVT_TIMER] & APIC_LVT_MASKED)) {
1488 LogFlow(("apic_timer: trigger irq\n"));
1489 apic_set_irq(dev, s, s->lvt[APIC_LVT_TIMER] & 0xff, APIC_TRIGGER_EDGE);
1490 }
1491 apic_timer_update(dev, s, s->next_time);
1492}
1493
1494#else /* VBOX */
1495
1496/**
1497 * Implementation of the 0380h access: Timer reset + new initial count.
1498 *
1499 * @param dev The device state.
1500 * @param pThis The APIC sub-device state.
1501 * @param u32NewInitialCount The new initial count for the timer.
1502 */
1503static void apicTimerSetInitialCount(APICDeviceInfo *dev, APICState *pThis, uint32_t u32NewInitialCount)
1504{
1505 STAM_COUNTER_INC(&pThis->StatTimerSetInitialCount);
1506 pThis->initial_count = u32NewInitialCount;
1507
1508 /*
1509 * Don't (re-)arm the timer if the it's masked or if it's
1510 * a zero length one-shot timer.
1511 */
1512 /** @todo check the correct behavior of setting a 0 initial_count for a one-shot
1513 * timer. This is just copying the behavior of the original code. */
1514 if ( !(pThis->lvt[APIC_LVT_TIMER] & APIC_LVT_MASKED)
1515 && ( (pThis->lvt[APIC_LVT_TIMER] & APIC_LVT_TIMER_PERIODIC)
1516 || u32NewInitialCount != 0))
1517 {
1518 /*
1519 * Calculate the relative next time and perform a combined timer get/set
1520 * operation. This avoids racing the clock between get and set.
1521 */
1522 uint64_t cTicksNext = u32NewInitialCount;
1523 cTicksNext += 1;
1524 cTicksNext <<= pThis->count_shift;
1525 TMTimerSetRelative(pThis->CTX_SUFF(pTimer), cTicksNext, &pThis->initial_count_load_time);
1526 pThis->next_time = pThis->initial_count_load_time + cTicksNext;
1527 pThis->fTimerArmed = true;
1528 STAM_COUNTER_INC(&pThis->StatTimerSetInitialCountArm);
1529 }
1530 else
1531 {
1532 /* Stop it if necessary and record the load time for unmasking. */
1533 if (pThis->fTimerArmed)
1534 {
1535 STAM_COUNTER_INC(&pThis->StatTimerSetInitialCountDisarm);
1536 TMTimerStop(pThis->CTX_SUFF(pTimer));
1537 pThis->fTimerArmed = false;
1538 }
1539 pThis->initial_count_load_time = TMTimerGet(pThis->CTX_SUFF(pTimer));
1540 }
1541}
1542
1543/**
1544 * Implementation of the 0320h access: change the LVT flags.
1545 *
1546 * @param dev The device state.
1547 * @param pThis The APIC sub-device state to operate on.
1548 * @param fNew The new flags.
1549 */
1550static void apicTimerSetLvt(APICDeviceInfo *dev, APICState *pThis, uint32_t fNew)
1551{
1552 STAM_COUNTER_INC(&pThis->StatTimerSetLvt);
1553
1554 /*
1555 * Make the flag change, saving the old ones so we can avoid
1556 * unnecessary work.
1557 */
1558 uint32_t const fOld = pThis->lvt[APIC_LVT_TIMER];
1559 pThis->lvt[APIC_LVT_TIMER] = fNew;
1560
1561 /* Only the masked and peridic bits are relevant (see apic_timer_update). */
1562 if ( (fOld & (APIC_LVT_MASKED | APIC_LVT_TIMER_PERIODIC))
1563 != (fNew & (APIC_LVT_MASKED | APIC_LVT_TIMER_PERIODIC)))
1564 {
1565 /*
1566 * If changed to one-shot from periodic, stop the timer if we're not
1567 * in the first period.
1568 */
1569 /** @todo check how clearing the periodic flag really should behave when not
1570 * in period 1. The current code just mirrors the behavior of the
1571 * original implementation. */
1572 if ( (fOld & APIC_LVT_TIMER_PERIODIC)
1573 && !(fNew & APIC_LVT_TIMER_PERIODIC))
1574 {
1575 STAM_COUNTER_INC(&pThis->StatTimerSetLvtClearPeriodic);
1576 uint64_t cTicks = (pThis->next_time - pThis->initial_count_load_time) >> pThis->count_shift;
1577 if (cTicks >= pThis->initial_count)
1578 {
1579 /* not first period, stop it. */
1580 TMTimerStop(pThis->CTX_SUFF(pTimer));
1581 pThis->fTimerArmed = false;
1582 }
1583 /* else: first period, let it fire normally. */
1584 }
1585
1586 /*
1587 * We postpone stopping the timer when it's masked, this way we can
1588 * avoid some timer work when the guest temporarily masks the timer.
1589 * (apicTimerCallback will stop it if still masked.)
1590 */
1591 if (fNew & APIC_LVT_MASKED)
1592 STAM_COUNTER_INC(&pThis->StatTimerSetLvtPostponed);
1593 else if (pThis->fTimerArmed)
1594 STAM_COUNTER_INC(&pThis->StatTimerSetLvtArmed);
1595 /*
1596 * If unmasked and not armed, we have to rearm the timer so it will
1597 * fire at the end of the current period.
1598 * This is code is currently RACING the virtual sync clock!
1599 */
1600 else if (fOld & APIC_LVT_MASKED)
1601 {
1602 STAM_COUNTER_INC(&pThis->StatTimerSetLvtArm);
1603 for (unsigned cTries = 0; ; cTries++)
1604 {
1605 uint64_t NextTS;
1606 uint64_t cTicks = (TMTimerGet(pThis->CTX_SUFF(pTimer)) - pThis->initial_count_load_time) >> pThis->count_shift;
1607 if (fNew & APIC_LVT_TIMER_PERIODIC)
1608 NextTS = ((cTicks / ((uint64_t)pThis->initial_count + 1)) + 1) * ((uint64_t)pThis->initial_count + 1);
1609 else
1610 {
1611 if (cTicks >= pThis->initial_count)
1612 break;
1613 NextTS = (uint64_t)pThis->initial_count + 1;
1614 }
1615 NextTS <<= pThis->count_shift;
1616 NextTS += pThis->initial_count_load_time;
1617
1618 /* Try avoid the assertion in TM.cpp... this isn't perfect! */
1619 if ( NextTS > TMTimerGet(pThis->CTX_SUFF(pTimer))
1620 || cTries > 10)
1621 {
1622 TMTimerSet(pThis->CTX_SUFF(pTimer), NextTS);
1623 pThis->next_time = NextTS;
1624 pThis->fTimerArmed = true;
1625 break;
1626 }
1627 STAM_COUNTER_INC(&pThis->StatTimerSetLvtArmRetries);
1628 }
1629 }
1630 }
1631 else
1632 STAM_COUNTER_INC(&pThis->StatTimerSetLvtNoRelevantChange);
1633}
1634
1635# ifdef IN_RING3
1636/**
1637 * Timer callback function.
1638 *
1639 * @param pDevIns The device state.
1640 * @param pTimer The timer handle.
1641 * @param pvUser User argument pointing to the APIC instance.
1642 */
1643static DECLCALLBACK(void) apicTimerCallback(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1644{
1645 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
1646 APICState *pThis = (APICState *)pvUser;
1647 Assert(pThis->pTimerR3 == pTimer);
1648 Assert(pThis->fTimerArmed);
1649
1650 if (!(pThis->lvt[APIC_LVT_TIMER] & APIC_LVT_MASKED)) {
1651 LogFlow(("apic_timer: trigger irq\n"));
1652 apic_set_irq(dev, pThis, pThis->lvt[APIC_LVT_TIMER] & 0xff, APIC_TRIGGER_EDGE);
1653
1654 if (pThis->lvt[APIC_LVT_TIMER] & APIC_LVT_TIMER_PERIODIC) {
1655 /* new interval. */
1656 pThis->next_time += (((uint64_t)pThis->initial_count + 1) << pThis->count_shift);
1657 TMTimerSet(pThis->CTX_SUFF(pTimer), pThis->next_time);
1658 pThis->fTimerArmed = true;
1659 } else {
1660 /* single shot. */
1661 pThis->fTimerArmed = false;
1662 }
1663 } else {
1664 /* masked, do not rearm. */
1665 pThis->fTimerArmed = false;
1666 }
1667}
1668# endif /* IN_RING3 */
1669
1670#endif /* VBOX */
1671
1672#ifndef VBOX
1673static uint32_t apic_mem_readb(void *opaque, target_phys_addr_t addr)
1674{
1675 return 0;
1676}
1677static uint32_t apic_mem_readw(void *opaque, target_phys_addr_t addr)
1678{
1679 return 0;
1680}
1681
1682static void apic_mem_writeb(void *opaque, target_phys_addr_t addr, uint32_t val)
1683{
1684}
1685
1686static void apic_mem_writew(void *opaque, target_phys_addr_t addr, uint32_t val)
1687{
1688}
1689#endif /* !VBOX */
1690
1691
1692#ifndef VBOX
1693static uint32_t apic_mem_readl(void *opaque, target_phys_addr_t addr)
1694{
1695 CPUState *env;
1696 APICState *s;
1697#else /* VBOX */
1698static uint32_t apic_mem_readl(APICDeviceInfo* dev, APICState *s, target_phys_addr_t addr)
1699{
1700#endif /* VBOX */
1701 uint32_t val;
1702 int index;
1703
1704#ifndef VBOX
1705 env = cpu_single_env;
1706 if (!env)
1707 return 0;
1708 s = env->apic_state;
1709#endif /* !VBOX */
1710
1711 index = (addr >> 4) & 0xff;
1712 switch(index) {
1713 case 0x02: /* id */
1714 val = s->id << 24;
1715 break;
1716 case 0x03: /* version */
1717 val = APIC_HW_VERSION | ((APIC_LVT_NB - 1) << 16);
1718 break;
1719 case 0x08:
1720 val = s->tpr;
1721 break;
1722 case 0x09:
1723 val = apic_get_arb_pri(s);
1724 break;
1725 case 0x0a:
1726 /* ppr */
1727 val = apic_get_ppr(s);
1728 break;
1729 case 0x0b:
1730 Log(("apic_mem_readl %x %x -> write only returning 0\n", addr, index));
1731 val = 0;
1732 break;
1733 case 0x0d:
1734 val = s->log_dest << 24;
1735 break;
1736 case 0x0e:
1737#ifdef VBOX
1738 /* Bottom 28 bits are always 1 */
1739 val = (s->dest_mode << 28) | 0xfffffff;
1740#else
1741 val = s->dest_mode << 28;
1742#endif
1743 break;
1744 case 0x0f:
1745 val = s->spurious_vec;
1746 break;
1747#ifndef VBOX
1748 case 0x10 ... 0x17:
1749#else /* VBOX */
1750 case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
1751#endif /* VBOX */
1752 val = s->isr[index & 7];
1753 break;
1754#ifndef VBOX
1755 case 0x18 ... 0x1f:
1756#else /* VBOX */
1757 case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
1758#endif /* VBOX */
1759 val = s->tmr[index & 7];
1760 break;
1761#ifndef VBOX
1762 case 0x20 ... 0x27:
1763#else /* VBOX */
1764 case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
1765#endif /* VBOX */
1766 val = s->irr[index & 7];
1767 break;
1768 case 0x28:
1769 val = s->esr;
1770 break;
1771 case 0x30:
1772 case 0x31:
1773 val = s->icr[index & 1];
1774 break;
1775#ifndef VBOX
1776 case 0x32 ... 0x37:
1777#else /* VBOX */
1778 case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
1779#endif /* VBOX */
1780 val = s->lvt[index - 0x32];
1781 break;
1782 case 0x38:
1783 val = s->initial_count;
1784 break;
1785 case 0x39:
1786 val = apic_get_current_count(dev, s);
1787 break;
1788 case 0x3e:
1789 val = s->divide_conf;
1790 break;
1791 case 0x2f:
1792 /**
1793 * Correctable machine check exception vector, @todo: implement me!
1794 */
1795 default:
1796 AssertMsgFailed(("apic_mem_readl: unknown index %x\n", index));
1797 s->esr |= ESR_ILLEGAL_ADDRESS;
1798 val = 0;
1799 break;
1800 }
1801#ifdef DEBUG_APIC
1802 Log(("CPU%d: APIC read: %08x = %08x\n", s->phys_id, (uint32_t)addr, val));
1803#endif
1804 return val;
1805}
1806
1807#ifndef VBOX
1808static void apic_mem_writel(void *opaque, target_phys_addr_t addr, uint32_t val)
1809{
1810 CPUState *env;
1811 APICState *s;
1812#else /* VBOX */
1813static int apic_mem_writel(APICDeviceInfo* dev, APICState *s, target_phys_addr_t addr, uint32_t val)
1814{
1815 int rc = VINF_SUCCESS;
1816#endif /* VBOX */
1817 int index;
1818
1819#ifndef VBOX
1820 env = cpu_single_env;
1821 if (!env)
1822 return;
1823 s = env->apic_state;
1824#endif /* !VBOX */
1825
1826#ifdef DEBUG_APIC
1827 Log(("CPU%d: APIC write: %08x = %08x\n", s->phys_id, (uint32_t)addr, val));
1828#endif
1829
1830 index = (addr >> 4) & 0xff;
1831 switch(index) {
1832 case 0x02:
1833 s->id = (val >> 24);
1834 break;
1835 case 0x03:
1836 Log(("apic_mem_writel: write to version register; ignored\n"));
1837 break;
1838 case 0x08:
1839#ifdef VBOX
1840 apic_update_tpr(dev, s, val);
1841#else
1842 s->tpr = val;
1843 apic_update_irq(s);
1844#endif
1845 break;
1846 case 0x09:
1847 case 0x0a:
1848 Log(("apic_mem_writel: write to read-only register %d ignored\n", index));
1849 break;
1850 case 0x0b: /* EOI */
1851 apic_eoi(dev, s);
1852 break;
1853 case 0x0d:
1854 s->log_dest = val >> 24;
1855 break;
1856 case 0x0e:
1857 s->dest_mode = val >> 28;
1858 break;
1859 case 0x0f:
1860 s->spurious_vec = val & 0x1ff;
1861 apic_update_irq(dev, s);
1862 break;
1863#ifndef VBOX
1864 case 0x10 ... 0x17:
1865 case 0x18 ... 0x1f:
1866 case 0x20 ... 0x27:
1867 case 0x28:
1868#else
1869 case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
1870 case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
1871 case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
1872 case 0x28:
1873 Log(("apic_mem_writel: write to read-only register %d ignored\n", index));
1874#endif
1875 break;
1876
1877 case 0x30:
1878 s->icr[0] = val;
1879 rc = apic_deliver(dev, s, (s->icr[1] >> 24) & 0xff,
1880 (s->icr[0] >> 11) & 1,
1881 (s->icr[0] >> 8) & 7, (s->icr[0] & 0xff),
1882 (s->icr[0] >> 14) & 1, (s->icr[0] >> 15) & 1);
1883 break;
1884 case 0x31:
1885 s->icr[1] = val;
1886 break;
1887#ifndef VBOX
1888 case 0x32 ... 0x37:
1889#else /* VBOX */
1890 case 0x32 + APIC_LVT_TIMER:
1891 AssertCompile(APIC_LVT_TIMER == 0);
1892 apicTimerSetLvt(dev, s, val);
1893 break;
1894 case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
1895#endif /* VBOX */
1896 {
1897 int n = index - 0x32;
1898 s->lvt[n] = val;
1899#ifndef VBOX
1900 if (n == APIC_LVT_TIMER)
1901 apic_timer_update(s, qemu_get_clock(vm_clock));
1902#endif /* !VBOX*/
1903 }
1904 break;
1905 case 0x38:
1906#ifndef VBOX
1907 s->initial_count = val;
1908 s->initial_count_load_time = qemu_get_clock(vm_clock);
1909 apic_timer_update(dev, s, s->initial_count_load_time);
1910#else /* VBOX */
1911 apicTimerSetInitialCount(dev, s, val);
1912#endif /* VBOX*/
1913 break;
1914 case 0x39:
1915 Log(("apic_mem_writel: write to read-only register %d ignored\n", index));
1916 break;
1917 case 0x3e:
1918 {
1919 int v;
1920 s->divide_conf = val & 0xb;
1921 v = (s->divide_conf & 3) | ((s->divide_conf >> 1) & 4);
1922 s->count_shift = (v + 1) & 7;
1923 }
1924 break;
1925 default:
1926 AssertMsgFailed(("apic_mem_writel: unknown index %x\n", index));
1927 s->esr |= ESR_ILLEGAL_ADDRESS;
1928 break;
1929 }
1930#ifdef VBOX
1931 return rc;
1932#endif
1933}
1934
1935#ifdef IN_RING3
1936
1937static void apic_save(QEMUFile *f, void *opaque)
1938{
1939 APICState *s = (APICState*)opaque;
1940 int i;
1941
1942 qemu_put_be32s(f, &s->apicbase);
1943#ifdef VBOX
1944 qemu_put_be32s(f, &s->id);
1945 qemu_put_be32s(f, &s->phys_id);
1946 qemu_put_be32s(f, &s->arb_id);
1947 qemu_put_be32s(f, &s->tpr);
1948#else
1949 qemu_put_8s(f, &s->id);
1950 qemu_put_8s(f, &s->arb_id);
1951 qemu_put_8s(f, &s->tpr);
1952#endif
1953 qemu_put_be32s(f, &s->spurious_vec);
1954 qemu_put_8s(f, &s->log_dest);
1955 qemu_put_8s(f, &s->dest_mode);
1956 for (i = 0; i < 8; i++) {
1957 qemu_put_be32s(f, &s->isr[i]);
1958 qemu_put_be32s(f, &s->tmr[i]);
1959 qemu_put_be32s(f, &s->irr[i]);
1960 }
1961 for (i = 0; i < APIC_LVT_NB; i++) {
1962 qemu_put_be32s(f, &s->lvt[i]);
1963 }
1964 qemu_put_be32s(f, &s->esr);
1965 qemu_put_be32s(f, &s->icr[0]);
1966 qemu_put_be32s(f, &s->icr[1]);
1967 qemu_put_be32s(f, &s->divide_conf);
1968 qemu_put_be32s(f, &s->count_shift);
1969 qemu_put_be32s(f, &s->initial_count);
1970 qemu_put_be64s(f, &s->initial_count_load_time);
1971 qemu_put_be64s(f, &s->next_time);
1972
1973#ifdef VBOX
1974 TMR3TimerSave(s->CTX_SUFF(pTimer), f);
1975#endif
1976}
1977
1978static int apic_load(QEMUFile *f, void *opaque, int version_id)
1979{
1980 APICState *s = (APICState*)opaque;
1981 int i;
1982
1983#ifdef VBOX
1984 /* XXX: what if the base changes? (registered memory regions) */
1985 qemu_get_be32s(f, &s->apicbase);
1986
1987 switch (version_id)
1988 {
1989 case APIC_SAVED_STATE_VERSION_ANCIENT:
1990 {
1991 uint8_t val = 0;
1992 qemu_get_8s(f, &val);
1993 s->id = val;
1994 /* UP only in old saved states */
1995 s->phys_id = 0;
1996 qemu_get_8s(f, &val);
1997 s->arb_id = val;
1998 break;
1999 }
2000 case APIC_SAVED_STATE_VERSION:
2001 case APIC_SAVED_STATE_VERSION_VBOX_30:
2002 qemu_get_be32s(f, &s->id);
2003 qemu_get_be32s(f, &s->phys_id);
2004 qemu_get_be32s(f, &s->arb_id);
2005 break;
2006 default:
2007 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
2008 }
2009 qemu_get_be32s(f, &s->tpr);
2010#else
2011 if (version_id != 1)
2012 return -EINVAL;
2013
2014 /* XXX: what if the base changes? (registered memory regions) */
2015 qemu_get_be32s(f, &s->apicbase);
2016 qemu_get_8s(f, &s->id);
2017 qemu_get_8s(f, &s->arb_id);
2018 qemu_get_8s(f, &s->tpr);
2019#endif
2020 qemu_get_be32s(f, &s->spurious_vec);
2021 qemu_get_8s(f, &s->log_dest);
2022 qemu_get_8s(f, &s->dest_mode);
2023 for (i = 0; i < 8; i++) {
2024 qemu_get_be32s(f, &s->isr[i]);
2025 qemu_get_be32s(f, &s->tmr[i]);
2026 qemu_get_be32s(f, &s->irr[i]);
2027 }
2028 for (i = 0; i < APIC_LVT_NB; i++) {
2029 qemu_get_be32s(f, &s->lvt[i]);
2030 }
2031 qemu_get_be32s(f, &s->esr);
2032 qemu_get_be32s(f, &s->icr[0]);
2033 qemu_get_be32s(f, &s->icr[1]);
2034 qemu_get_be32s(f, &s->divide_conf);
2035 qemu_get_be32s(f, (uint32_t *)&s->count_shift);
2036 qemu_get_be32s(f, (uint32_t *)&s->initial_count);
2037 qemu_get_be64s(f, (uint64_t *)&s->initial_count_load_time);
2038 qemu_get_be64s(f, (uint64_t *)&s->next_time);
2039
2040#ifdef VBOX
2041 int rc = TMR3TimerLoad(s->CTX_SUFF(pTimer), f);
2042 s->fTimerArmed = TMTimerIsActive(s->CTX_SUFF(pTimer));
2043#endif
2044
2045 return VINF_SUCCESS; /** @todo darn mess! */
2046}
2047#ifndef VBOX
2048static void apic_reset(void *opaque)
2049{
2050 APICState *s = (APICState*)opaque;
2051 apic_init_ipi(s);
2052}
2053#endif
2054
2055#endif /* IN_RING3 */
2056
2057#ifndef VBOX
2058static CPUReadMemoryFunc *apic_mem_read[3] = {
2059 apic_mem_readb,
2060 apic_mem_readw,
2061 apic_mem_readl,
2062};
2063
2064static CPUWriteMemoryFunc *apic_mem_write[3] = {
2065 apic_mem_writeb,
2066 apic_mem_writew,
2067 apic_mem_writel,
2068};
2069
2070int apic_init(CPUState *env)
2071{
2072 APICState *s;
2073
2074 s = qemu_mallocz(sizeof(APICState));
2075 if (!s)
2076 return -1;
2077 env->apic_state = s;
2078 apic_init_ipi(s);
2079 s->id = last_apic_id++;
2080 s->cpu_env = env;
2081 s->apicbase = 0xfee00000 |
2082 (s->id ? 0 : MSR_IA32_APICBASE_BSP) | MSR_IA32_APICBASE_ENABLE;
2083
2084 /* XXX: mapping more APICs at the same memory location */
2085 if (apic_io_memory == 0) {
2086 /* NOTE: the APIC is directly connected to the CPU - it is not
2087 on the global memory bus. */
2088 apic_io_memory = cpu_register_io_memory(0, apic_mem_read,
2089 apic_mem_write, NULL);
2090 cpu_register_physical_memory(s->apicbase & ~0xfff, 0x1000,
2091 apic_io_memory);
2092 }
2093 s->timer = qemu_new_timer(vm_clock, apic_timer, s);
2094
2095 register_savevm("apic", 0, 1, apic_save, apic_load, s);
2096 qemu_register_reset(apic_reset, s);
2097
2098 s->next_apic = first_local_apic;
2099 first_local_apic = s;
2100
2101 return 0;
2102}
2103#endif /* !VBOX */
2104
2105static void ioapic_service(IOAPICState *s)
2106{
2107 uint8_t i;
2108 uint8_t trig_mode;
2109 uint8_t vector;
2110 uint8_t delivery_mode;
2111 uint32_t mask;
2112 uint64_t entry;
2113 uint8_t dest;
2114 uint8_t dest_mode;
2115 uint8_t polarity;
2116
2117 for (i = 0; i < IOAPIC_NUM_PINS; i++) {
2118 mask = 1 << i;
2119 if (s->irr & mask) {
2120 entry = s->ioredtbl[i];
2121 if (!(entry & APIC_LVT_MASKED)) {
2122 trig_mode = ((entry >> 15) & 1);
2123 dest = entry >> 56;
2124 dest_mode = (entry >> 11) & 1;
2125 delivery_mode = (entry >> 8) & 7;
2126 polarity = (entry >> 13) & 1;
2127 if (trig_mode == APIC_TRIGGER_EDGE)
2128 s->irr &= ~mask;
2129 if (delivery_mode == APIC_DM_EXTINT)
2130#ifndef VBOX /* malc: i'm still not so sure about ExtINT delivery */
2131 vector = pic_read_irq(isa_pic);
2132#else /* VBOX */
2133 {
2134 AssertMsgFailed(("Delivery mode ExtINT"));
2135 vector = 0xff; /* incorrect but shuts up gcc. */
2136 }
2137#endif /* VBOX */
2138 else
2139 vector = entry & 0xff;
2140
2141#ifndef VBOX
2142 apic_bus_deliver(apic_get_delivery_bitmask(dest, dest_mode),
2143 delivery_mode, vector, polarity, trig_mode);
2144#else /* VBOX */
2145 int rc = s->CTX_SUFF(pIoApicHlp)->pfnApicBusDeliver(s->CTX_SUFF(pDevIns),
2146 dest,
2147 dest_mode,
2148 delivery_mode,
2149 vector,
2150 polarity,
2151 trig_mode);
2152 /* We must be sure that attempts to reschedule in R3
2153 never get here */
2154 Assert(rc == VINF_SUCCESS);
2155#endif /* VBOX */
2156 }
2157 }
2158 }
2159}
2160
2161#ifdef VBOX
2162static
2163#endif
2164void ioapic_set_irq(void *opaque, int vector, int level)
2165{
2166 IOAPICState *s = (IOAPICState*)opaque;
2167
2168 if (vector >= 0 && vector < IOAPIC_NUM_PINS) {
2169 uint32_t mask = 1 << vector;
2170 uint64_t entry = s->ioredtbl[vector];
2171
2172 if ((entry >> 15) & 1) {
2173 /* level triggered */
2174 if (level) {
2175 s->irr |= mask;
2176 ioapic_service(s);
2177#ifdef VBOX
2178 if ((level & PDM_IRQ_LEVEL_FLIP_FLOP) == PDM_IRQ_LEVEL_FLIP_FLOP) {
2179 s->irr &= ~mask;
2180 }
2181#endif
2182 } else {
2183 s->irr &= ~mask;
2184 }
2185 } else {
2186 /* edge triggered */
2187 if (level) {
2188 s->irr |= mask;
2189 ioapic_service(s);
2190 }
2191 }
2192 }
2193}
2194
2195static uint32_t ioapic_mem_readl(void *opaque, target_phys_addr_t addr)
2196{
2197 IOAPICState *s = (IOAPICState*)opaque;
2198 int index;
2199 uint32_t val = 0;
2200
2201 addr &= 0xff;
2202 if (addr == 0x00) {
2203 val = s->ioregsel;
2204 } else if (addr == 0x10) {
2205 switch (s->ioregsel) {
2206 case 0x00:
2207 val = s->id << 24;
2208 break;
2209 case 0x01:
2210 val = 0x11 | ((IOAPIC_NUM_PINS - 1) << 16); /* version 0x11 */
2211 break;
2212 case 0x02:
2213 val = 0;
2214 break;
2215 default:
2216 index = (s->ioregsel - 0x10) >> 1;
2217 if (index >= 0 && index < IOAPIC_NUM_PINS) {
2218 if (s->ioregsel & 1)
2219 val = s->ioredtbl[index] >> 32;
2220 else
2221 val = s->ioredtbl[index] & 0xffffffff;
2222 }
2223 }
2224#ifdef DEBUG_IOAPIC
2225 Log(("I/O APIC read: %08x = %08x\n", s->ioregsel, val));
2226#endif
2227 }
2228 return val;
2229}
2230
2231static void ioapic_mem_writel(void *opaque, target_phys_addr_t addr, uint32_t val)
2232{
2233 IOAPICState *s = (IOAPICState*)opaque;
2234 int index;
2235
2236 addr &= 0xff;
2237 if (addr == 0x00) {
2238 s->ioregsel = val;
2239 return;
2240 } else if (addr == 0x10) {
2241#ifdef DEBUG_IOAPIC
2242 Log(("I/O APIC write: %08x = %08x\n", s->ioregsel, val));
2243#endif
2244 switch (s->ioregsel) {
2245 case 0x00:
2246 s->id = (val >> 24) & 0xff;
2247 return;
2248 case 0x01:
2249 case 0x02:
2250 return;
2251 default:
2252 index = (s->ioregsel - 0x10) >> 1;
2253 if (index >= 0 && index < IOAPIC_NUM_PINS) {
2254 if (s->ioregsel & 1) {
2255 s->ioredtbl[index] &= 0xffffffff;
2256 s->ioredtbl[index] |= (uint64_t)val << 32;
2257 } else {
2258#ifdef VBOX
2259 /* According to IOAPIC spec, vectors should be from 0x10 to 0xfe */
2260 uint8_t vec = val & 0xff;
2261 if ((val & APIC_LVT_MASKED) ||
2262 ((vec >= 0x10) && (vec < 0xff)))
2263 {
2264 s->ioredtbl[index] &= ~0xffffffffULL;
2265 s->ioredtbl[index] |= val;
2266 }
2267 else
2268 {
2269 /*
2270 * Linux 2.6 kernels has pretty strange function
2271 * unlock_ExtINT_logic() which writes
2272 * absolutely bogus (all 0) value into the vector
2273 * with pretty vague explanation why.
2274 * So we just ignore such writes.
2275 */
2276 LogRel(("IOAPIC GUEST BUG: bad vector writing %x(sel=%x) to %d\n", val, s->ioregsel, index));
2277 }
2278 }
2279#else
2280 s->ioredtbl[index] &= ~0xffffffffULL;
2281 s->ioredtbl[index] |= val;
2282#endif
2283 ioapic_service(s);
2284 }
2285 }
2286 }
2287}
2288
2289#ifdef IN_RING3
2290
2291static void ioapic_save(QEMUFile *f, void *opaque)
2292{
2293 IOAPICState *s = (IOAPICState*)opaque;
2294 int i;
2295
2296 qemu_put_8s(f, &s->id);
2297 qemu_put_8s(f, &s->ioregsel);
2298 for (i = 0; i < IOAPIC_NUM_PINS; i++) {
2299 qemu_put_be64s(f, &s->ioredtbl[i]);
2300 }
2301}
2302
2303static int ioapic_load(QEMUFile *f, void *opaque, int version_id)
2304{
2305 IOAPICState *s = (IOAPICState*)opaque;
2306 int i;
2307
2308 if (version_id != 1)
2309 return -EINVAL;
2310
2311 qemu_get_8s(f, &s->id);
2312 qemu_get_8s(f, &s->ioregsel);
2313 for (i = 0; i < IOAPIC_NUM_PINS; i++) {
2314 qemu_get_be64s(f, &s->ioredtbl[i]);
2315 }
2316 return 0;
2317}
2318
2319static void ioapic_reset(void *opaque)
2320{
2321 IOAPICState *s = (IOAPICState*)opaque;
2322#ifdef VBOX
2323 PPDMDEVINSR3 pDevIns = s->pDevInsR3;
2324 PCPDMIOAPICHLPR3 pIoApicHlp = s->pIoApicHlpR3;
2325#endif
2326 int i;
2327
2328 memset(s, 0, sizeof(*s));
2329 for(i = 0; i < IOAPIC_NUM_PINS; i++)
2330 s->ioredtbl[i] = 1 << 16; /* mask LVT */
2331
2332#ifdef VBOX
2333 if (pDevIns)
2334 {
2335 s->pDevInsR3 = pDevIns;
2336 s->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2337 s->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2338 }
2339 if (pIoApicHlp)
2340 {
2341 s->pIoApicHlpR3 = pIoApicHlp;
2342 s->pIoApicHlpRC = s->pIoApicHlpR3->pfnGetRCHelpers(pDevIns);
2343 s->pIoApicHlpR0 = s->pIoApicHlpR3->pfnGetR0Helpers(pDevIns);
2344 }
2345#endif
2346}
2347
2348#endif /* IN_RING3 */
2349
2350#ifndef VBOX
2351static CPUReadMemoryFunc *ioapic_mem_read[3] = {
2352 ioapic_mem_readl,
2353 ioapic_mem_readl,
2354 ioapic_mem_readl,
2355};
2356
2357static CPUWriteMemoryFunc *ioapic_mem_write[3] = {
2358 ioapic_mem_writel,
2359 ioapic_mem_writel,
2360 ioapic_mem_writel,
2361};
2362
2363IOAPICState *ioapic_init(void)
2364{
2365 IOAPICState *s;
2366 int io_memory;
2367
2368 s = qemu_mallocz(sizeof(IOAPICState));
2369 if (!s)
2370 return NULL;
2371 ioapic_reset(s);
2372 s->id = last_apic_id++;
2373
2374 io_memory = cpu_register_io_memory(0, ioapic_mem_read,
2375 ioapic_mem_write, s);
2376 cpu_register_physical_memory(0xfec00000, 0x1000, io_memory);
2377
2378 register_savevm("ioapic", 0, 1, ioapic_save, ioapic_load, s);
2379 qemu_register_reset(ioapic_reset, s);
2380
2381 return s;
2382}
2383#endif /* !VBOX */
2384
2385/* LAPIC */
2386PDMBOTHCBDECL(int) apicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
2387{
2388 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2389 APICState *s = getLapic(dev);
2390
2391 Log(("CPU%d: apicMMIORead at %llx\n", s->phys_id, (uint64_t)GCPhysAddr));
2392
2393 /** @todo: add LAPIC range validity checks (different LAPICs can theoretically have
2394 different physical addresses, see #3092) */
2395
2396 STAM_COUNTER_INC(&CTXSUFF(dev->StatMMIORead));
2397 switch (cb)
2398 {
2399 case 1:
2400 *(uint8_t *)pv = 0;
2401 break;
2402
2403 case 2:
2404 *(uint16_t *)pv = 0;
2405 break;
2406
2407 case 4:
2408 {
2409#if 0 /** @note experimental */
2410#ifndef IN_RING3
2411 uint32_t index = (GCPhysAddr >> 4) & 0xff;
2412
2413 if ( index == 0x08 /* TPR */
2414 && ++s->cTPRPatchAttempts < APIC_MAX_PATCH_ATTEMPTS)
2415 {
2416#ifdef IN_RC
2417 pDevIns->pDevHlpGC->pfnPATMSetMMIOPatchInfo(pDevIns, GCPhysAddr, &s->tpr);
2418#else
2419 RTGCPTR pDevInsGC = PDMINS2DATA_GCPTR(pDevIns);
2420 pDevIns->pHlpR0->pfnPATMSetMMIOPatchInfo(pDevIns, GCPhysAddr, pDevIns + RT_OFFSETOF(APICState, tpr));
2421#endif
2422 return VINF_PATM_HC_MMIO_PATCH_READ;
2423 }
2424#endif
2425#endif /* experimental */
2426 APIC_LOCK(dev, VINF_IOM_HC_MMIO_READ);
2427 *(uint32_t *)pv = apic_mem_readl(dev, s, GCPhysAddr);
2428 APIC_UNLOCK(dev);
2429 break;
2430 }
2431 default:
2432 AssertReleaseMsgFailed(("cb=%d\n", cb)); /* for now we assume simple accesses. */
2433 return VERR_INTERNAL_ERROR;
2434 }
2435 return VINF_SUCCESS;
2436}
2437
2438PDMBOTHCBDECL(int) apicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
2439{
2440 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2441 APICState *s = getLapic(dev);
2442
2443 Log(("CPU%d: apicMMIOWrite at %llx\n", s->phys_id, (uint64_t)GCPhysAddr));
2444
2445 /** @todo: add LAPIC range validity checks (multiple LAPICs can theoretically have
2446 different physical addresses, see #3092) */
2447
2448 STAM_COUNTER_INC(&CTXSUFF(dev->StatMMIOWrite));
2449 switch (cb)
2450 {
2451 case 1:
2452 case 2:
2453 /* ignore */
2454 break;
2455
2456 case 4:
2457 {
2458 int rc;
2459 APIC_LOCK(dev, VINF_IOM_HC_MMIO_WRITE);
2460 rc = apic_mem_writel(dev, s, GCPhysAddr, *(uint32_t *)pv);
2461 APIC_UNLOCK(dev);
2462 return rc;
2463 }
2464
2465 default:
2466 AssertReleaseMsgFailed(("cb=%d\n", cb)); /* for now we assume simple accesses. */
2467 return VERR_INTERNAL_ERROR;
2468 }
2469 return VINF_SUCCESS;
2470}
2471
2472#ifdef IN_RING3
2473
2474/* Print a 8-dword LAPIC bit map (256 bits). */
2475static void lapicDumpVec(APICDeviceInfo *dev, APICState *lapic, PCDBGFINFOHLP pHlp, unsigned start)
2476{
2477 unsigned i;
2478 uint32_t val;
2479
2480 for (i = 0; i < 8; ++i)
2481 {
2482 val = apic_mem_readl(dev, lapic, start + (i << 4));
2483 pHlp->pfnPrintf(pHlp, "%08X", val);
2484 }
2485 pHlp->pfnPrintf(pHlp, "\n");
2486}
2487
2488/* Print basic LAPIC state. */
2489static DECLCALLBACK(void) lapicInfoBasic(APICDeviceInfo *dev, APICState *lapic, PCDBGFINFOHLP pHlp)
2490{
2491 uint32_t val;
2492 unsigned max_lvt;
2493
2494 pHlp->pfnPrintf(pHlp, "Local APIC at %08X:\n", lapic->apicbase);
2495 val = apic_mem_readl(dev, lapic, 0x20);
2496 pHlp->pfnPrintf(pHlp, " LAPIC ID : %08X\n", val);
2497 pHlp->pfnPrintf(pHlp, " APIC ID = %02X\n", (val >> 24) & 0xff);
2498 val = apic_mem_readl(dev, lapic, 0x30);
2499 max_lvt = (val >> 16) & 0xff;
2500 pHlp->pfnPrintf(pHlp, " APIC VER : %08X\n", val);
2501 pHlp->pfnPrintf(pHlp, " version = %02X\n", val & 0xff);
2502 pHlp->pfnPrintf(pHlp, " lvts = %d\n", ((val >> 16) & 0xff) + 1);
2503 val = apic_mem_readl(dev, lapic, 0x80);
2504 pHlp->pfnPrintf(pHlp, " TPR : %08X\n", val);
2505 pHlp->pfnPrintf(pHlp, " task pri = %d/%d\n", (val >> 4) & 0xf, val & 0xf);
2506 val = apic_mem_readl(dev, lapic, 0xA0);
2507 pHlp->pfnPrintf(pHlp, " PPR : %08X\n", val);
2508 pHlp->pfnPrintf(pHlp, " cpu pri = %d/%d\n", (val >> 4) & 0xf, val & 0xf);
2509 val = apic_mem_readl(dev, lapic, 0xD0);
2510 pHlp->pfnPrintf(pHlp, " LDR : %08X\n", val);
2511 pHlp->pfnPrintf(pHlp, " log id = %02X\n", (val >> 24) & 0xff);
2512 val = apic_mem_readl(dev, lapic, 0xE0);
2513 pHlp->pfnPrintf(pHlp, " DFR : %08X\n", val);
2514 val = apic_mem_readl(dev, lapic, 0xF0);
2515 pHlp->pfnPrintf(pHlp, " SVR : %08X\n", val);
2516 pHlp->pfnPrintf(pHlp, " focus = %s\n", val & (1 << 9) ? "check off" : "check on");
2517 pHlp->pfnPrintf(pHlp, " lapic = %s\n", val & (1 << 8) ? "ENABLED" : "DISABLED");
2518 pHlp->pfnPrintf(pHlp, " vector = %02X\n", val & 0xff);
2519 pHlp->pfnPrintf(pHlp, " ISR : ");
2520 lapicDumpVec(dev, lapic, pHlp, 0x100);
2521 val = get_highest_priority_int(lapic->isr);
2522 pHlp->pfnPrintf(pHlp, " highest = %02X\n", val == ~0U ? 0 : val);
2523 pHlp->pfnPrintf(pHlp, " IRR : ");
2524 lapicDumpVec(dev, lapic, pHlp, 0x200);
2525 val = get_highest_priority_int(lapic->irr);
2526 pHlp->pfnPrintf(pHlp, " highest = %02X\n", val == ~0U ? 0 : val);
2527 val = apic_mem_readl(dev, lapic, 0x320);
2528}
2529
2530/* Print the more interesting LAPIC LVT entries. */
2531static DECLCALLBACK(void) lapicInfoLVT(APICDeviceInfo *dev, APICState *lapic, PCDBGFINFOHLP pHlp)
2532{
2533 uint32_t val;
2534 static const char *dmodes[] = { "Fixed ", "Reserved", "SMI", "Reserved",
2535 "NMI", "INIT", "Reserved", "ExtINT" };
2536
2537 val = apic_mem_readl(dev, lapic, 0x320);
2538 pHlp->pfnPrintf(pHlp, " LVT Timer : %08X\n", val);
2539 pHlp->pfnPrintf(pHlp, " mode = %s\n", val & (1 << 17) ? "periodic" : "one-shot");
2540 pHlp->pfnPrintf(pHlp, " mask = %d\n", (val >> 16) & 1);
2541 pHlp->pfnPrintf(pHlp, " status = %s\n", val & (1 << 12) ? "pending" : "idle");
2542 pHlp->pfnPrintf(pHlp, " vector = %02X\n", val & 0xff);
2543 val = apic_mem_readl(dev, lapic, 0x350);
2544 pHlp->pfnPrintf(pHlp, " LVT LINT0 : %08X\n", val);
2545 pHlp->pfnPrintf(pHlp, " mask = %d\n", (val >> 16) & 1);
2546 pHlp->pfnPrintf(pHlp, " trigger = %s\n", val & (1 << 15) ? "level" : "edge");
2547 pHlp->pfnPrintf(pHlp, " rem irr = %d\n", (val >> 14) & 1);
2548 pHlp->pfnPrintf(pHlp, " polarty = %d\n", (val >> 13) & 1);
2549 pHlp->pfnPrintf(pHlp, " status = %s\n", val & (1 << 12) ? "pending" : "idle");
2550 pHlp->pfnPrintf(pHlp, " delivry = %s\n", dmodes[(val >> 8) & 7]);
2551 pHlp->pfnPrintf(pHlp, " vector = %02X\n", val & 0xff);
2552 val = apic_mem_readl(dev, lapic, 0x360);
2553 pHlp->pfnPrintf(pHlp, " LVT LINT1 : %08X\n", val);
2554 pHlp->pfnPrintf(pHlp, " mask = %d\n", (val >> 16) & 1);
2555 pHlp->pfnPrintf(pHlp, " trigger = %s\n", val & (1 << 15) ? "level" : "edge");
2556 pHlp->pfnPrintf(pHlp, " rem irr = %d\n", (val >> 14) & 1);
2557 pHlp->pfnPrintf(pHlp, " polarty = %d\n", (val >> 13) & 1);
2558 pHlp->pfnPrintf(pHlp, " status = %s\n", val & (1 << 12) ? "pending" : "idle");
2559 pHlp->pfnPrintf(pHlp, " delivry = %s\n", dmodes[(val >> 8) & 7]);
2560 pHlp->pfnPrintf(pHlp, " vector = %02X\n", val & 0xff);
2561}
2562
2563/* Print LAPIC timer state. */
2564static DECLCALLBACK(void) lapicInfoTimer(APICDeviceInfo *dev, APICState *lapic, PCDBGFINFOHLP pHlp)
2565{
2566 uint32_t val;
2567 unsigned divider;
2568
2569 pHlp->pfnPrintf(pHlp, "Local APIC timer:\n");
2570 val = apic_mem_readl(dev, lapic, 0x380);
2571 pHlp->pfnPrintf(pHlp, " Initial count : %08X\n", val);
2572 val = apic_mem_readl(dev, lapic, 0x390);
2573 pHlp->pfnPrintf(pHlp, " Current count : %08X\n", val);
2574 val = apic_mem_readl(dev, lapic, 0x3E0);
2575 pHlp->pfnPrintf(pHlp, " Divide config : %08X\n", val);
2576 divider = ((val >> 1) & 0x04) | (val & 0x03);
2577 pHlp->pfnPrintf(pHlp, " divider = %d\n", divider == 7 ? 1 : 2 << divider);
2578}
2579
2580/**
2581 * Info handler, device version. Dumps Local APIC(s) state according to given argument.
2582 *
2583 * @param pDevIns Device instance which registered the info.
2584 * @param pHlp Callback functions for doing output.
2585 * @param pszArgs Argument string. Optional.
2586 */
2587static DECLCALLBACK(void) lapicInfo(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
2588{
2589 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2590 APICState *lapic;
2591
2592 lapic = getLapic(dev);
2593
2594 if (pszArgs == NULL || !strcmp(pszArgs, "basic"))
2595 {
2596 lapicInfoBasic(dev, lapic, pHlp);
2597 }
2598 else if (!strcmp(pszArgs, "lvt"))
2599 {
2600 lapicInfoLVT(dev, lapic, pHlp);
2601 }
2602 else if (!strcmp(pszArgs, "timer"))
2603 {
2604 lapicInfoTimer(dev, lapic, pHlp);
2605 }
2606 else
2607 {
2608 pHlp->pfnPrintf(pHlp, "Invalid argument. Recognized arguments are 'basic', 'lvt', 'timer'.\n");
2609 }
2610}
2611
2612/**
2613 * @copydoc FNSSMDEVLIVEEXEC
2614 */
2615static DECLCALLBACK(int) apicLiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
2616{
2617 APICDeviceInfo *pThis = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2618
2619 SSMR3PutU32( pSSM, pThis->cCpus);
2620 SSMR3PutBool(pSSM, pThis->fIoApic);
2621 SSMR3PutU32( pSSM, pThis->enmVersion);
2622 AssertCompile(PDMAPICVERSION_APIC == 2);
2623
2624 return VINF_SSM_DONT_CALL_AGAIN;
2625}
2626
2627/**
2628 * @copydoc FNSSMDEVSAVEEXEC
2629 */
2630static DECLCALLBACK(int) apicSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
2631{
2632 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2633
2634 /* config */
2635 apicLiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
2636
2637 /* save all APICs data, @todo: is it correct? */
2638 foreach_apic(dev, 0xffffffff, apic_save(pSSM, apic));
2639
2640 return VINF_SUCCESS;
2641}
2642
2643/**
2644 * @copydoc FNSSMDEVLOADEXEC
2645 */
2646static DECLCALLBACK(int) apicLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
2647{
2648 APICDeviceInfo *pThis = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2649
2650 if ( uVersion != APIC_SAVED_STATE_VERSION
2651 && uVersion != APIC_SAVED_STATE_VERSION_VBOX_30
2652 && uVersion != APIC_SAVED_STATE_VERSION_ANCIENT)
2653 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
2654
2655 /* config */
2656 if (uVersion > APIC_SAVED_STATE_VERSION_VBOX_30) {
2657 uint32_t cCpus;
2658 int rc = SSMR3GetU32(pSSM, &cCpus); AssertRCReturn(rc, rc);
2659 if (cCpus != pThis->cCpus)
2660 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - cCpus: saved=%#x config=%#x"), cCpus, pThis->cCpus);
2661 bool fIoApic;
2662 rc = SSMR3GetBool(pSSM, &fIoApic); AssertRCReturn(rc, rc);
2663 if (fIoApic != pThis->fIoApic)
2664 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - fIoApic: saved=%RTbool config=%RTbool"), fIoApic, pThis->fIoApic);
2665 uint32_t uApicVersion;
2666 rc = SSMR3GetU32(pSSM, &uApicVersion); AssertRCReturn(rc, rc);
2667 if (uApicVersion != (uint32_t)pThis->enmVersion)
2668 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - uApicVersion: saved=%#x config=%#x"), uApicVersion, pThis->enmVersion);
2669 }
2670
2671 if (uPass != SSM_PASS_FINAL)
2672 return VINF_SUCCESS;
2673
2674 /* load all APICs data */ /** @todo: is it correct? */
2675 foreach_apic(pThis, 0xffffffff,
2676 if (apic_load(pSSM, apic, uVersion)) {
2677 AssertFailed();
2678 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
2679 }
2680 );
2681 return VINF_SUCCESS;
2682}
2683
2684/**
2685 * @copydoc FNPDMDEVRESET
2686 */
2687static DECLCALLBACK(void) apicReset(PPDMDEVINS pDevIns)
2688{
2689 APICDeviceInfo *dev = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2690 unsigned i;
2691
2692 APIC_LOCK_VOID(dev, VERR_INTERNAL_ERROR);
2693
2694 /* Reset all APICs. */
2695 for (i = 0; i < dev->cCpus; i++) {
2696 APICState *pApic = &dev->CTX_SUFF(paLapics)[i];
2697 TMTimerStop(pApic->CTX_SUFF(pTimer));
2698
2699 /* Clear LAPIC state as if an INIT IPI was sent. */
2700 apic_init_ipi(dev, pApic);
2701 /* The IDs are not touched by apic_init_ipi() and must be reset now. */
2702 pApic->arb_id = pApic->id = i;
2703 Assert(pApic->id == pApic->phys_id); /* The two should match again. */
2704 /* Reset should re-enable the APIC. */
2705 pApic->apicbase = 0xfee00000 | MSR_IA32_APICBASE_ENABLE;
2706 if (pApic->phys_id == 0)
2707 pApic->apicbase |= MSR_IA32_APICBASE_BSP;
2708
2709 /* Clear any pending APIC interrupt action flag. */
2710 cpuClearInterrupt(dev, pApic);
2711 }
2712 /** @todo r=bird: Why is this done everytime, while the constructor first
2713 * checks the CPUID? Who is right? */
2714 dev->pApicHlpR3->pfnChangeFeature(dev->pDevInsR3, dev->enmVersion);
2715
2716 APIC_UNLOCK(dev);
2717}
2718
2719/**
2720 * @copydoc FNPDMDEVRELOCATE
2721 */
2722static DECLCALLBACK(void) apicRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
2723{
2724 APICDeviceInfo *pThis = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2725 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2726 pThis->pApicHlpRC = pThis->pApicHlpR3->pfnGetRCHelpers(pDevIns);
2727 pThis->paLapicsRC = MMHyperR3ToRC(PDMDevHlpGetVM(pDevIns), pThis->paLapicsR3);
2728 pThis->pCritSectRC = pThis->pApicHlpR3->pfnGetRCCritSect(pDevIns);
2729 for (uint32_t i = 0; i < pThis->cCpus; i++)
2730 pThis->paLapicsR3[i].pTimerRC = TMTimerRCPtr(pThis->paLapicsR3[i].pTimerR3);
2731}
2732
2733DECLINLINE(void) initApicData(APICState* apic, uint8_t id)
2734{
2735 int i;
2736 memset(apic, 0, sizeof(*apic));
2737 apic->apicbase = UINT32_C(0xfee00000) | MSR_IA32_APICBASE_ENABLE;
2738 /* Mark first CPU as BSP */
2739 if (id == 0)
2740 apic->apicbase |= MSR_IA32_APICBASE_BSP;
2741 for (i = 0; i < APIC_LVT_NB; i++)
2742 apic->lvt[i] = 1 << 16; /* mask LVT */
2743 apic->spurious_vec = 0xff;
2744 apic->phys_id = apic->id = id;
2745}
2746
2747/**
2748 * @copydoc FNPDMDEVCONSTRUCT
2749 */
2750static DECLCALLBACK(int) apicConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
2751{
2752 PDMAPICREG ApicReg;
2753 int rc;
2754 uint32_t i;
2755 bool fIoApic;
2756 bool fGCEnabled;
2757 bool fR0Enabled;
2758 APICDeviceInfo *pThis = PDMINS_2_DATA(pDevIns, APICDeviceInfo *);
2759 uint32_t cCpus;
2760
2761 /*
2762 * Only single device instance.
2763 */
2764 Assert(iInstance == 0);
2765
2766 /*
2767 * Validate configuration.
2768 */
2769 if (!CFGMR3AreValuesValid(pCfg,
2770 "IOAPIC\0"
2771 "GCEnabled\0"
2772 "R0Enabled\0"
2773 "NumCPUs\0"))
2774 return VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
2775
2776 rc = CFGMR3QueryBoolDef(pCfg, "IOAPIC", &fIoApic, true);
2777 if (RT_FAILURE(rc))
2778 return PDMDEV_SET_ERROR(pDevIns, rc,
2779 N_("Configuration error: Failed to read \"IOAPIC\""));
2780
2781 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &fGCEnabled, true);
2782 if (RT_FAILURE(rc))
2783 return PDMDEV_SET_ERROR(pDevIns, rc,
2784 N_("Configuration error: Failed to query boolean value \"GCEnabled\""));
2785
2786 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &fR0Enabled, true);
2787 if (RT_FAILURE(rc))
2788 return PDMDEV_SET_ERROR(pDevIns, rc,
2789 N_("Configuration error: Failed to query boolean value \"R0Enabled\""));
2790
2791 rc = CFGMR3QueryU32Def(pCfg, "NumCPUs", &cCpus, 1);
2792 if (RT_FAILURE(rc))
2793 return PDMDEV_SET_ERROR(pDevIns, rc,
2794 N_("Configuration error: Failed to query integer value \"NumCPUs\""));
2795
2796 Log(("APIC: cCpus=%d fR0Enabled=%RTbool fGCEnabled=%RTbool fIoApic=%RTbool\n", cCpus, fR0Enabled, fGCEnabled, fIoApic));
2797
2798 /** @todo Current implementation is limited to 32 CPUs due to the use of 32
2799 * bits bitmasks. */
2800 if (cCpus > 32)
2801 return PDMDEV_SET_ERROR(pDevIns, rc,
2802 N_("Configuration error: Invalid value for \"NumCPUs\""));
2803
2804 /*
2805 * Init the data.
2806 */
2807 pThis->pDevInsR3 = pDevIns;
2808 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2809 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2810 pThis->cCpus = cCpus;
2811 pThis->fIoApic = fIoApic;
2812 /* Use PDMAPICVERSION_X2APIC to activate x2APIC mode */
2813 pThis->enmVersion = PDMAPICVERSION_APIC;
2814
2815 PVM pVM = PDMDevHlpGetVM(pDevIns);
2816 /*
2817 * We are not freeing this memory, as it's automatically released when guest exits.
2818 */
2819 rc = MMHyperAlloc(pVM, cCpus * sizeof(APICState), 1, MM_TAG_PDM_DEVICE_USER, (void **)&pThis->paLapicsR3);
2820 if (RT_FAILURE(rc))
2821 return VERR_NO_MEMORY;
2822 pThis->paLapicsR0 = MMHyperR3ToR0(pVM, pThis->paLapicsR3);
2823 pThis->paLapicsRC = MMHyperR3ToRC(pVM, pThis->paLapicsR3);
2824
2825 for (i = 0; i < cCpus; i++)
2826 initApicData(&pThis->paLapicsR3[i], i);
2827
2828 /*
2829 * Register the APIC.
2830 */
2831 ApicReg.u32Version = PDM_APICREG_VERSION;
2832 ApicReg.pfnGetInterruptR3 = apicGetInterrupt;
2833 ApicReg.pfnHasPendingIrqR3 = apicHasPendingIrq;
2834 ApicReg.pfnSetBaseR3 = apicSetBase;
2835 ApicReg.pfnGetBaseR3 = apicGetBase;
2836 ApicReg.pfnSetTPRR3 = apicSetTPR;
2837 ApicReg.pfnGetTPRR3 = apicGetTPR;
2838 ApicReg.pfnWriteMSRR3 = apicWriteMSR;
2839 ApicReg.pfnReadMSRR3 = apicReadMSR;
2840 ApicReg.pfnBusDeliverR3 = apicBusDeliverCallback;
2841 ApicReg.pfnLocalInterruptR3 = apicLocalInterrupt;
2842 if (fGCEnabled) {
2843 ApicReg.pszGetInterruptRC = "apicGetInterrupt";
2844 ApicReg.pszHasPendingIrqRC = "apicHasPendingIrq";
2845 ApicReg.pszSetBaseRC = "apicSetBase";
2846 ApicReg.pszGetBaseRC = "apicGetBase";
2847 ApicReg.pszSetTPRRC = "apicSetTPR";
2848 ApicReg.pszGetTPRRC = "apicGetTPR";
2849 ApicReg.pszWriteMSRRC = "apicWriteMSR";
2850 ApicReg.pszReadMSRRC = "apicReadMSR";
2851 ApicReg.pszBusDeliverRC = "apicBusDeliverCallback";
2852 ApicReg.pszLocalInterruptRC = "apicLocalInterrupt";
2853 } else {
2854 ApicReg.pszGetInterruptRC = NULL;
2855 ApicReg.pszHasPendingIrqRC = NULL;
2856 ApicReg.pszSetBaseRC = NULL;
2857 ApicReg.pszGetBaseRC = NULL;
2858 ApicReg.pszSetTPRRC = NULL;
2859 ApicReg.pszGetTPRRC = NULL;
2860 ApicReg.pszWriteMSRRC = NULL;
2861 ApicReg.pszReadMSRRC = NULL;
2862 ApicReg.pszBusDeliverRC = NULL;
2863 ApicReg.pszLocalInterruptRC = NULL;
2864 }
2865 if (fR0Enabled) {
2866 ApicReg.pszGetInterruptR0 = "apicGetInterrupt";
2867 ApicReg.pszHasPendingIrqR0 = "apicHasPendingIrq";
2868 ApicReg.pszSetBaseR0 = "apicSetBase";
2869 ApicReg.pszGetBaseR0 = "apicGetBase";
2870 ApicReg.pszSetTPRR0 = "apicSetTPR";
2871 ApicReg.pszGetTPRR0 = "apicGetTPR";
2872 ApicReg.pszWriteMSRR0 = "apicWriteMSR";
2873 ApicReg.pszReadMSRR0 = "apicReadMSR";
2874 ApicReg.pszBusDeliverR0 = "apicBusDeliverCallback";
2875 ApicReg.pszLocalInterruptR0 = "apicLocalInterrupt";
2876 } else {
2877 ApicReg.pszGetInterruptR0 = NULL;
2878 ApicReg.pszHasPendingIrqR0 = NULL;
2879 ApicReg.pszSetBaseR0 = NULL;
2880 ApicReg.pszGetBaseR0 = NULL;
2881 ApicReg.pszSetTPRR0 = NULL;
2882 ApicReg.pszGetTPRR0 = NULL;
2883 ApicReg.pszWriteMSRR0 = NULL;
2884 ApicReg.pszReadMSRR0 = NULL;
2885 ApicReg.pszBusDeliverR0 = NULL;
2886 ApicReg.pszLocalInterruptR0 = NULL;
2887 }
2888
2889 rc = PDMDevHlpAPICRegister(pDevIns, &ApicReg, &pThis->pApicHlpR3);
2890 AssertLogRelRCReturn(rc, rc);
2891 pThis->pCritSectR3 = pThis->pApicHlpR3->pfnGetR3CritSect(pDevIns);
2892
2893 /*
2894 * The the CPUID feature bit.
2895 */
2896 /** @todo r=bird: See remark in the apicReset. */
2897 uint32_t u32Eax, u32Ebx, u32Ecx, u32Edx;
2898 PDMDevHlpGetCpuId(pDevIns, 0, &u32Eax, &u32Ebx, &u32Ecx, &u32Edx);
2899 if (u32Eax >= 1) {
2900 if ( fIoApic /* If IOAPIC is enabled, enable Local APIC in any case */
2901 || ( u32Ebx == X86_CPUID_VENDOR_INTEL_EBX
2902 && u32Ecx == X86_CPUID_VENDOR_INTEL_ECX
2903 && u32Edx == X86_CPUID_VENDOR_INTEL_EDX /* GenuineIntel */)
2904 || ( u32Ebx == X86_CPUID_VENDOR_AMD_EBX
2905 && u32Ecx == X86_CPUID_VENDOR_AMD_ECX
2906 && u32Edx == X86_CPUID_VENDOR_AMD_EDX /* AuthenticAMD */)) {
2907 LogRel(("Activating Local APIC\n"));
2908 pThis->pApicHlpR3->pfnChangeFeature(pDevIns, pThis->enmVersion);
2909 }
2910 }
2911
2912 /*
2913 * Register the MMIO range.
2914 */
2915 uint32_t ApicBase = pThis->paLapicsR3[0].apicbase & ~0xfff;
2916 rc = PDMDevHlpMMIORegister(pDevIns, ApicBase, 0x1000, pThis,
2917 apicMMIOWrite, apicMMIORead, NULL, "APIC Memory");
2918 if (RT_FAILURE(rc))
2919 return rc;
2920
2921 if (fGCEnabled) {
2922 pThis->pApicHlpRC = pThis->pApicHlpR3->pfnGetRCHelpers(pDevIns);
2923 pThis->pCritSectRC = pThis->pApicHlpR3->pfnGetRCCritSect(pDevIns);
2924
2925 rc = PDMDevHlpMMIORegisterRC(pDevIns, ApicBase, 0x1000, 0,
2926 "apicMMIOWrite", "apicMMIORead", NULL);
2927 if (RT_FAILURE(rc))
2928 return rc;
2929 }
2930
2931 if (fR0Enabled) {
2932 pThis->pApicHlpR0 = pThis->pApicHlpR3->pfnGetR0Helpers(pDevIns);
2933 pThis->pCritSectR0 = pThis->pApicHlpR3->pfnGetR0CritSect(pDevIns);
2934
2935 rc = PDMDevHlpMMIORegisterR0(pDevIns, ApicBase, 0x1000, 0,
2936 "apicMMIOWrite", "apicMMIORead", NULL);
2937 if (RT_FAILURE(rc))
2938 return rc;
2939 }
2940
2941 /*
2942 * Create the APIC timers.
2943 */
2944 for (i = 0; i < cCpus; i++) {
2945 APICState *pApic = &pThis->paLapicsR3[i];
2946 pApic->pszDesc = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_USER, "APIC Timer #%u", i);
2947 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL_SYNC, apicTimerCallback, pApic,
2948 TMTIMER_FLAGS_NO_CRIT_SECT, pApic->pszDesc, &pApic->pTimerR3);
2949 if (RT_FAILURE(rc))
2950 return rc;
2951 pApic->pTimerR0 = TMTimerR0Ptr(pApic->pTimerR3);
2952 pApic->pTimerRC = TMTimerRCPtr(pApic->pTimerR3);
2953 TMR3TimerSetCritSect(pApic->pTimerR3, pThis->pCritSectR3);
2954 }
2955
2956 /*
2957 * Saved state.
2958 */
2959 rc = PDMDevHlpSSMRegister3(pDevIns, APIC_SAVED_STATE_VERSION, sizeof(*pThis),
2960 apicLiveExec, apicSaveExec, apicLoadExec);
2961 if (RT_FAILURE(rc))
2962 return rc;
2963
2964 /*
2965 * Register debugger info callback.
2966 */
2967 PDMDevHlpDBGFInfoRegister(pDevIns, "lapic", "Display Local APIC state for current CPU. "
2968 "Recognizes 'basic', 'lvt', 'timer' as arguments, defaulting to 'basic'.", lapicInfo);
2969
2970#ifdef VBOX_WITH_STATISTICS
2971 /*
2972 * Statistics.
2973 */
2974 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadGC, STAMTYPE_COUNTER, "/Devices/APIC/MMIOReadGC", STAMUNIT_OCCURENCES, "Number of APIC MMIO reads in GC.");
2975 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadHC, STAMTYPE_COUNTER, "/Devices/APIC/MMIOReadHC", STAMUNIT_OCCURENCES, "Number of APIC MMIO reads in HC.");
2976 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteGC, STAMTYPE_COUNTER, "/Devices/APIC/MMIOWriteGC", STAMUNIT_OCCURENCES, "Number of APIC MMIO writes in GC.");
2977 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteHC, STAMTYPE_COUNTER, "/Devices/APIC/MMIOWriteHC", STAMUNIT_OCCURENCES, "Number of APIC MMIO writes in HC.");
2978 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatClearedActiveIrq,STAMTYPE_COUNTER, "/Devices/APIC/MaskedActiveIRQ", STAMUNIT_OCCURENCES, "Number of cleared irqs.");
2979 for (i = 0; i < cCpus; i++) {
2980 APICState *pApic = &pThis->paLapicsR3[i];
2981 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetInitialCount, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Calls to apicTimerSetInitialCount.", "/Devices/APIC/%u/TimerSetInitialCount", i);
2982 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetInitialCountArm, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerSetRelative calls.", "/Devices/APIC/%u/TimerSetInitialCount/Arm", i);
2983 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetInitialCountDisarm, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerStop calls.", "/Devices/APIC/%u/TimerSetInitialCount/Disasm", i);
2984 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvt, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Calls to apicTimerSetLvt.", "/Devices/APIC/%u/TimerSetLvt", i);
2985 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtClearPeriodic, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Clearing APIC_LVT_TIMER_PERIODIC.", "/Devices/APIC/%u/TimerSetLvt/ClearPeriodic", i);
2986 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtPostponed, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerStop postponed.", "/Devices/APIC/%u/TimerSetLvt/Postponed", i);
2987 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtArmed, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerSet avoided.", "/Devices/APIC/%u/TimerSetLvt/Armed", i);
2988 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtArm, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerSet necessary.", "/Devices/APIC/%u/TimerSetLvt/Arm", i);
2989 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtArmRetries, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "TMTimerSet retries.", "/Devices/APIC/%u/TimerSetLvt/ArmRetries", i);
2990 PDMDevHlpSTAMRegisterF(pDevIns, &pApic->StatTimerSetLvtNoRelevantChange,STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "No relevant flags changed.", "/Devices/APIC/%u/TimerSetLvt/NoRelevantChange", i);
2991 }
2992#endif
2993
2994 return VINF_SUCCESS;
2995}
2996
2997
2998/**
2999 * APIC device registration structure.
3000 */
3001const PDMDEVREG g_DeviceAPIC =
3002{
3003 /* u32Version */
3004 PDM_DEVREG_VERSION,
3005 /* szName */
3006 "apic",
3007 /* szRCMod */
3008 "VBoxDD2GC.gc",
3009 /* szR0Mod */
3010 "VBoxDD2R0.r0",
3011 /* pszDescription */
3012 "Advanced Programmable Interrupt Controller (APIC) Device",
3013 /* fFlags */
3014 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_32_64 | PDM_DEVREG_FLAGS_PAE36 | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
3015 /* fClass */
3016 PDM_DEVREG_CLASS_PIC,
3017 /* cMaxInstances */
3018 1,
3019 /* cbInstance */
3020 sizeof(APICState),
3021 /* pfnConstruct */
3022 apicConstruct,
3023 /* pfnDestruct */
3024 NULL,
3025 /* pfnRelocate */
3026 apicRelocate,
3027 /* pfnIOCtl */
3028 NULL,
3029 /* pfnPowerOn */
3030 NULL,
3031 /* pfnReset */
3032 apicReset,
3033 /* pfnSuspend */
3034 NULL,
3035 /* pfnResume */
3036 NULL,
3037 /* pfnAttach */
3038 NULL,
3039 /* pfnDetach */
3040 NULL,
3041 /* pfnQueryInterface. */
3042 NULL,
3043 /* pfnInitComplete */
3044 NULL,
3045 /* pfnPowerOff */
3046 NULL,
3047 /* pfnSoftReset */
3048 NULL,
3049 /* u32VersionEnd */
3050 PDM_DEVREG_VERSION
3051};
3052
3053#endif /* IN_RING3 */
3054
3055
3056/* IOAPIC */
3057
3058PDMBOTHCBDECL(int) ioapicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3059{
3060 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3061 IOAPIC_LOCK(s, VINF_IOM_HC_MMIO_READ);
3062
3063 STAM_COUNTER_INC(&CTXSUFF(s->StatMMIORead));
3064 switch (cb) {
3065 case 1:
3066 *(uint8_t *)pv = ioapic_mem_readl(s, GCPhysAddr);
3067 break;
3068
3069 case 2:
3070 *(uint16_t *)pv = ioapic_mem_readl(s, GCPhysAddr);
3071 break;
3072
3073 case 4:
3074 *(uint32_t *)pv = ioapic_mem_readl(s, GCPhysAddr);
3075 break;
3076
3077 default:
3078 AssertReleaseMsgFailed(("cb=%d\n", cb)); /* for now we assume simple accesses. */
3079 IOAPIC_UNLOCK(s);
3080 return VERR_INTERNAL_ERROR;
3081 }
3082 IOAPIC_UNLOCK(s);
3083 return VINF_SUCCESS;
3084}
3085
3086PDMBOTHCBDECL(int) ioapicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3087{
3088 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3089
3090 STAM_COUNTER_INC(&CTXSUFF(s->StatMMIOWrite));
3091 switch (cb) {
3092 case 1:
3093 case 2:
3094 case 4:
3095 IOAPIC_LOCK(s, VINF_IOM_HC_MMIO_WRITE);
3096 ioapic_mem_writel(s, GCPhysAddr, *(uint32_t *)pv);
3097 IOAPIC_UNLOCK(s);
3098 break;
3099
3100 default:
3101 AssertReleaseMsgFailed(("cb=%d\n", cb)); /* for now we assume simple accesses. */
3102 return VERR_INTERNAL_ERROR;
3103 }
3104 return VINF_SUCCESS;
3105}
3106
3107PDMBOTHCBDECL(void) ioapicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
3108{
3109 /* PDM lock is taken here; @todo add assertion */
3110 IOAPICState *pThis = PDMINS_2_DATA(pDevIns, IOAPICState *);
3111 STAM_COUNTER_INC(&pThis->CTXSUFF(StatSetIrq));
3112 LogFlow(("ioapicSetIrq: iIrq=%d iLevel=%d\n", iIrq, iLevel));
3113 ioapic_set_irq(pThis, iIrq, iLevel);
3114}
3115
3116
3117#ifdef IN_RING3
3118
3119/**
3120 * Info handler, device version. Dumps I/O APIC state.
3121 *
3122 * @param pDevIns Device instance which registered the info.
3123 * @param pHlp Callback functions for doing output.
3124 * @param pszArgs Argument string. Optional and specific to the handler.
3125 */
3126static DECLCALLBACK(void) ioapicInfo(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
3127{
3128 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3129 uint32_t val;
3130 unsigned i;
3131 unsigned max_redir;
3132
3133 pHlp->pfnPrintf(pHlp, "I/O APIC at %08X:\n", 0xfec00000);
3134 val = s->id << 24; /* Would be nice to call ioapic_mem_readl() directly, but that's not so simple. */
3135 pHlp->pfnPrintf(pHlp, " IOAPICID : %08X\n", val);
3136 pHlp->pfnPrintf(pHlp, " APIC ID = %02X\n", (val >> 24) & 0xff);
3137 val = 0x11 | ((IOAPIC_NUM_PINS - 1) << 16);
3138 max_redir = (val >> 16) & 0xff;
3139 pHlp->pfnPrintf(pHlp, " IOAPICVER : %08X\n", val);
3140 pHlp->pfnPrintf(pHlp, " version = %02X\n", val & 0xff);
3141 pHlp->pfnPrintf(pHlp, " redirs = %d\n", ((val >> 16) & 0xff) + 1);
3142 val = 0;
3143 pHlp->pfnPrintf(pHlp, " IOAPICARB : %08X\n", val);
3144 pHlp->pfnPrintf(pHlp, " arb ID = %02X\n", (val >> 24) & 0xff);
3145 Assert(sizeof(s->ioredtbl) / sizeof(s->ioredtbl[0]) > max_redir);
3146 pHlp->pfnPrintf(pHlp, "I/O redirection table\n");
3147 pHlp->pfnPrintf(pHlp, " idx dst_mode dst_addr mask trigger rirr polarity dlvr_st dlvr_mode vector\n");
3148 for (i = 0; i <= max_redir; ++i)
3149 {
3150 static const char *dmodes[] = { "Fixed ", "LowPri", "SMI ", "Resrvd",
3151 "NMI ", "INIT ", "Resrvd", "ExtINT" };
3152
3153 pHlp->pfnPrintf(pHlp, " %02d %s %02X %d %s %d %s %s %s %3d (%016llX)\n",
3154 i,
3155 s->ioredtbl[i] & (1 << 11) ? "log " : "phys", /* dest mode */
3156 (int)(s->ioredtbl[i] >> 56), /* dest addr */
3157 (int)(s->ioredtbl[i] >> 16) & 1, /* mask */
3158 s->ioredtbl[i] & (1 << 15) ? "level" : "edge ", /* trigger */
3159 (int)(s->ioredtbl[i] >> 14) & 1, /* remote IRR */
3160 s->ioredtbl[i] & (1 << 13) ? "activelo" : "activehi", /* polarity */
3161 s->ioredtbl[i] & (1 << 12) ? "pend" : "idle", /* delivery status */
3162 dmodes[(s->ioredtbl[i] >> 8) & 0x07], /* delivery mode */
3163 (int)s->ioredtbl[i] & 0xff, /* vector */
3164 s->ioredtbl[i] /* entire register */
3165 );
3166 }
3167}
3168
3169/**
3170 * @copydoc FNSSMDEVSAVEEXEC
3171 */
3172static DECLCALLBACK(int) ioapicSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3173{
3174 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3175 ioapic_save(pSSM, s);
3176 return VINF_SUCCESS;
3177}
3178
3179/**
3180 * @copydoc FNSSMDEVLOADEXEC
3181 */
3182static DECLCALLBACK(int) ioapicLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
3183{
3184 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3185
3186 if (ioapic_load(pSSM, s, uVersion)) {
3187 AssertFailed();
3188 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
3189 }
3190 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
3191
3192 return VINF_SUCCESS;
3193}
3194
3195/**
3196 * @copydoc FNPDMDEVRESET
3197 */
3198static DECLCALLBACK(void) ioapicReset(PPDMDEVINS pDevIns)
3199{
3200 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3201 s->pIoApicHlpR3->pfnLock(pDevIns, VERR_INTERNAL_ERROR);
3202 ioapic_reset(s);
3203 IOAPIC_UNLOCK(s);
3204}
3205
3206/**
3207 * @copydoc FNPDMDEVRELOCATE
3208 */
3209static DECLCALLBACK(void) ioapicRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
3210{
3211 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3212 s->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
3213 s->pIoApicHlpRC = s->pIoApicHlpR3->pfnGetRCHelpers(pDevIns);
3214}
3215
3216/**
3217 * @copydoc FNPDMDEVCONSTRUCT
3218 */
3219static DECLCALLBACK(int) ioapicConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
3220{
3221 IOAPICState *s = PDMINS_2_DATA(pDevIns, IOAPICState *);
3222 PDMIOAPICREG IoApicReg;
3223 bool fGCEnabled;
3224 bool fR0Enabled;
3225 int rc;
3226
3227 Assert(iInstance == 0);
3228
3229 /*
3230 * Validate and read the configuration.
3231 */
3232 if (!CFGMR3AreValuesValid(pCfg, "GCEnabled\0" "R0Enabled\0"))
3233 return VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
3234
3235 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &fGCEnabled, true);
3236 if (RT_FAILURE(rc))
3237 return PDMDEV_SET_ERROR(pDevIns, rc,
3238 N_("Configuration error: Failed to query boolean value \"GCEnabled\""));
3239
3240 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &fR0Enabled, true);
3241 if (RT_FAILURE(rc))
3242 return PDMDEV_SET_ERROR(pDevIns, rc,
3243 N_("Configuration error: Failed to query boolean value \"R0Enabled\""));
3244 Log(("IOAPIC: fR0Enabled=%RTbool fGCEnabled=%RTbool\n", fR0Enabled, fGCEnabled));
3245
3246 /*
3247 * Initialize the state data.
3248 */
3249 s->pDevInsR3 = pDevIns;
3250 s->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
3251 s->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
3252 ioapic_reset(s);
3253 s->id = 0;
3254
3255 /*
3256 * Register the IOAPIC and get helpers.
3257 */
3258 IoApicReg.u32Version = PDM_IOAPICREG_VERSION;
3259 IoApicReg.pfnSetIrqR3 = ioapicSetIrq;
3260 IoApicReg.pszSetIrqRC = fGCEnabled ? "ioapicSetIrq" : NULL;
3261 IoApicReg.pszSetIrqR0 = fR0Enabled ? "ioapicSetIrq" : NULL;
3262 rc = PDMDevHlpIOAPICRegister(pDevIns, &IoApicReg, &s->pIoApicHlpR3);
3263 if (RT_FAILURE(rc))
3264 {
3265 AssertMsgFailed(("IOAPICRegister -> %Rrc\n", rc));
3266 return rc;
3267 }
3268
3269 /*
3270 * Register MMIO callbacks and saved state.
3271 */
3272 rc = PDMDevHlpMMIORegister(pDevIns, 0xfec00000, 0x1000, s,
3273 ioapicMMIOWrite, ioapicMMIORead, NULL, "I/O APIC Memory");
3274 if (RT_FAILURE(rc))
3275 return rc;
3276
3277 if (fGCEnabled) {
3278 s->pIoApicHlpRC = s->pIoApicHlpR3->pfnGetRCHelpers(pDevIns);
3279
3280 rc = PDMDevHlpMMIORegisterRC(pDevIns, 0xfec00000, 0x1000, 0,
3281 "ioapicMMIOWrite", "ioapicMMIORead", NULL);
3282 if (RT_FAILURE(rc))
3283 return rc;
3284 }
3285
3286 if (fR0Enabled) {
3287 s->pIoApicHlpR0 = s->pIoApicHlpR3->pfnGetR0Helpers(pDevIns);
3288
3289 rc = PDMDevHlpMMIORegisterR0(pDevIns, 0xfec00000, 0x1000, 0,
3290 "ioapicMMIOWrite", "ioapicMMIORead", NULL);
3291 if (RT_FAILURE(rc))
3292 return rc;
3293 }
3294
3295 rc = PDMDevHlpSSMRegister(pDevIns, 1 /* version */, sizeof(*s), ioapicSaveExec, ioapicLoadExec);
3296 if (RT_FAILURE(rc))
3297 return rc;
3298
3299 /*
3300 * Register debugger info callback.
3301 */
3302 PDMDevHlpDBGFInfoRegister(pDevIns, "ioapic", "Display I/O APIC state.", ioapicInfo);
3303
3304#ifdef VBOX_WITH_STATISTICS
3305 /*
3306 * Statistics.
3307 */
3308 PDMDevHlpSTAMRegister(pDevIns, &s->StatMMIOReadGC, STAMTYPE_COUNTER, "/Devices/IOAPIC/MMIOReadGC", STAMUNIT_OCCURENCES, "Number of IOAPIC MMIO reads in GC.");
3309 PDMDevHlpSTAMRegister(pDevIns, &s->StatMMIOReadHC, STAMTYPE_COUNTER, "/Devices/IOAPIC/MMIOReadHC", STAMUNIT_OCCURENCES, "Number of IOAPIC MMIO reads in HC.");
3310 PDMDevHlpSTAMRegister(pDevIns, &s->StatMMIOWriteGC, STAMTYPE_COUNTER, "/Devices/IOAPIC/MMIOWriteGC", STAMUNIT_OCCURENCES, "Number of IOAPIC MMIO writes in GC.");
3311 PDMDevHlpSTAMRegister(pDevIns, &s->StatMMIOWriteHC, STAMTYPE_COUNTER, "/Devices/IOAPIC/MMIOWriteHC", STAMUNIT_OCCURENCES, "Number of IOAPIC MMIO writes in HC.");
3312 PDMDevHlpSTAMRegister(pDevIns, &s->StatSetIrqGC, STAMTYPE_COUNTER, "/Devices/IOAPIC/SetIrqGC", STAMUNIT_OCCURENCES, "Number of IOAPIC SetIrq calls in GC.");
3313 PDMDevHlpSTAMRegister(pDevIns, &s->StatSetIrqHC, STAMTYPE_COUNTER, "/Devices/IOAPIC/SetIrqHC", STAMUNIT_OCCURENCES, "Number of IOAPIC SetIrq calls in HC.");
3314#endif
3315
3316 return VINF_SUCCESS;
3317}
3318
3319/**
3320 * IO APIC device registration structure.
3321 */
3322const PDMDEVREG g_DeviceIOAPIC =
3323{
3324 /* u32Version */
3325 PDM_DEVREG_VERSION,
3326 /* szName */
3327 "ioapic",
3328 /* szRCMod */
3329 "VBoxDD2GC.gc",
3330 /* szR0Mod */
3331 "VBoxDD2R0.r0",
3332 /* pszDescription */
3333 "I/O Advanced Programmable Interrupt Controller (IO-APIC) Device",
3334 /* fFlags */
3335 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_32_64 | PDM_DEVREG_FLAGS_PAE36 | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
3336 /* fClass */
3337 PDM_DEVREG_CLASS_PIC,
3338 /* cMaxInstances */
3339 1,
3340 /* cbInstance */
3341 sizeof(IOAPICState),
3342 /* pfnConstruct */
3343 ioapicConstruct,
3344 /* pfnDestruct */
3345 NULL,
3346 /* pfnRelocate */
3347 ioapicRelocate,
3348 /* pfnIOCtl */
3349 NULL,
3350 /* pfnPowerOn */
3351 NULL,
3352 /* pfnReset */
3353 ioapicReset,
3354 /* pfnSuspend */
3355 NULL,
3356 /* pfnResume */
3357 NULL,
3358 /* pfnAttach */
3359 NULL,
3360 /* pfnDetach */
3361 NULL,
3362 /* pfnQueryInterface. */
3363 NULL,
3364 /* pfnInitComplete */
3365 NULL,
3366 /* pfnPowerOff */
3367 NULL,
3368 /* pfnSoftReset */
3369 NULL,
3370 /* u32VersionEnd */
3371 PDM_DEVREG_VERSION
3372};
3373
3374#endif /* IN_RING3 */
3375#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette