VirtualBox

Changeset 9570 in vbox for trunk/src


Ignore:
Timestamp:
Jun 10, 2008 1:30:03 PM (17 years ago)
Author:
vboxsync
Message:

AMD64 paging updates

Location:
trunk/src/VBox/VMM
Files:
3 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/VMM/PGMInternal.h

    r9398 r9570  
    35243524    return 0ULL;
    35253525}
     3526
     3527/**
     3528 * Gets the GUEST page directory pointer for the specified address.
     3529 *
     3530 * @returns The page directory in question.
     3531 * @returns NULL if the page directory is not present or on an invalid page.
     3532 * @param   pPGM        Pointer to the PGM instance data.
     3533 * @param   GCPtr       The address.
     3534 * @param   piPD        Receives the index into the returned page directory
     3535 */
     3536DECLINLINE(PX86PDPAE) pgmGstGetLongModePDPtr(PPGM pPGM, RTGCUINTPTR64 GCPtr, unsigned *piPD)
     3537{
     3538    PX86PML4E pPml4e;
     3539    PX86PDPE pPdpe;
     3540    const unsigned iPml4e = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
     3541
     3542    pPml4e = &pPGM->pGstPaePML4HC->a[iPml4e];
     3543    if (pPml4e->n.u1Present)
     3544    {
     3545        PX86PDPT pPdptTemp;
     3546        int rc = PGM_GCPHYS_2_PTR(PGM2VM(pPGM), pPml4e->u & X86_PML4E_PG_MASK, &pPdptTemp);
     3547        if (VBOX_FAILURE(rc))
     3548        {
     3549            AssertFailed();
     3550            return 0ULL;
     3551        }
     3552
     3553        const unsigned iPdPt  = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_AMD64;
     3554        pPdpe = &pPdptTemp->a[iPdPt];
     3555        if (pPdpe->n.u1Present)
     3556        {
     3557            PX86PDPAE pPD;
     3558
     3559            rc = PGM_GCPHYS_2_PTR(PGM2VM(pPGM), pPdpe->u & X86_PDPE_PG_MASK, &pPD);
     3560            if (VBOX_FAILURE(rc))
     3561            {
     3562                AssertFailed();
     3563                return 0ULL;
     3564            }
     3565            *piPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
     3566            return pPD;
     3567        }
     3568    }
     3569    return 0ULL;
     3570}
     3571
    35263572#endif /* !IN_GC */
    35273573
  • trunk/src/VBox/VMM/VMMAll/PGMAll.cpp

    r9539 r9570  
    731731 * @param   ppPD        Receives address of page directory
    732732 */
    733 PGMDECL(int) PGMShwGetLongModePDPtr(PVM pVM, RTGCUINTPTR64 GCPtr, PX86PDPAE *ppPD)
     733PGMDECL(int) PGMShwGetAllocLongModePDPtr(PVM pVM, RTGCUINTPTR64 GCPtr, PX86PDPAE *ppPD)
    734734{
    735735    PPGM           pPGM   = &pVM->pgm.s;
     
    783783        AssertRCReturn(rc, rc);
    784784
    785         /* The PDPT was cached or created; hook it up now. */
     785        /* The PD was cached or created; hook it up now. */
    786786        pPdpe->u |= pShwPage->Core.Key;
    787787    }
     
    791791        AssertReturn(pShwPage, VERR_INTERNAL_ERROR);
    792792    }
     793
     794    *ppPD = (PX86PDPAE)PGMPOOL_PAGE_2_PTR(pVM, pShwPage);
     795    return VINF_SUCCESS;
     796}
     797
     798/**
     799 * Gets the SHADOW page directory pointer for the specified address.
     800 *
     801 * @returns VBox status.
     802 * @param   pVM         VM handle.
     803 * @param   GCPtr       The address.
     804 * @param   ppPdpt      Receives address of pdpt
     805 * @param   ppPD        Receives address of page directory
     806 */
     807PGMDECL(int) PGMShwGetLongModePDPtr(PVM pVM, RTGCUINTPTR64 GCPtr, PX86PDPT *ppPdpt, PX86PDPAE *ppPD)
     808{
     809    PPGM           pPGM   = &pVM->pgm.s;
     810    const unsigned iPml4e = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
     811    PPGMPOOL       pPool  = pPGM->CTXSUFF(pPool);
     812    PX86PML4E      pPml4e;
     813    PPGMPOOLPAGE   pShwPage;
     814
     815    Assert(!HWACCMIsNestedPagingActive(pVM));
     816
     817    pPml4e = &pPGM->pHCPaePML4->a[iPml4e];
     818    if (!pPml4e->n.u1Present)
     819        return VERR_PAGE_TABLE_NOT_PRESENT;
     820
     821    pShwPage = pgmPoolGetPage(pPool, pPml4e->u & X86_PML4E_PG_MASK);
     822    AssertReturn(pShwPage, VERR_INTERNAL_ERROR);
     823
     824    const unsigned iPdPt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_AMD64;
     825    PX86PDPT  pPdpt = (PX86PDPT)PGMPOOL_PAGE_2_PTR(pVM, pShwPage);   
     826    PX86PDPE  pPdpe = &pPdpt->a[iPdPt];
     827
     828    *ppPdpt = pPdpt;
     829    if (!pPdpe->n.u1Present)
     830        return VERR_PAGE_TABLE_NOT_PRESENT;
     831
     832    pShwPage = pgmPoolGetPage(pPool, pPdpe->u & X86_PDPE_PG_MASK);
     833    AssertReturn(pShwPage, VERR_INTERNAL_ERROR);
    793834
    794835    *ppPD = (PX86PDPAE)PGMPOOL_PAGE_2_PTR(pVM, pShwPage);
  • trunk/src/VBox/VMM/VMMAll/PGMAllBth.h

    r9344 r9570  
    150150    PX86PDPAE       pPDDst;
    151151
    152     rc = PGMShwGetLongModePDPtr(pVM, (RTGCUINTPTR)pvFault, &pPDDst);
     152    rc = PGMShwGetAllocLongModePDPtr(pVM, (RTGCUINTPTR)pvFault, &pPDDst);
    153153    if (rc != VINF_SUCCESS)
    154154    {
     
    14191419
    14201420#if    (   PGM_GST_TYPE == PGM_TYPE_32BIT \
    1421         || PGM_GST_TYPE == PGM_TYPE_PAE) \
     1421        || PGM_GST_TYPE == PGM_TYPE_PAE \
     1422        || PGM_GST_TYPE == PGM_TYPE_AMD64) \
    14221423    && PGM_SHW_TYPE != PGM_TYPE_NESTED
    14231424
     
    14361437     * Get the shadow PDE, find the shadow page table in the pool.
    14371438     */
     1439# if PGM_SHW_TYPE == PGM_TYPE_32BIT
    14381440    const unsigned iPDDst = GCPtrPage >> SHW_PD_SHIFT;
    1439 # if PGM_SHW_TYPE == PGM_TYPE_32BIT
    14401441    X86PDE          PdeDst = pVM->pgm.s.CTXMID(p,32BitPD)->a[iPDDst];
    1441 # else /* PAE */
     1442# elif PGM_SHW_TYPE == PGM_TYPE_PAE
     1443    const unsigned iPDDst = GCPtrPage >> SHW_PD_SHIFT;
    14421444    X86PDEPAE       PdeDst = pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[iPDDst];
     1445# elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     1446    const unsigned  iPDDst = ((GCPtrPage >> SHW_PD_SHIFT) & SHW_PD_MASK);
     1447    PX86PDPAE       pPDDst;
     1448    X86PDEPAE       PdeDst;
     1449
     1450    int rc = PGMShwGetAllocLongModePDPtr(pVM, GCPtrPage, &pPDDst);
     1451    if (rc != VINF_SUCCESS)
     1452    {
     1453        AssertMsg(rc == VINF_PGM_SYNC_CR3, ("Unexpected rc=%Vrc\n", rc));
     1454        return rc;
     1455    }
     1456    Assert(pPDDst);
     1457    PdeDst = pPDDst->a[iPDDst];
    14431458# endif
    14441459    Assert(PdeDst.n.u1Present);
     
    16271642#  if PGM_SHW_TYPE == PGM_TYPE_32BIT
    16281643                    pVM->pgm.s.CTXMID(p,32BitPD)->a[iPDDst]    = PdeDst;
    1629 #  else /* PAE */
     1644#  elif PGM_SHW_TYPE == PGM_TYPE_PAE
    16301645                    pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[iPDDst] = PdeDst;
     1646#  elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     1647                    pPDDst->a[iPDDst] = PdeDst;
    16311648#  endif
    16321649                    Log2(("SyncPage: BIG %VGv PdeSrc:{P=%d RW=%d U=%d raw=%08llx} GCPhys=%VGp%s\n",
     
    16551672#  if PGM_SHW_TYPE == PGM_TYPE_32BIT
    16561673    pVM->pgm.s.CTXMID(p,32BitPD)->a[iPDDst].u    = 0;
    1657 #  else /* PAE */
     1674#  elif PGM_SHW_TYPE == PGM_TYPE_PAE
    16581675    pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[iPDDst].u = 0;
     1676#  elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     1677    pPDDst->a[iPDDst].u = 0;
    16591678#  endif
    16601679    PGM_INVL_GUEST_TLBS();
     
    17551774    return VINF_SUCCESS;
    17561775
    1757 #else /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
     1776#else
    17581777    AssertReleaseMsgFailed(("Shw=%d Gst=%d is not implemented!\n", PGM_GST_TYPE, PGM_SHW_TYPE));
    17591778    return VERR_INTERNAL_ERROR;
    1760 #endif /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
     1779#endif
    17611780}
    17621781
     
    20932112    LogFlow(("SyncPT: GCPtrPage=%VGv\n", GCPtrPage));
    20942113
    2095 #if   (   PGM_GST_TYPE == PGM_TYPE_32BIT \
    2096        || PGM_GST_TYPE == PGM_TYPE_PAE) \
     2114#if   (   PGM_GST_TYPE == PGM_TYPE_32BIT  \
     2115       || PGM_GST_TYPE == PGM_TYPE_PAE    \
     2116       || PGM_GST_TYPE == PGM_TYPE_AMD64) \
    20972117    && PGM_SHW_TYPE != PGM_TYPE_NESTED
     2118
     2119    int rc = VINF_SUCCESS;
    20982120
    20992121    /*
     
    21022124    AssertMsg(iPDSrc == ((GCPtrPage >> GST_PD_SHIFT) & GST_PD_MASK), ("iPDSrc=%x GCPtrPage=%VGv\n", iPDSrc, GCPtrPage));
    21032125# if PGM_SHW_TYPE == PGM_TYPE_32BIT
     2126    const unsigned  iPDDst = GCPtrPage >> SHW_PD_SHIFT;
    21042127    PX86PD          pPDDst = pVM->pgm.s.CTXMID(p,32BitPD);
    2105 # else
     2128# elif PGM_SHW_TYPE == PGM_TYPE_PAE
     2129    const unsigned  iPDDst = GCPtrPage >> SHW_PD_SHIFT;
    21062130    PX86PDPAE       pPDDst = pVM->pgm.s.CTXMID(ap,PaePDs)[0];
    2107 # endif
    2108     const unsigned  iPDDst = GCPtrPage >> SHW_PD_SHIFT;
     2131# elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     2132    const unsigned  iPDDst = (GCPtrPage >> SHW_PD_SHIFT) & SHW_PD_MASK;
     2133    PX86PDPAE       pPDDst;
     2134    rc = PGMShwGetAllocLongModePDPtr(pVM, GCPtrPage, &pPDDst);
     2135    if (rc != VINF_SUCCESS)
     2136    {
     2137        AssertMsg(rc == VINF_PGM_SYNC_CR3, ("Unexpected rc=%Vrc\n", rc));
     2138        return rc;
     2139    }
     2140    Assert(pPDDst);
     2141# endif
    21092142    PSHWPDE         pPdeDst = &pPDDst->a[iPDDst];
    21102143    SHWPDE          PdeDst = *pPdeDst;
     
    21302163#   elif PGM_GST_TYPE == PGM_TYPE_PAE
    21312164        int rc = pgmR3SyncPTResolveConflictPAE(pVM, pMapping, GCPtrPage & (GST_PD_MASK << GST_PD_SHIFT));
     2165#   else
     2166        AssertFailed();     /* can't happen for amd64 */
    21322167#   endif
    21332168        if (VBOX_FAILURE(rc))
     
    21472182     * Sync page directory entry.
    21482183     */
    2149     int     rc     = VINF_SUCCESS;
    21502184    GSTPDE  PdeSrc = pPDSrc->a[iPDSrc];
    21512185    if (PdeSrc.n.u1Present)
     
    25022536    return rc;
    25032537
    2504 #else /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
     2538#else
    25052539    AssertReleaseMsgFailed(("Shw=%d Gst=%d is not implemented!\n", PGM_GST_TYPE, PGM_SHW_TYPE));
    25062540    STAM_PROFILE_STOP(&pVM->pgm.s.CTXMID(Stat,SyncPT), a);
    25072541    return VERR_INTERNAL_ERROR;
    2508 #endif /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
     2542#endif
    25092543}
    25102544
     
    25232557PGM_BTH_DECL(int, PrefetchPage)(PVM pVM, RTGCUINTPTR GCPtrPage)
    25242558{
    2525 #if   (PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_REAL || PGM_GST_TYPE == PGM_TYPE_PROT || PGM_GST_TYPE == PGM_TYPE_PAE) \
    2526     && PGM_SHW_TYPE != PGM_TYPE_AMD64 && PGM_SHW_TYPE != PGM_TYPE_NESTED
     2559#if   (PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_REAL || PGM_GST_TYPE == PGM_TYPE_PROT || PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64) \
     2560    && PGM_SHW_TYPE != PGM_TYPE_NESTED
    25272561    /*
    25282562     * Check that all Guest levels thru the PDE are present, getting the
     
    25342568    const unsigned  iPDSrc = (RTGCUINTPTR)GCPtrPage >> GST_PD_SHIFT;
    25352569    PGSTPD          pPDSrc = CTXSUFF(pVM->pgm.s.pGuestPD);
    2536 #  else /* PAE */
     2570#  elif PGM_GST_TYPE == PGM_TYPE_PAE
    25372571    unsigned        iPDSrc;
    25382572    PGSTPD          pPDSrc = pgmGstGetPaePDPtr(&pVM->pgm.s, GCPtrPage, &iPDSrc);
     2573    if (!pPDSrc)
     2574        return VINF_SUCCESS; /* not present */
     2575#  elif PGM_GST_TYPE == PGM_TYPE_AMD64
     2576    unsigned        iPDSrc;
     2577    PX86PML4E       pPml4e;
     2578    X86PDPE         Pdpe;
     2579    PGSTPD          pPDSrc = pgmGstGetLongModePDPtr(&pVM->pgm.s, GCPtrPage, &pPml4e, &Pdpe, &iPDSrc);
    25392580    if (!pPDSrc)
    25402581        return VINF_SUCCESS; /* not present */
     
    25572598# if PGM_SHW_TYPE == PGM_TYPE_32BIT
    25582599        const X86PDE    PdeDst = pVM->pgm.s.CTXMID(p,32BitPD)->a[GCPtrPage >> SHW_PD_SHIFT];
    2559 # else
     2600# elif PGM_SHW_TYPE == PGM_TYPE_PAE
    25602601        const X86PDEPAE PdeDst = pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[GCPtrPage >> SHW_PD_SHIFT];
     2602# elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     2603        const unsigned  iPDDst = ((GCPtrPage >> SHW_PD_SHIFT) & SHW_PD_MASK);
     2604        PX86PDPAE       pPDDst;
     2605        PX86PDPT        pPdptDst;
     2606        X86PDEPAE       PdeDst;
     2607
     2608        int rc = PGMShwGetLongModePDPtr(pVM, GCPtrPage, &pPdptDst, &pPDDst);
     2609        if (rc != VINF_SUCCESS)
     2610        {
     2611            AssertMsg(rc == VERR_PAGE_TABLE_NOT_PRESENT, ("Unexpected rc=%Vrc\n", rc));
     2612            return rc;
     2613        }
     2614        Assert(pPDDst);
     2615        PdeDst = pPDDst->a[iPDDst];
    25612616# endif
    25622617        if (!(PdeDst.u & PGM_PDFLAGS_MAPPING))
     
    25802635#elif PGM_SHW_TYPE == PGM_TYPE_NESTED
    25812636    return VINF_SUCCESS; /* ignore */
    2582 #else /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
    2583 
    2584     AssertReleaseMsgFailed(("Shw=%d Gst=%d is not implemented!\n", PGM_SHW_TYPE, PGM_GST_TYPE));
    2585     return VERR_INTERNAL_ERROR;
    2586 #endif /* PGM_GST_TYPE == PGM_TYPE_AMD64 */
     2637#endif
    25872638}
    25882639
     
    26032654
    26042655    Assert(!HWACCMIsNestedPagingActive(pVM));
    2605 #if   (PGM_GST_TYPE == PGM_TYPE_32BIT ||  PGM_GST_TYPE == PGM_TYPE_REAL ||  PGM_GST_TYPE == PGM_TYPE_PROT || PGM_GST_TYPE == PGM_TYPE_PAE) \
    2606     && PGM_SHW_TYPE != PGM_TYPE_AMD64 && PGM_SHW_TYPE != PGM_TYPE_NESTED
     2656#if   (PGM_GST_TYPE == PGM_TYPE_32BIT ||  PGM_GST_TYPE == PGM_TYPE_REAL ||  PGM_GST_TYPE == PGM_TYPE_PROT || PGM_GST_TYPE == PGM_TYPE_PAE || PGM_TYPE_AMD64) \
     2657    && PGM_SHW_TYPE != PGM_TYPE_NESTED
    26072658
    26082659# ifndef IN_RING0
     
    26252676    const unsigned  iPDSrc = (RTGCUINTPTR)GCPtrPage >> GST_PD_SHIFT;
    26262677    PGSTPD          pPDSrc = CTXSUFF(pVM->pgm.s.pGuestPD);
    2627 #  else /* PAE */
     2678#  elif PGM_GST_TYPE == PGM_TYPE_PAE
    26282679    unsigned        iPDSrc;
    26292680    PGSTPD          pPDSrc = pgmGstGetPaePDPtr(&pVM->pgm.s, GCPtrPage, &iPDSrc);
    26302681
    26312682    if (pPDSrc)
     2683    {
     2684        Log(("PGMVerifyAccess: access violation for %VGv due to non-present PDPTR\n", GCPtrPage));
     2685        return VINF_EM_RAW_GUEST_TRAP;
     2686    }
     2687#  elif PGM_GST_TYPE == PGM_TYPE_AMD64
     2688    unsigned        iPDSrc;
     2689    PX86PML4E       pPml4e;
     2690    X86PDPE         Pdpe;
     2691    PGSTPD          pPDSrc = pgmGstGetLongModePDPtr(&pVM->pgm.s, GCPtrPage, &pPml4e, &Pdpe, &iPDSrc);
     2692    if (!pPDSrc)
    26322693    {
    26332694        Log(("PGMVerifyAccess: access violation for %VGv due to non-present PDPTR\n", GCPtrPage));
     
    26462707# if PGM_SHW_TYPE == PGM_TYPE_32BIT
    26472708    PX86PDE     pPdeDst = &pVM->pgm.s.CTXMID(p,32BitPD)->a[GCPtrPage >> SHW_PD_SHIFT];
    2648 # else
     2709# elif PGM_SHW_TYPE == PGM_TYPE_PAE
    26492710    PX86PDEPAE  pPdeDst = &pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[GCPtrPage >> SHW_PD_SHIFT];
     2711# elif PGM_SHW_TYPE == PGM_TYPE_AMD64
     2712    const unsigned  iPDDst = ((GCPtrPage >> SHW_PD_SHIFT) & SHW_PD_MASK);
     2713    PX86PDPAE       pPDDst;
     2714    PX86PDEPAE      pPdeDst;
     2715
     2716    rc = PGMShwGetAllocLongModePDPtr(pVM, GCPtrPage, &pPDDst);
     2717    if (rc != VINF_SUCCESS)
     2718    {
     2719        AssertMsg(rc == VINF_PGM_SYNC_CR3, ("Unexpected rc=%Vrc\n", rc));
     2720        return rc;
     2721    }
     2722    Assert(pPDDst);
     2723    pPdeDst = &pPDDst->a[iPDDst];
    26502724# endif
    26512725    if (!pPdeDst->n.u1Present)
     
    27042778
    27052779
    2706 #if PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_PAE
    2707 # if PGM_SHW_TYPE == PGM_TYPE_32BIT || PGM_SHW_TYPE == PGM_TYPE_PAE
     2780#if PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
     2781# if PGM_SHW_TYPE == PGM_TYPE_32BIT || PGM_SHW_TYPE == PGM_TYPE_PAE || PGM_SHW_TYPE == PGM_TYPE_AMD64
    27082782/**
    27092783 * Figures out which kind of shadow page this guest PDE warrants.
     
    28052879    MY_STAM_COUNTER_INC(fGlobal ? &pVM->pgm.s.CTXMID(Stat,SyncCR3Global) : &pVM->pgm.s.CTXMID(Stat,SyncCR3NotGlobal));
    28062880
    2807 # if PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_PAE
     2881# if PGM_GST_TYPE == PGM_TYPE_32BIT || PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
    28082882    /*
    28092883     * Get page directory addresses.
     
    28112885#  if PGM_SHW_TYPE == PGM_TYPE_32BIT
    28122886    PX86PDE     pPDEDst = &pVM->pgm.s.CTXMID(p,32BitPD)->a[0];
    2813 #  else /* PGM_SHW_TYPE == PGM_TYPE_PAE */
     2887#  else /* PGM_SHW_TYPE == PGM_TYPE_PAE || PGM_SHW_TYPE == PGM_TYPE_AMD64*/
    28142888#   if PGM_GST_TYPE == PGM_TYPE_32BIT
    28152889    PX86PDEPAE  pPDEDst = &pVM->pgm.s.CTXMID(ap,PaePDs)[0]->a[0];
     
    28442918        iPdNoMapping  = ~0U;
    28452919    }
     2920#  if PGM_GST_TYPE == PGM_TYPE_AMD64
     2921    for (uint64_t iPML4E = 0; iPML4E < X86_PG_PAE_ENTRIES; iPML4E++)
     2922    {
     2923#  else
     2924    {
     2925#  endif
    28462926#  if PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
    2847     for (unsigned iPDPTE = 0; iPDPTE < GST_PDPE_ENTRIES; iPDPTE++)
    2848     {
    2849         unsigned        iPDSrc;
    2850 #   if PGM_SHW_TYPE == PGM_TYPE_PAE
    2851         PX86PDPAE       pPDPAE    = pVM->pgm.s.CTXMID(ap,PaePDs)[0];
     2927        for (uint64_t iPDPTE = 0; iPDPTE < GST_PDPE_ENTRIES; iPDPTE++)
     2928        {
     2929            unsigned        iPDSrc;
     2930#   if PGM_GST_TYPE == PGM_TYPE_PAE
     2931            PX86PDPAE       pPDPAE    = pVM->pgm.s.CTXMID(ap,PaePDs)[0];
     2932            PX86PDEPAE      pPDEDst   = &pPDPAE->a[iPDPTE * X86_PG_PAE_ENTRIES];
     2933            PGSTPD          pPDSrc    = pgmGstGetPaePDPtr(&pVM->pgm.s, iPDPTE << X86_PDPT_SHIFT, &iPDSrc);
    28522934#   else
    2853         AssertFailed(); /* @todo */
    2854         PX86PDPE        pPDPAE    = pVM->pgm.s.CTXMID(ap,PaePDs)[iPDPTE * X86_PG_AMD64_ENTRIES];
     2935            PX86PML4E       pPml4eSrc;
     2936            X86PDPE         PdpeSrc;
     2937            PX86PDPT        pPdptDst;
     2938            PX86PDPAE       pPDDst;
     2939            PX86PDEPAE      pPDEDst;
     2940            RTGCUINTPTR     GCPtr   = (iPML4E << X86_PML4_SHIFT) || (iPDPTE << X86_PDPT_SHIFT);
     2941            PGSTPD          pPDSrc    = pgmGstGetLongModePDPtr(&pVM->pgm.s, GCPtr, &pPml4eSrc, &PdpeSrc, &iPDSrc);
     2942
     2943            int rc = PGMShwGetLongModePDPtr(pVM, GCPtr, &pPdptDst, &pPDDst);
     2944            if (rc != VINF_SUCCESS)
     2945            {
     2946                AssertMsg(rc == VERR_PAGE_TABLE_NOT_PRESENT, ("Unexpected rc=%Vrc\n", rc));
     2947                return rc;
     2948            }
     2949            Assert(pPDDst);
     2950            pPDEDst = &pPDDst->a[0];
     2951
     2952            if (!pPml4eSrc->n.u1Present)
     2953            {
     2954                /* Guest PML4 not present (anymore). */
     2955                if (pVM->pgm.s.CTXMID(p,PaePML4)->a[iPML4E].n.u1Present)
     2956                {
     2957                    /* Shadow PML4 present, so free all pdpt & pd entries. */
     2958                    for (iPDPTE = 0; iPDPTE < ELEMENTS(pPdptDst->a); iPDPTE++)
     2959                    {
     2960                        if (pPdptDst->a[iPDPTE].n.u1Present)
     2961                        {
     2962                            GCPtr = (iPML4E << X86_PML4_SHIFT) || (iPDPTE << X86_PDPT_SHIFT);
     2963
     2964                            rc = PGMShwGetLongModePDPtr(pVM, GCPtr, &pPdptDst, &pPDDst);
     2965                            if (rc != VINF_SUCCESS)
     2966                            {
     2967                                AssertMsg(rc == VERR_PAGE_TABLE_NOT_PRESENT, ("Unexpected rc=%Vrc\n", rc));
     2968                                return rc;
     2969                            }
     2970
     2971                            for (unsigned iPD = 0; iPD < ELEMENTS(pPDDst->a); iPD++)
     2972                            {
     2973                                if (   pPDDst->a[iPD].n.u1Present
     2974                                    && !(pPDDst->a[iPD].u & PGM_PDFLAGS_MAPPING))
     2975                                {
     2976                                    pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDDst->a[iPD].u & SHW_PDE_PG_MASK), PGMPOOL_IDX_PAE_PD, (iPML4E * X86_PG_PAE_ENTRIES + iPDPTE) * X86_PG_PAE_ENTRIES + iPD);
     2977                                    pPDDst->a[iPD].u = 0;
     2978                                }
     2979                            }
     2980
     2981                            pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPdptDst->a[iPDPTE].u & SHW_PDE_PG_MASK), PGMPOOL_IDX_PDPT, iPDPTE);
     2982                            pPdptDst->a[iPDPTE].u = 0;
     2983                        }
     2984                    }
     2985                }
     2986                pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pVM->pgm.s.CTXMID(p,PaePML4)->a[iPML4E].u & SHW_PDE_PG_MASK), PGMPOOL_IDX_PML4, iPML4E);
     2987                pVM->pgm.s.CTXMID(p,PaePML4)->a[iPML4E].n.u1Present = 0;
     2988                break;
     2989            }
    28552990#   endif
    2856         PX86PDEPAE      pPDEDst   = &pPDPAE->a[iPDPTE * X86_PG_PAE_ENTRIES];
    2857         PGSTPD          pPDSrc    = pgmGstGetPaePDPtr(&pVM->pgm.s, iPDPTE << X86_PDPT_SHIFT, &iPDSrc);
    2858 
    2859         if (pPDSrc == NULL)
    2860         {
    2861             /* PDPT not present */
    2862             if (pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].n.u1Present)
    2863             {
    2864                 for (unsigned iPD = 0; iPD < ELEMENTS(pPDSrc->a); iPD++)
     2991            Assert(iPDSrc == 0);
     2992
     2993            if (pPDSrc == NULL)
     2994            {
     2995                /* PDPE not present */
     2996                if (pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].n.u1Present)
    28652997                {
    2866                     if (   pPDEDst[iPD].n.u1Present
    2867                         && !(pPDEDst[iPD].u & PGM_PDFLAGS_MAPPING))
    2868                     {
    2869                         pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDEDst[iPD].u & SHW_PDE_PG_MASK), SHW_POOL_ROOT_IDX, iPDPTE * X86_PG_PAE_ENTRIES + iPD);
    2870                         pPDEDst[iPD].u = 0;
     2998                    /* for each page directory entry */
     2999                    for (unsigned iPD = 0; iPD < ELEMENTS(pPDSrc->a); iPD++)
     3000                    {
     3001                        if (   pPDEDst[iPD].n.u1Present
     3002                            && !(pPDEDst[iPD].u & PGM_PDFLAGS_MAPPING))
     3003                        {
     3004#   if PGM_GST_TYPE == PGM_TYPE_AMD64
     3005                            pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDEDst[iPD].u & SHW_PDE_PG_MASK), PGMPOOL_IDX_PAE_PD, (iPML4E * X86_PG_PAE_ENTRIES + iPDPTE) * X86_PG_PAE_ENTRIES + iPD);
     3006#   else
     3007                            pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDEDst[iPD].u & SHW_PDE_PG_MASK), PGMPOOL_IDX_PAE_PD, iPDPTE * X86_PG_PAE_ENTRIES + iPD);
     3008#   endif
     3009                            pPDEDst[iPD].u = 0;
     3010                        }
    28713011                    }
    28723012                }
     3013                if (!(pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].u & PGM_PLXFLAGS_MAPPING))
     3014                    pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].n.u1Present = 0;
     3015                continue;
    28733016            }
    2874             if (!(pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].u & PGM_PLXFLAGS_MAPPING))
    2875                 pVM->pgm.s.CTXMID(p,PaePDPT)->a[iPDPTE].n.u1Present = 0;
    2876             continue;
    2877         }
    28783017#  else  /* PGM_GST_TYPE != PGM_TYPE_PAE && PGM_GST_TYPE != PGM_TYPE_AMD64 */
    2879     {
     3018        {
    28803019#  endif /* PGM_GST_TYPE != PGM_TYPE_PAE && PGM_GST_TYPE != PGM_TYPE_AMD64 */
    2881         for (unsigned iPD = 0; iPD < ELEMENTS(pPDSrc->a); iPD++)
    2882         {
     3020            for (unsigned iPD = 0; iPD < ELEMENTS(pPDSrc->a); iPD++)
     3021            {
    28833022#  if PGM_SHW_TYPE == PGM_TYPE_32BIT
    2884             Assert(&pVM->pgm.s.CTXMID(p,32BitPD)->a[iPD] == pPDEDst);
     3023                Assert(&pVM->pgm.s.CTXMID(p,32BitPD)->a[iPD] == pPDEDst);
    28853024#  elif PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
    2886             AssertMsg(&pVM->pgm.s.CTXMID(ap,PaePDs)[iPD * 2 / 512]->a[iPD * 2 % 512] == pPDEDst, ("%p vs %p\n", &pVM->pgm.s.CTXMID(ap,PaePDs)[iPD * 2 / 512]->a[iPD * 2 % 512], pPDEDst));
     3025                AssertMsg(&pVM->pgm.s.CTXMID(ap,PaePDs)[iPD * 2 / 512]->a[iPD * 2 % 512] == pPDEDst, ("%p vs %p\n", &pVM->pgm.s.CTXMID(ap,PaePDs)[iPD * 2 / 512]->a[iPD * 2 % 512], pPDEDst));
    28873026#  endif
    2888             register GSTPDE PdeSrc = pPDSrc->a[iPD];
    2889             if (    PdeSrc.n.u1Present
    2890                 &&  (PdeSrc.n.u1User || fRawR0Enabled))
    2891             {
     3027                register GSTPDE PdeSrc = pPDSrc->a[iPD];
     3028                if (    PdeSrc.n.u1Present
     3029                    &&  (PdeSrc.n.u1User || fRawR0Enabled))
     3030                {
    28923031#  if    (   PGM_GST_TYPE == PGM_TYPE_32BIT \
    28933032          || PGM_GST_TYPE == PGM_TYPE_PAE) \
    28943033      && !defined(PGM_WITHOUT_MAPPINGS)
    28953034
    2896                 /*
    2897                  * Check for conflicts with GC mappings.
    2898                  */
     3035                    /*
     3036                    * Check for conflicts with GC mappings.
     3037                    */
    28993038#   if PGM_GST_TYPE == PGM_TYPE_PAE
    2900                 if (iPD + iPDPTE * X86_PG_PAE_ENTRIES == iPdNoMapping)
     3039                    if (iPD + iPDPTE * X86_PG_PAE_ENTRIES == iPdNoMapping)
    29013040#   else
    2902                 if (iPD == iPdNoMapping)
     3041                    if (iPD == iPdNoMapping)
    29033042#   endif
    2904                 {
    2905                     if (pVM->pgm.s.fMappingsFixed)
    2906                     {
    2907                         /* It's fixed, just skip the mapping. */
    2908                         const unsigned cPTs = pMapping->cb >> GST_PD_SHIFT;
    2909                         iPD += cPTs - 1;
    2910                         pPDEDst += cPTs + (PGM_GST_TYPE != PGM_SHW_TYPE) * cPTs;    /* Only applies to the pae shadow and 32 bits guest case */
    2911                         pMapping = pMapping->CTXALLSUFF(pNext);
    2912                         iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
    2913                         continue;
    2914                     }
     3043                    {
     3044                        if (pVM->pgm.s.fMappingsFixed)
     3045                        {
     3046                            /* It's fixed, just skip the mapping. */
     3047                            const unsigned cPTs = pMapping->cb >> GST_PD_SHIFT;
     3048                            iPD += cPTs - 1;
     3049                            pPDEDst += cPTs + (PGM_GST_TYPE != PGM_SHW_TYPE) * cPTs;    /* Only applies to the pae shadow and 32 bits guest case */
     3050                            pMapping = pMapping->CTXALLSUFF(pNext);
     3051                            iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
     3052                            continue;
     3053                        }
    29153054#   ifdef IN_RING3
    29163055#    if PGM_GST_TYPE == PGM_TYPE_32BIT
    2917                     int rc = pgmR3SyncPTResolveConflict(pVM, pMapping, pPDSrc, iPD << GST_PD_SHIFT);
     3056                        int rc = pgmR3SyncPTResolveConflict(pVM, pMapping, pPDSrc, iPD << GST_PD_SHIFT);
    29183057#    elif PGM_GST_TYPE == PGM_TYPE_PAE
    2919                     int rc = pgmR3SyncPTResolveConflictPAE(pVM, pMapping, (iPDPTE << GST_PDPT_SHIFT) + (iPD << GST_PD_SHIFT));
     3058                        int rc = pgmR3SyncPTResolveConflictPAE(pVM, pMapping, (iPDPTE << GST_PDPT_SHIFT) + (iPD << GST_PD_SHIFT));
    29203059#    endif
    2921                     if (VBOX_FAILURE(rc))
    2922                         return rc;
    2923 
     3060                        if (VBOX_FAILURE(rc))
     3061                            return rc;
     3062
     3063                        /*
     3064                        * Update iPdNoMapping and pMapping.
     3065                        */
     3066                        pMapping = pVM->pgm.s.pMappingsR3;
     3067                        while (pMapping && pMapping->GCPtr < (iPD << GST_PD_SHIFT))
     3068                            pMapping = pMapping->pNextR3;
     3069                        iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
     3070#   else
     3071                        LogFlow(("SyncCR3: detected conflict -> VINF_PGM_SYNC_CR3\n"));
     3072                        return VINF_PGM_SYNC_CR3;
     3073#   endif
     3074                    }
     3075#  else  /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
     3076                    Assert(!pgmMapAreMappingsEnabled(&pVM->pgm.s));
     3077#  endif /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
    29243078                    /*
    2925                      * Update iPdNoMapping and pMapping.
    2926                      */
    2927                     pMapping = pVM->pgm.s.pMappingsR3;
    2928                     while (pMapping && pMapping->GCPtr < (iPD << GST_PD_SHIFT))
    2929                         pMapping = pMapping->pNextR3;
    2930                     iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
    2931 #   else
    2932                     LogFlow(("SyncCR3: detected conflict -> VINF_PGM_SYNC_CR3\n"));
    2933                     return VINF_PGM_SYNC_CR3;
    2934 #   endif
     3079                    * Sync page directory entry.
     3080                    *
     3081                    * The current approach is to allocated the page table but to set
     3082                    * the entry to not-present and postpone the page table synching till
     3083                    * it's actually used.
     3084                    */
     3085#  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
     3086                    for (unsigned i = 0, iPdShw = iPD * 2; i < 2; i++, iPdShw++) /* pray that the compiler unrolls this */
     3087#  elif PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
     3088                    const unsigned iPdShw = iPD + iPDPTE * X86_PG_PAE_ENTRIES; NOREF(iPdShw);
     3089#  else
     3090                    const unsigned iPdShw = iPD; NOREF(iPdShw);
     3091#  endif
     3092                    {
     3093                        SHWPDE PdeDst = *pPDEDst;
     3094                        if (PdeDst.n.u1Present)
     3095                        {
     3096                            PPGMPOOLPAGE pShwPage = pgmPoolGetPage(pPool, PdeDst.u & SHW_PDE_PG_MASK);
     3097                            RTGCPHYS     GCPhys;
     3098                            if (    !PdeSrc.b.u1Size
     3099                                ||  !(cr4 & X86_CR4_PSE))
     3100                            {
     3101                                GCPhys = PdeSrc.u & GST_PDE_PG_MASK;
     3102#  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
     3103                                /* Select the right PDE as we're emulating a 4kb page table with 2 shadow page tables. */
     3104                                GCPhys |= i * (PAGE_SIZE / 2);
     3105#  endif
     3106                            }
     3107                            else
     3108                            {
     3109                                GCPhys = PdeSrc.u & GST_PDE_BIG_PG_MASK;
     3110#  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
     3111                                /* Select the right PDE as we're emulating a 4MB page directory with two 2 MB shadow PDEs.*/
     3112                                GCPhys |= i * X86_PAGE_2M_SIZE;
     3113#  endif
     3114                            }
     3115
     3116                            if (    pShwPage->GCPhys == GCPhys
     3117                                &&  pShwPage->enmKind == PGM_BTH_NAME(CalcPageKind)(&PdeSrc, cr4)
     3118                                &&  (   pShwPage->fCached
     3119                                    || (   !fGlobal
     3120                                        && (   false
     3121#  ifdef PGM_SKIP_GLOBAL_PAGEDIRS_ON_NONGLOBAL_FLUSH
     3122                                            || (   (PdeSrc.u & (X86_PDE4M_PS | X86_PDE4M_G)) == (X86_PDE4M_PS | X86_PDE4M_G)
     3123                                                && (cr4 & (X86_CR4_PGE | X86_CR4_PSE)) == (X86_CR4_PGE | X86_CR4_PSE)) /* global 2/4MB page. */
     3124                                            || (  !pShwPage->fSeenNonGlobal
     3125                                                && (cr4 & X86_CR4_PGE))
     3126#  endif
     3127                                            )
     3128                                        )
     3129                                    )
     3130                                &&  (   (PdeSrc.u & (X86_PDE_US | X86_PDE_RW)) == (PdeDst.u & (X86_PDE_US | X86_PDE_RW))
     3131                                    || (   (cr4 & X86_CR4_PSE)
     3132                                        &&     ((PdeSrc.u & (X86_PDE_US | X86_PDE4M_PS | X86_PDE4M_D)) | PGM_PDFLAGS_TRACK_DIRTY)
     3133                                            ==  ((PdeDst.u & (X86_PDE_US | X86_PDE_RW | PGM_PDFLAGS_TRACK_DIRTY)) | X86_PDE4M_PS))
     3134                                    )
     3135                            )
     3136                            {
     3137#  ifdef VBOX_WITH_STATISTICS
     3138                                if (   !fGlobal
     3139                                    && (PdeSrc.u & (X86_PDE4M_PS | X86_PDE4M_G)) == (X86_PDE4M_PS | X86_PDE4M_G)
     3140                                    && (cr4 & (X86_CR4_PGE | X86_CR4_PSE)) == (X86_CR4_PGE | X86_CR4_PSE))
     3141                                    MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstSkippedGlobalPD));
     3142                                else if (!fGlobal && !pShwPage->fSeenNonGlobal && (cr4 & X86_CR4_PGE))
     3143                                    MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstSkippedGlobalPT));
     3144                                else
     3145                                    MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstCacheHit));
     3146#  endif /* VBOX_WITH_STATISTICS */
     3147        /** @todo a replacement strategy isn't really needed unless we're using a very small pool < 512 pages.
     3148        * The whole ageing stuff should be put in yet another set of #ifdefs. For now, let's just skip it. */
     3149        //#  ifdef PGMPOOL_WITH_CACHE
     3150        //                        pgmPoolCacheUsed(pPool, pShwPage);
     3151        //#  endif
     3152                            }
     3153                            else
     3154                            {
     3155                                pgmPoolFreeByPage(pPool, pShwPage, SHW_POOL_ROOT_IDX, iPdShw);
     3156                                pPDEDst->u = 0;
     3157                                MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstFreed));
     3158                            }
     3159                        }
     3160                        else
     3161                            MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstNotPresent));
     3162                        pPDEDst++;
     3163                    }
    29353164                }
    2936 #  else  /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
    2937                 Assert(!pgmMapAreMappingsEnabled(&pVM->pgm.s));
    2938 #  endif /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
    2939                 /*
    2940                  * Sync page directory entry.
    2941                  *
    2942                  * The current approach is to allocated the page table but to set
    2943                  * the entry to not-present and postpone the page table synching till
    2944                  * it's actually used.
    2945                  */
    2946 #  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
    2947                 for (unsigned i = 0, iPdShw = iPD * 2; i < 2; i++, iPdShw++) /* pray that the compiler unrolls this */
    2948 #  elif PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
    2949                 const unsigned iPdShw = iPD + iPDPTE * X86_PG_PAE_ENTRIES; NOREF(iPdShw);
     3165#  if PGM_GST_TYPE == PGM_TYPE_PAE
     3166                else if (iPD + iPDPTE * X86_PG_PAE_ENTRIES != iPdNoMapping)
    29503167#  else
    2951                 const unsigned iPdShw = iPD; NOREF(iPdShw);
     3168                else if (iPD != iPdNoMapping)
    29523169#  endif
    29533170                {
    2954                     SHWPDE PdeDst = *pPDEDst;
    2955                     if (PdeDst.n.u1Present)
    2956                     {
    2957                         PPGMPOOLPAGE pShwPage = pgmPoolGetPage(pPool, PdeDst.u & SHW_PDE_PG_MASK);
    2958                         RTGCPHYS     GCPhys;
    2959                         if (    !PdeSrc.b.u1Size
    2960                             ||  !(cr4 & X86_CR4_PSE))
     3171                    /*
     3172                    * Check if there is any page directory to mark not present here.
     3173                    */
     3174#  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
     3175                    for (unsigned i = 0, iPdShw = iPD * 2; i < 2; i++, iPdShw++) /* pray that the compiler unrolls this */
     3176#  elif PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
     3177                    const unsigned iPdShw = iPD + iPDPTE * X86_PG_PAE_ENTRIES; NOREF(iPdShw);
     3178#  else
     3179                    const unsigned iPdShw = iPD; NOREF(iPdShw);
     3180#  endif
     3181                    {
     3182                        if (pPDEDst->n.u1Present)
    29613183                        {
    2962                             GCPhys = PdeSrc.u & GST_PDE_PG_MASK;
    2963 #  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
    2964                             /* Select the right PDE as we're emulating a 4kb page table with 2 shadow page tables. */
    2965                             GCPhys |= i * (PAGE_SIZE / 2);
    2966 #  endif
     3184                            pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDEDst->u & SHW_PDE_PG_MASK), SHW_POOL_ROOT_IDX, iPdShw);
     3185                            pPDEDst->u = 0;
     3186                            MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstFreedSrcNP));
    29673187                        }
    2968                         else
    2969                         {
    2970                             GCPhys = PdeSrc.u & GST_PDE_BIG_PG_MASK;
    2971 #  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
    2972                             /* Select the right PDE as we're emulating a 4MB page directory with two 2 MB shadow PDEs.*/
    2973                             GCPhys |= i * X86_PAGE_2M_SIZE;
    2974 #  endif
    2975                         }
    2976 
    2977                         if (    pShwPage->GCPhys == GCPhys
    2978                             &&  pShwPage->enmKind == PGM_BTH_NAME(CalcPageKind)(&PdeSrc, cr4)
    2979                             &&  (   pShwPage->fCached
    2980                                 || (   !fGlobal
    2981                                     && (   false
    2982 #  ifdef PGM_SKIP_GLOBAL_PAGEDIRS_ON_NONGLOBAL_FLUSH
    2983                                         || (   (PdeSrc.u & (X86_PDE4M_PS | X86_PDE4M_G)) == (X86_PDE4M_PS | X86_PDE4M_G)
    2984                                             && (cr4 & (X86_CR4_PGE | X86_CR4_PSE)) == (X86_CR4_PGE | X86_CR4_PSE)) /* global 2/4MB page. */
    2985                                         || (  !pShwPage->fSeenNonGlobal
    2986                                             && (cr4 & X86_CR4_PGE))
    2987 #  endif
    2988                                         )
    2989                                     )
    2990                                 )
    2991                             &&  (   (PdeSrc.u & (X86_PDE_US | X86_PDE_RW)) == (PdeDst.u & (X86_PDE_US | X86_PDE_RW))
    2992                                 || (   (cr4 & X86_CR4_PSE)
    2993                                     &&     ((PdeSrc.u & (X86_PDE_US | X86_PDE4M_PS | X86_PDE4M_D)) | PGM_PDFLAGS_TRACK_DIRTY)
    2994                                         ==  ((PdeDst.u & (X86_PDE_US | X86_PDE_RW | PGM_PDFLAGS_TRACK_DIRTY)) | X86_PDE4M_PS))
    2995                                 )
    2996                         )
    2997                         {
    2998 #  ifdef VBOX_WITH_STATISTICS
    2999                             if (   !fGlobal
    3000                                 && (PdeSrc.u & (X86_PDE4M_PS | X86_PDE4M_G)) == (X86_PDE4M_PS | X86_PDE4M_G)
    3001                                 && (cr4 & (X86_CR4_PGE | X86_CR4_PSE)) == (X86_CR4_PGE | X86_CR4_PSE))
    3002                                 MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstSkippedGlobalPD));
    3003                             else if (!fGlobal && !pShwPage->fSeenNonGlobal && (cr4 & X86_CR4_PGE))
    3004                                 MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstSkippedGlobalPT));
    3005                             else
    3006                                 MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstCacheHit));
    3007 #  endif /* VBOX_WITH_STATISTICS */
    3008     /** @todo a replacement strategy isn't really needed unless we're using a very small pool < 512 pages.
    3009     * The whole ageing stuff should be put in yet another set of #ifdefs. For now, let's just skip it. */
    3010     //#  ifdef PGMPOOL_WITH_CACHE
    3011     //                        pgmPoolCacheUsed(pPool, pShwPage);
    3012     //#  endif
    3013                         }
    3014                         else
    3015                         {
    3016                             pgmPoolFreeByPage(pPool, pShwPage, SHW_POOL_ROOT_IDX, iPdShw);
    3017                             pPDEDst->u = 0;
    3018                             MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstFreed));
    3019                         }
    3020                     }
    3021                     else
    3022                         MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstNotPresent));
    3023                     pPDEDst++;
     3188                        pPDEDst++;
     3189                    }
    30243190                }
    3025             }
    3026 #  if PGM_GST_TYPE == PGM_TYPE_PAE
    3027             else if (iPD + iPDPTE * X86_PG_PAE_ENTRIES != iPdNoMapping)
    3028 #  else
    3029             else if (iPD != iPdNoMapping)
    3030 #  endif
    3031             {
    3032                 /*
    3033                  * Check if there is any page directory to mark not present here.
    3034                  */
    3035 #  if PGM_SHW_TYPE == PGM_TYPE_PAE && PGM_GST_TYPE == PGM_TYPE_32BIT
    3036                 for (unsigned i = 0, iPdShw = iPD * 2; i < 2; i++, iPdShw++) /* pray that the compiler unrolls this */
    3037 #  elif PGM_GST_TYPE == PGM_TYPE_PAE || PGM_GST_TYPE == PGM_TYPE_AMD64
    3038                 const unsigned iPdShw = iPD + iPDPTE * X86_PG_PAE_ENTRIES; NOREF(iPdShw);
    3039 #  else
    3040                 const unsigned iPdShw = iPD; NOREF(iPdShw);
    3041 #  endif
     3191                else
    30423192                {
    3043                     if (pPDEDst->n.u1Present)
    3044                     {
    3045                         pgmPoolFreeByPage(pPool, pgmPoolGetPage(pPool, pPDEDst->u & SHW_PDE_PG_MASK), SHW_POOL_ROOT_IDX, iPdShw);
    3046                         pPDEDst->u = 0;
    3047                         MY_STAM_COUNTER_INC(&pVM->pgm.s.CTXMID(Stat,SyncCR3DstFreedSrcNP));
    3048                     }
    3049                     pPDEDst++;
    3050                 }
    3051             }
    3052             else
    3053             {
    30543193#  if    (   PGM_GST_TYPE == PGM_TYPE_32BIT \
    30553194          || PGM_GST_TYPE == PGM_TYPE_PAE)  \
    30563195      && !defined(PGM_WITHOUT_MAPPINGS)
    30573196
    3058                 const unsigned cPTs = pMapping->cb >> GST_PD_SHIFT;
    3059 
    3060                 Assert(pgmMapAreMappingsEnabled(&pVM->pgm.s));
    3061                 if (pVM->pgm.s.fMappingsFixed)
    3062                 {
    3063                     /* It's fixed, just skip the mapping. */
    3064                     pMapping = pMapping->CTXALLSUFF(pNext);
    3065                     iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
    3066                 }
    3067                 else
    3068                 {
    3069                     /*
    3070                      * Check for conflicts for subsequent pagetables
    3071                      * and advance to the next mapping.
    3072                      */
    3073                     iPdNoMapping = ~0U;
    3074                     unsigned iPT = cPTs;
    3075                     while (iPT-- > 1)
    3076                     {
    3077                         if (    pPDSrc->a[iPD + iPT].n.u1Present
    3078                             &&  (pPDSrc->a[iPD + iPT].n.u1User || fRawR0Enabled))
     3197                    const unsigned cPTs = pMapping->cb >> GST_PD_SHIFT;
     3198
     3199                    Assert(pgmMapAreMappingsEnabled(&pVM->pgm.s));
     3200                    if (pVM->pgm.s.fMappingsFixed)
     3201                    {
     3202                        /* It's fixed, just skip the mapping. */
     3203                        pMapping = pMapping->CTXALLSUFF(pNext);
     3204                        iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
     3205                    }
     3206                    else
     3207                    {
     3208                        /*
     3209                        * Check for conflicts for subsequent pagetables
     3210                        * and advance to the next mapping.
     3211                        */
     3212                        iPdNoMapping = ~0U;
     3213                        unsigned iPT = cPTs;
     3214                        while (iPT-- > 1)
    30793215                        {
     3216                            if (    pPDSrc->a[iPD + iPT].n.u1Present
     3217                                &&  (pPDSrc->a[iPD + iPT].n.u1User || fRawR0Enabled))
     3218                            {
    30803219#   ifdef IN_RING3
    30813220#    if PGM_GST_TYPE == PGM_TYPE_32BIT
    3082                             int rc = pgmR3SyncPTResolveConflict(pVM, pMapping, pPDSrc, iPD << GST_PD_SHIFT);
     3221                                int rc = pgmR3SyncPTResolveConflict(pVM, pMapping, pPDSrc, iPD << GST_PD_SHIFT);
    30833222#    elif PGM_GST_TYPE == PGM_TYPE_PAE
    3084                             int rc = pgmR3SyncPTResolveConflictPAE(pVM, pMapping, (iPDPTE << GST_PDPT_SHIFT) + (iPD << GST_PD_SHIFT));
     3223                                int rc = pgmR3SyncPTResolveConflictPAE(pVM, pMapping, (iPDPTE << GST_PDPT_SHIFT) + (iPD << GST_PD_SHIFT));
    30853224#    endif
    3086                             if (VBOX_FAILURE(rc))
    3087                                 return rc;
    3088 
    3089                             /*
    3090                              * Update iPdNoMapping and pMapping.
    3091                              */
    3092                             pMapping = pVM->pgm.s.CTXALLSUFF(pMappings);
    3093                             while (pMapping && pMapping->GCPtr < (iPD << GST_PD_SHIFT))
    3094                                 pMapping = pMapping->CTXALLSUFF(pNext);
    3095                             iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
    3096                             break;
     3225                                if (VBOX_FAILURE(rc))
     3226                                    return rc;
     3227
     3228                                /*
     3229                                * Update iPdNoMapping and pMapping.
     3230                                */
     3231                                pMapping = pVM->pgm.s.CTXALLSUFF(pMappings);
     3232                                while (pMapping && pMapping->GCPtr < (iPD << GST_PD_SHIFT))
     3233                                    pMapping = pMapping->CTXALLSUFF(pNext);
     3234                                iPdNoMapping = pMapping ? pMapping->GCPtr >> GST_PD_SHIFT : ~0U;
     3235                                break;
    30973236#   else
    3098                             LogFlow(("SyncCR3: detected conflict -> VINF_PGM_SYNC_CR3\n"));
    3099                             return VINF_PGM_SYNC_CR3;
     3237                                LogFlow(("SyncCR3: detected conflict -> VINF_PGM_SYNC_CR3\n"));
     3238                                return VINF_PGM_SYNC_CR3;
    31003239#   endif
     3240                            }
    31013241                        }
    3102                     }
    3103                     if (iPdNoMapping == ~0U && pMapping)
    3104                     {
    3105                         pMapping = pMapping->CTXALLSUFF(pNext);
    3106                         if (pMapping)
    3107                             iPdNoMapping = pMapping->GCPtr >> GST_PD_SHIFT;
    3108                     }
    3109                 }
    3110 
    3111                 /* advance. */
    3112                 iPD += cPTs - 1;
    3113                 pPDEDst += cPTs + (PGM_GST_TYPE != PGM_SHW_TYPE) * cPTs;    /* Only applies to the pae shadow and 32 bits guest case */
     3242                        if (iPdNoMapping == ~0U && pMapping)
     3243                        {
     3244                            pMapping = pMapping->CTXALLSUFF(pNext);
     3245                            if (pMapping)
     3246                                iPdNoMapping = pMapping->GCPtr >> GST_PD_SHIFT;
     3247                        }
     3248                    }
     3249
     3250                    /* advance. */
     3251                    iPD += cPTs - 1;
     3252                    pPDEDst += cPTs + (PGM_GST_TYPE != PGM_SHW_TYPE) * cPTs;    /* Only applies to the pae shadow and 32 bits guest case */
    31143253#   if PGM_GST_TYPE != PGM_SHW_TYPE
    3115                 AssertCompile(PGM_GST_TYPE == PGM_TYPE_32BIT && PGM_SHW_TYPE == PGM_TYPE_PAE);
     3254                    AssertCompile(PGM_GST_TYPE == PGM_TYPE_32BIT && PGM_SHW_TYPE == PGM_TYPE_PAE);
    31163255#   endif
    31173256#  else  /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
    3118                 Assert(!pgmMapAreMappingsEnabled(&pVM->pgm.s));
     3257                    Assert(!pgmMapAreMappingsEnabled(&pVM->pgm.s));
    31193258#  endif /* (PGM_GST_TYPE != PGM_TYPE_32BIT && PGM_GST_TYPE != PGM_TYPE_PAE) || PGM_WITHOUT_MAPPINGS */
    3120             }
    3121 
    3122         } /* for iPD */
    3123     } /* for each PDPTE (PAE) */
    3124 
     3259                }
     3260
     3261            } /* for iPD */
     3262        } /* for each PDPTE (PAE) */
     3263    } /* for each page map level 4 entry (amd64) */
    31253264    return VINF_SUCCESS;
    31263265
    3127 # elif PGM_GST_TYPE == PGM_TYPE_AMD64
    3128 //# error not implemented
    3129     return VERR_INTERNAL_ERROR;
    31303266# else /* guest real and protected mode */
    31313267    return VINF_SUCCESS;
Note: See TracChangeset for help on using the changeset viewer.

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