/* $Id: VBoxAcpi.cpp 26185 2010-02-03 13:06:17Z vboxsync $ */ /** @file * VBoxAcpi - VirtualBox ACPI maniputation functionality. */ /* * Copyright (C) 2009 Sun Microsystems, Inc. * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa * Clara, CA 95054 USA or visit http://www.sun.com if you need * additional information or have any questions. */ #if !defined(IN_RING3) #error Pure R3 code #endif #define LOG_GROUP LOG_GROUP_DEV_ACPI #include #include #include #include #include #include #include #include #include #include #ifdef VBOX_WITH_DYNAMIC_DSDT /* vbox.dsl - input to generate proper DSDT on the fly */ # include #else /* Statically compiled AML */ # include # include # include #endif #ifdef VBOX_WITH_DYNAMIC_DSDT static int prepareDynamicDsdt(PPDMDEVINS pDevIns, void* *ppPtr, size_t *puDsdtLen) { *ppPtr = NULL; *puDsdtLen = 0; return 0; } static int cleanupDynamicDsdt(PPDMDEVINS pDevIns, void* pPtr) { return 0; } #else static int patchAml(PPDMDEVINS pDevIns, uint8_t* pAml, size_t uAmlLen) { uint16_t cNumCpus; int rc; rc = CFGMR3QueryU16Def(pDevIns->pCfg, "NumCPUs", &cNumCpus, 1); if (RT_FAILURE(rc)) return rc; /* Clear CPU objects at all, if needed */ bool fShowCpu; rc = CFGMR3QueryBoolDef(pDevIns->pCfg, "ShowCpu", &fShowCpu, false); if (RT_FAILURE(rc)) return rc; if (!fShowCpu) cNumCpus = 0; /** * Now search AML for: * AML_PROCESSOR_OP (UINT16) 0x5b83 * and replace whole block with * AML_NOOP_OP (UINT16) 0xa3 * for VCPU not configured */ for (uint32_t i = 0; i < uAmlLen - 7; i++) { /* * AML_PROCESSOR_OP * * DefProcessor := ProcessorOp PkgLength NameString ProcID PblkAddr PblkLen ObjectList * ProcessorOp := ExtOpPrefix 0x83 * ProcID := ByteData * PblkAddr := DwordData * PblkLen := ByteData */ if ((pAml[i] == 0x5b) && (pAml[i+1] == 0x83)) { if ((pAml[i+3] != 'C') || (pAml[i+4] != 'P')) /* false alarm, not named starting CP */ continue; /* Processor ID */ if (pAml[i+7] < cNumCpus) continue; /* Will fill unwanted CPU block with NOOPs */ /* * See 18.2.4 Package Length Encoding in ACPI spec * for full format */ uint32_t cBytes = pAml[i + 2]; AssertReleaseMsg((cBytes >> 6) == 0, ("So far, we only understand simple package length")); /* including AML_PROCESSOR_OP itself */ for (uint32_t j = 0; j < cBytes + 2; j++) pAml[i+j] = 0xa3; /* Can increase i by cBytes + 1, but not really worth it */ } } /* now recompute checksum, whole file byte sum must be 0 */ pAml[9] = 0; uint8_t aSum = 0; for (uint32_t i = 0; i < uAmlLen; i++) aSum = aSum + (uint8_t)pAml[i]; pAml[9] = (uint8_t) (0 - aSum); return 0; } #endif /** * Loads an AML file if present in CFGM * * @returns VBox status code * @param pDevIns The device instance * @param pcszCfgName The configuration key holding the file path * @param pcszSignature The signature to check for * @param ppbAmlCode Where to store the pointer to the AML code on success. * @param pcbAmlCode Where to store the number of bytes of the AML code on success. */ static int acpiAmlLoadExternal(PPDMDEVINS pDevIns, const char *pcszCfgName, const char *pcszSignature, uint8_t **ppbAmlCode, size_t *pcbAmlCode) { uint8_t *pbAmlCode = NULL; size_t cbAmlCode = 0; char *pszAmlFilePath = NULL; int rc = CFGMR3QueryStringAlloc(pDevIns->pCfg, pcszCfgName, &pszAmlFilePath); if (RT_SUCCESS(rc)) { /* Load from file. */ RTFILE FileAml = NIL_RTFILE; rc = RTFileOpen(&FileAml, pszAmlFilePath, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE); if (RT_SUCCESS(rc)) { /* * An AML file contains the raw DSDT thus the size of the file * is equal to the size of the DSDT. */ uint64_t cbAmlFile = 0; rc = RTFileGetSize(FileAml, &cbAmlFile); cbAmlCode = (size_t)cbAmlFile; /* Don't use AML files over 4GB ;) */ if ( RT_SUCCESS(rc) && ((uint64_t)cbAmlCode == cbAmlFile)) { pbAmlCode = (uint8_t *)RTMemAllocZ(cbAmlCode); if (pbAmlCode) { rc = RTFileReadAt(FileAml, 0, pbAmlCode, cbAmlCode, NULL); /* * We fail if reading failed or the identifier at the * beginning is wrong. */ if ( RT_FAILURE(rc) || strncmp((const char *)pbAmlCode, pcszSignature, 4)) { RTMemFree(pbAmlCode); pbAmlCode = NULL; /* Return error if file header check failed */ if (RT_SUCCESS(rc)) rc = VERR_PARSE_ERROR; } else { *ppbAmlCode = pbAmlCode; *pcbAmlCode = cbAmlCode; rc = VINF_SUCCESS; } } else rc = VERR_NO_MEMORY; } RTFileClose(FileAml); } MMR3HeapFree(pszAmlFilePath); } return rc; } /* Two only public functions */ int acpiPrepareDsdt(PPDMDEVINS pDevIns, void * *ppPtr, size_t *puDsdtLen) { #ifdef VBOX_WITH_DYNAMIC_DSDT return prepareDynamicDsdt(pDevIns, ppPtr, puDsdtLen); #else uint8_t *pbAmlCodeDsdt = NULL; size_t cbAmlCodeDsdt = 0; int rc = acpiAmlLoadExternal(pDevIns, "DsdtFilePath", "DSDT", &pbAmlCodeDsdt, &cbAmlCodeDsdt); if (rc == VERR_CFGM_VALUE_NOT_FOUND) { rc = VINF_SUCCESS; /* Use the compiled in AML code */ cbAmlCodeDsdt = sizeof(AmlCode); pbAmlCodeDsdt = (uint8_t *)RTMemAllocZ(cbAmlCodeDsdt); if (pbAmlCodeDsdt) memcpy(pbAmlCodeDsdt, AmlCode, cbAmlCodeDsdt); else rc = VERR_NO_MEMORY; } else if (RT_FAILURE(rc)) return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed to read \"DsdtFilePath\"")); if (RT_SUCCESS(rc)) { patchAml(pDevIns, pbAmlCodeDsdt, cbAmlCodeDsdt); *ppPtr = pbAmlCodeDsdt; *puDsdtLen = cbAmlCodeDsdt; } return rc; #endif } int acpiCleanupDsdt(PPDMDEVINS pDevIns, void * pPtr) { #ifdef VBOX_WITH_DYNAMIC_DSDT return cleanupDynamicDsdt(pDevIns, pPtr); #else if (pPtr) RTMemFree(pPtr); return VINF_SUCCESS; #endif } int acpiPrepareSsdt(PPDMDEVINS pDevIns, void* *ppPtr, size_t *puSsdtLen) { uint8_t *pbAmlCodeSsdt = NULL; size_t cbAmlCodeSsdt = 0; int rc = acpiAmlLoadExternal(pDevIns, "SsdtFilePath", "SSDT", &pbAmlCodeSsdt, &cbAmlCodeSsdt); if (rc == VERR_CFGM_VALUE_NOT_FOUND) { bool fCpuHotPlug = false; uint8_t *pbAmlCode = NULL; rc = CFGMR3QueryBoolDef(pDevIns->pCfg, "CpuHotPlug", &fCpuHotPlug, false); if (RT_FAILURE(rc)) return rc; if (fCpuHotPlug) { pbAmlCode = AmlCodeSsdtCpuHotPlug; cbAmlCodeSsdt = sizeof(AmlCodeSsdtCpuHotPlug); } else { pbAmlCode = AmlCodeSsdtStandard; cbAmlCodeSsdt = sizeof(AmlCodeSsdtStandard); } pbAmlCodeSsdt = (uint8_t *)RTMemAllocZ(cbAmlCodeSsdt); if (pbAmlCodeSsdt) memcpy(pbAmlCodeSsdt, pbAmlCode, cbAmlCodeSsdt); else rc = VERR_NO_MEMORY; } else if (RT_FAILURE(rc)) return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed to read \"SsdtFilePath\"")); if (RT_SUCCESS(rc)) { patchAml(pDevIns, pbAmlCodeSsdt, cbAmlCodeSsdt); *ppPtr = pbAmlCodeSsdt; *puSsdtLen = cbAmlCodeSsdt; } return VINF_SUCCESS; } int acpiCleanupSsdt(PPDMDEVINS pDevIns, void* pPtr) { if (pPtr) RTMemFree(pPtr); return VINF_SUCCESS; }