- Timestamp:
- Apr 13, 2021 5:46:02 AM (4 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Devices/Bus/DevIommuIntel.cpp
r88427 r88481 455 455 * @param uSagaw The CAP_REG.SAGAW value. 456 456 */ 457 static uint8_t vtd GetSupGstAddrBits(uint8_t uSagaw)457 static uint8_t vtdCapRegGetSagawBits(uint8_t uSagaw) 458 458 { 459 459 if (RT_LIKELY(uSagaw > 0 && uSagaw < 4)) … … 467 467 * address width (MGAW). 468 468 * 469 * @returns The CAP .SAGAW value.469 * @returns The CAP_REG.SAGAW value. 470 470 * @param uMgaw The CAP_REG.MGAW value. 471 471 */ 472 static uint8_t vtd GetSupGstAddrWidth(uint8_t uMgaw)472 static uint8_t vtdCapRegGetSagaw(uint8_t uMgaw) 473 473 { 474 474 switch (uMgaw + 1) … … 783 783 { 784 784 if (cb == 8) 785 { 785 786 *(uint64_t *)pv = dmarRegRead64(pThis, offReg); 787 LogFlowFunc(("offReg=%#x pv=%#RX64\n", offReg, *(uint64_t *)pv)); 788 } 786 789 else 790 { 787 791 *(uint32_t *)pv = dmarRegRead32(pThis, offReg); 788 789 LogFlowFunc(("offReg=%#x\n", offReg)); 792 LogFlowFunc(("offReg=%#x pv=%#RX32\n", offReg, *(uint32_t *)pv)); 793 } 794 790 795 return VINF_SUCCESS; 791 796 } … … 825 830 /* CAP_REG */ 826 831 { 827 uint8_t const fFl1gp = 1; /* First-Level 1GB pages support. */828 uint8_t const fFl5lp = 1; /* First-level 5-level paging support (PML5E). */829 uint8_t const fSl2mp = fSlts & 1; /* Second-Level 2MB pages support. */830 uint8_t const fSl2gp = fSlts & 1; /* Second-Level 1GB pages support. */831 uint8_t const fSllps = fSl2mp /* Second-Level large page Support. */832 | ((fSl2mp & fFl1gp) & RT_BIT(1));833 uint8_t const fMamv = (fSl2gp ? /* Maximum address mask value (for second-level invalidations). */834 X86_PAGE_1G_SHIFT : X86_PAGE_2M_SHIFT) - X86_PAGE_4K_SHIFT;835 uint8_t const fNd = 2; /* Number of domains (0=16, 1=64, 2=256, 3=1K, 4=4K, 5=16K, 6=64K,836 7=Reserved). */837 uint8_t const fPsi = 1; /* Page selective invalidation. */838 832 uint8_t cGstPhysAddrBits; 839 833 uint8_t cGstLinearAddrBits; 840 834 PDMDevHlpCpuGetGuestAddrWidths(pDevIns, &cGstPhysAddrBits, &cGstLinearAddrBits); 841 NOREF(cGstLinearAddrBits); 842 uint8_t const uMgaw = cGstPhysAddrBits - 1; /* Maximum guest address width. */ 843 uint8_t const uSagaw = vtdGetSupGstAddrWidth(uMgaw); /* Supported adjust guest address width. */ 835 836 uint8_t const fFl1gp = 1; /* First-Level 1GB pages support. */ 837 uint8_t const fFl5lp = 1; /* First-level 5-level paging support (PML5E). */ 838 uint8_t const fSl2mp = fSlts & 1; /* Second-Level 2MB pages support. */ 839 uint8_t const fSl2gp = fSlts & 1; /* Second-Level 1GB pages support. */ 840 uint8_t const fSllps = fSl2mp /* Second-Level large page Support. */ 841 | ((fSl2mp & fFl1gp) & RT_BIT(1)); 842 uint8_t const fMamv = (fSl2gp ? /* Maximum address mask value (for second-level invalidations). */ 843 X86_PAGE_1G_SHIFT : X86_PAGE_2M_SHIFT) - X86_PAGE_4K_SHIFT; 844 uint8_t const fNd = 2; /* Number of domains (0=16, 1=64, 2=256, 3=1K, 4=4K, 5=16K, 6=64K, 845 7=Reserved). */ 846 uint8_t const fPsi = 1; /* Page selective invalidation. */ 847 uint8_t const uMgaw = cGstPhysAddrBits - 1; /* Maximum guest address width. */ 848 uint8_t const uSagaw = vtdCapRegGetSagaw(uMgaw); /* Supported adjust guest address width. */ 849 uint16_t const offFro = DMAR_MMIO_OFF_FRCD_LO_REG >> 4; /* MMIO offset of FRCD registers. */ 844 850 845 851 pThis->fCap = RT_BF_MAKE(VTD_BF_CAP_REG_ND, fNd) … … 852 858 | RT_BF_MAKE(VTD_BF_CAP_REG_MGAW, uMgaw) 853 859 | RT_BF_MAKE(VTD_BF_CAP_REG_ZLR, 1) /** @todo Figure out if/how to support zero-length reads. */ 854 | RT_BF_MAKE(VTD_BF_CAP_REG_FRO, DMAR_MMIO_OFF_FRCD_LO_REG >> 4)860 | RT_BF_MAKE(VTD_BF_CAP_REG_FRO, offFro) 855 861 | RT_BF_MAKE(VTD_BF_CAP_REG_SLLPS, fSlts & fSllps) 856 862 | RT_BF_MAKE(VTD_BF_CAP_REG_PSI, fPsi) … … 866 872 /* ECAP_REG */ 867 873 { 868 uint8_t const fIr = 1; /* Interrupt remapping support. */ 869 uint8_t const fMhmv = 0xf; /* Maximum handle mask value. */ 870 uint16_t const offIro = DMAR_MMIO_OFF_IVA_REG >> 4; /* MMIO offset of IOTLB registers. */ 871 uint8_t const fSrs = 1; /* Supervisor request support. */ 872 uint8_t const fEim = 1; /* Extended interrupt mode.*/ 874 uint8_t const fQi = 1; /* Queued invalidations. */ 875 uint8_t const fIr = !!(DMAR_ACPI_DMAR_FLAGS & ACPI_DMAR_F_INTR_REMAP); /* Interrupt remapping support. */ 876 uint8_t const fMhmv = 0xf; /* Maximum handle mask value. */ 877 uint16_t const offIro = DMAR_MMIO_OFF_IVA_REG >> 4; /* MMIO offset of IOTLB registers. */ 878 uint8_t const fSrs = 1; /* Supervisor request support. */ 879 uint8_t const fEim = 1; /* Extended interrupt mode.*/ 873 880 874 881 pThis->fExtCap = RT_BF_MAKE(VTD_BF_ECAP_REG_C, 0) /* Accesses don't snoop CPU cache. */ 875 882 | RT_BF_MAKE(VTD_BF_ECAP_REG_QI, 1) 876 883 | RT_BF_MAKE(VTD_BF_ECAP_REG_DT, 0) /* Device-TLBs not supported. */ 877 | RT_BF_MAKE(VTD_BF_ECAP_REG_IR, f Ir)884 | RT_BF_MAKE(VTD_BF_ECAP_REG_IR, fQi & fIr) 878 885 | RT_BF_MAKE(VTD_BF_ECAP_REG_EIM, fIr & fEim) 879 886 | RT_BF_MAKE(VTD_BF_ECAP_REG_PT, fPt) … … 918 925 919 926 #ifdef VBOX_STRICT 920 uint64_t const fExtCap = dmarRegRead64(pThis, VTD_MMIO_OFF_ECAP_REG); 921 Assert(!RT_BF_GET(fExtCap, VTD_BF_ECAP_REG_PRS)); /* PECTL_REG - Reserved if don't support PRS. */ 922 Assert(!RT_BF_GET(fExtCap, VTD_BF_ECAP_REG_MTS)); /* MTRRCAP_REG - Reserved if we don't support MTS. */ 927 Assert(!RT_BF_GET(pThis->fExtCap, VTD_BF_ECAP_REG_PRS)); /* PECTL_REG - Reserved if don't support PRS. */ 928 Assert(!RT_BF_GET(pThis->fExtCap, VTD_BF_ECAP_REG_MTS)); /* MTRRCAP_REG - Reserved if we don't support MTS. */ 923 929 #endif 924 930 } … … 1065 1071 dmarR3RegsInit(pDevIns); 1066 1072 1067 uint8_t const cMaxGstAddrBits = RT_BF_GET(pThis->fCap, VTD_BF_CAP_REG_MGAW) + 1; 1068 uint8_t const cSupGstAddrBits = vtdGetSupGstAddrBits(RT_BF_GET(pThis->fCap, VTD_BF_CAP_REG_SAGAW)); 1069 LogRel(("%s: CAP=%#RX64 ECAP=%#RX64 (MGAW=%u bits, SAGAW=%u bits)\n", DMAR_LOG_PFX, pThis->fCap, pThis->fExtCap, 1070 cMaxGstAddrBits, cSupGstAddrBits)); 1073 /* 1074 * Log some of the features exposed to software. 1075 */ 1076 uint32_t const uVerReg = dmarRegRead32(pThis, VTD_MMIO_OFF_VER_REG); 1077 uint8_t const cMaxGstAddrBits = RT_BF_GET(pThis->fCap, VTD_BF_CAP_REG_MGAW) + 1; 1078 uint8_t const cSupGstAddrBits = vtdCapRegGetSagawBits(RT_BF_GET(pThis->fCap, VTD_BF_CAP_REG_SAGAW)); 1079 uint16_t const offFrcd = RT_BF_GET(pThis->fCap, VTD_BF_CAP_REG_FRO); 1080 uint16_t const offIva = RT_BF_GET(pThis->fExtCap, VTD_BF_ECAP_REG_IRO); 1081 LogRel(("%s: VER=%u.%u CAP=%#RX64 ECAP=%#RX64 (MGAW=%u bits, SAGAW=%u bits, FRO=%#x, IRO=%#x) mapped at %#RGp\n", DMAR_LOG_PFX, 1082 RT_BF_GET(uVerReg, VTD_BF_VER_REG_MAX), RT_BF_GET(uVerReg, VTD_BF_VER_REG_MIN), 1083 pThis->fCap, pThis->fExtCap, cMaxGstAddrBits, cSupGstAddrBits, offFrcd, offIva, DMAR_MMIO_BASE_PHYSADDR)); 1071 1084 return VINF_SUCCESS; 1072 1085 }
Note:
See TracChangeset
for help on using the changeset viewer.