diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 16:49:04 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 16:49:04 +0000 |
commit | 16f504a9dca3fe3b70568f67b7d41241ae485288 (patch) | |
tree | c60f36ada0496ba928b7161059ba5ab1ab224f9d /src/VBox/Devices/PC/DevPcArch.cpp | |
parent | Initial commit. (diff) | |
download | virtualbox-upstream.tar.xz virtualbox-upstream.zip |
Adding upstream version 7.0.6-dfsg.upstream/7.0.6-dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/VBox/Devices/PC/DevPcArch.cpp')
-rw-r--r-- | src/VBox/Devices/PC/DevPcArch.cpp | 314 |
1 files changed, 314 insertions, 0 deletions
diff --git a/src/VBox/Devices/PC/DevPcArch.cpp b/src/VBox/Devices/PC/DevPcArch.cpp new file mode 100644 index 00000000..855522b7 --- /dev/null +++ b/src/VBox/Devices/PC/DevPcArch.cpp @@ -0,0 +1,314 @@ +/* $Id: DevPcArch.cpp $ */ +/** @file + * DevPcArch - PC Architecture Device. + */ + +/* + * Copyright (C) 2006-2022 Oracle and/or its affiliates. + * + * This file is part of VirtualBox base platform packages, as + * available from https://www.virtualbox.org. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, in version 3 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses>. + * + * SPDX-License-Identifier: GPL-3.0-only + */ + + +/********************************************************************************************************************************* +* Header Files * +*********************************************************************************************************************************/ +#define LOG_GROUP LOG_GROUP_DEV_PC_ARCH +#include <VBox/vmm/pdmdev.h> +#include <VBox/vmm/mm.h> +#include <VBox/vmm/pgm.h> +#include <VBox/log.h> +#include <VBox/err.h> +#include <iprt/assert.h> +#include <iprt/string.h> + +#include "VBoxDD.h" + + +/********************************************************************************************************************************* +* Structures and Typedefs * +*********************************************************************************************************************************/ + +/** + * PC Bios instance data structure. + */ +typedef struct DEVPCARCH +{ + /** Pointer back to the device instance. */ + PPDMDEVINS pDevIns; +} DEVPCARCH, *PDEVPCARCH; + + + +/** + * @callback_method_impl{FNIOMIOPORTNEWIN, Math coprocessor.} + * @note offPort is absolute + */ +static DECLCALLBACK(VBOXSTRICTRC) +pcarchIOPortFPURead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb) +{ + int rc; + NOREF(pvUser); NOREF(pDevIns); NOREF(pu32); + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d\n", offPort, cb); + if (rc == VINF_SUCCESS) + rc = VERR_IOM_IOPORT_UNUSED; + return rc; +} + +/** + * @callback_method_impl{FNIOMIOPORTNEWOUT, Math coprocessor.} + * @note offPort is absolute + * @todo Add IGNNE support. + */ +static DECLCALLBACK(VBOXSTRICTRC) +pcarchIOPortFPUWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb) +{ + int rc = VINF_SUCCESS; + NOREF(pvUser); + if (cb == 1) + { + switch (offPort) + { + /* + * Clear busy latch. + */ + case 0xf0: + Log2(("PCARCH: FPU Clear busy latch u32=%#x\n", u32)); +/* This is triggered when booting Knoppix (3.7) */ +#if 0 + if (!u32) + rc = PDMDeviceDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d u32=%#x\n", offPort, cb, u32); +#endif + /* pDevIns->pHlp->pfnPICSetIrq(pDevIns, 13, 0); */ + break; + + /* Reset. */ + case 0xf1: + Log2(("PCARCH: FPU Reset cb=%d u32=%#x\n", cb, u32)); + /** @todo figure out what the difference between FPU ports 0xf0 and 0xf1 are... */ + /* pDevIns->pHlp->pfnPICSetIrq(pDevIns, 13, 0); */ + break; + + /* opcode transfers */ + case 0xf8: + case 0xfa: + case 0xfc: + default: + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d u32=%#x\n", offPort, cb, u32); + break; + } + /* this works better, but probably not entirely correct. */ + PDMDevHlpISASetIrq(pDevIns, 13, 0); + } + else + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d u32=%#x\n", offPort, cb, u32); + return rc; +} + + +/** + * @callback_method_impl{FNIOMIOPORTIN, PS/2 system control port A.} + * + * @todo Check if the A20 enable/disable method implemented here in any way + * should cooperate with the one implemented in the PS/2 keyboard device. + * This probably belongs together in the PS/2 keyboard device (since that + * is where the "port B" mentioned by Ralph Brown is implemented). + * + * @remark Ralph Brown and friends have this to say about this port: + * + * @verbatim +0092 RW PS/2 system control port A (port B is at PORT 0061h) (see #P0415) + +Bitfields for PS/2 system control port A: +Bit(s) Description (Table P0415) + 7-6 any bit set to 1 turns activity light on + 5 unused + 4 watchdog timout occurred + 3 =0 RTC/CMOS security lock (on password area) unlocked + =1 CMOS locked (done by POST) + 2 unused + 1 A20 is active + 0 =0 system reset or write + =1 pulse alternate reset pin (high-speed alternate CPU reset) +Notes: once set, bit 3 may only be cleared by a power-on reset + on at least the C&T 82C235, bit 0 remains set through a CPU reset to + allow the BIOS to determine the reset method + on the PS/2 30-286 & "Tortuga" the INT 15h/87h memory copy does + not use this port for A20 control, but instead uses the keyboard + controller (8042). Reportedly this may cause the system to crash + when access to the 8042 is disabled in password server mode + (see #P0398). +SeeAlso: #P0416,#P0417,MSR 00001000h + * @endverbatim + * @note offPort is absolute + */ +static DECLCALLBACK(VBOXSTRICTRC) +pcarchIOPortPS2SysControlPortARead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb) +{ + RT_NOREF1(pvUser); + if (cb == 1) + { + *pu32 = PDMDevHlpA20IsEnabled(pDevIns) << 1; + return VINF_SUCCESS; + } + return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d\n", offPort, cb); +} + + +/** + * @callback_method_impl{FNIOMIOPORTOUT, PS/2 system control port A.} + * @see Remark and todo of pcarchIOPortPS2SysControlPortARead(). + * @note offPort is absolute + */ +static DECLCALLBACK(VBOXSTRICTRC) +pcarchIOPortPS2SysControlPortAWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb) +{ + NOREF(pvUser); + if (cb == 1) + { + /* + * Fast reset? + */ + if (u32 & 1) + { + LogRel(("Reset initiated by system port A\n")); + return PDMDevHlpVMReset(pDevIns, PDMVMRESET_F_PORT_A); + } + + /* + * A20 is the only thing we care about of the other stuff. + */ + PDMDevHlpA20Set(pDevIns, !!(u32 & 2)); + return VINF_SUCCESS; + } + return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d u32=%#x\n", offPort, cb, u32); +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnConstruct} + */ +static DECLCALLBACK(int) pcarchConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg) +{ + PDMDEV_CHECK_VERSIONS_RETURN(pDevIns); + PDEVPCARCH pThis = PDMDEVINS_2_DATA(pDevIns, PDEVPCARCH); + int rc; + RT_NOREF(iInstance, pCfg); + Assert(iInstance == 0); + + /* + * Validate configuration. + */ + PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns, "", ""); + + /* + * Init the data. + */ + pThis->pDevIns = pDevIns; + + /* + * Register I/O Ports + */ + IOMIOPORTHANDLE hIoPorts; + rc = PDMDevHlpIoPortCreateFlagsAndMap(pDevIns, 0xf0 /*uPort*/, 0x10 /*cPorts*/, IOM_IOPORT_F_ABS, + pcarchIOPortFPUWrite, pcarchIOPortFPURead, + "Math Co-Processor (DOS/OS2 mode)", NULL /*paExtDescs*/, &hIoPorts); + AssertRCReturn(rc, rc); + rc = PDMDevHlpIoPortCreateFlagsAndMap(pDevIns, 0x92 /*uPort*/, 1 /*cPorts*/, IOM_IOPORT_F_ABS, + pcarchIOPortPS2SysControlPortAWrite, pcarchIOPortPS2SysControlPortARead, + "PS/2 system control port A (A20 and more)", NULL /*paExtDescs*/, &hIoPorts); + AssertRCReturn(rc, rc); + + return VINF_SUCCESS; +} + + +/** + * The device registration structure. + */ +const PDMDEVREG g_DevicePcArch = +{ + /* .u32Version = */ PDM_DEVREG_VERSION, + /* .uReserved0 = */ 0, + /* .szName = */ "pcarch", + /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_NEW_STYLE, + /* .fClass = */ PDM_DEVREG_CLASS_ARCH, + /* .cMaxInstances = */ 1, + /* .uSharedVersion = */ 42, + /* .cbInstanceShared = */ sizeof(DEVPCARCH), + /* .cbInstanceCC = */ 0, + /* .cbInstanceRC = */ 0, + /* .cMaxPciDevices = */ 0, + /* .cMaxMsixVectors = */ 0, + /* .pszDescription = */ "PC Architecture Device", +#if defined(IN_RING3) + /* .pszRCMod = */ "", + /* .pszR0Mod = */ "", + /* .pfnConstruct = */ pcarchConstruct, + /* .pfnDestruct = */ NULL, + /* .pfnRelocate = */ NULL, + /* .pfnMemSetup = */ NULL, + /* .pfnPowerOn = */ NULL, + /* .pfnReset = */ NULL, + /* .pfnSuspend = */ NULL, + /* .pfnResume = */ NULL, + /* .pfnAttach = */ NULL, + /* .pfnDetach = */ NULL, + /* .pfnQueryInterface = */ NULL, + /* .pfnInitComplete = */ NULL, + /* .pfnPowerOff = */ NULL, + /* .pfnSoftReset = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#elif defined(IN_RING0) + /* .pfnEarlyConstruct = */ NULL, + /* .pfnConstruct = */ NULL, + /* .pfnDestruct = */ NULL, + /* .pfnFinalDestruct = */ NULL, + /* .pfnRequest = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#elif defined(IN_RC) + /* .pfnConstruct = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#else +# error "Not in IN_RING3, IN_RING0 or IN_RC!" +#endif + /* .u32VersionEnd = */ PDM_DEVREG_VERSION +}; + |