/* $Id: DevPcArch.cpp $ */ /** @file * DevPcArch - PC Architecture Device. */ /* * Copyright (C) 2006-2019 Oracle Corporation * * 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. */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_DEV_PC_ARCH #include #include #include #include #include #include #include #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{FNIOMIOPORTIN, Math coprocessor.} */ static DECLCALLBACK(int) pcarchIOPortFPURead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb) { int rc; NOREF(pvUser); NOREF(pDevIns); NOREF(pu32); rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Port=%#x cb=%d\n", Port, cb); if (rc == VINF_SUCCESS) rc = VERR_IOM_IOPORT_UNUSED; return rc; } /** * @callback_method_impl{FNIOMIOPORTOUT, Math coprocessor.} * @todo Add IGNNE support. */ static DECLCALLBACK(int) pcarchIOPortFPUWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb) { int rc = VINF_SUCCESS; NOREF(pvUser); if (cb == 1) { switch (Port) { /* * 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", Port, 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", Port, 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", Port, 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 */ static DECLCALLBACK(int) pcarchIOPortPS2SysControlPortARead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, 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", Port, cb); } /** * @callback_method_impl{FNIOMIOPORTOUT, PS/2 system control port A.} * @see Remark and todo of pcarchIOPortPS2SysControlPortARead(). */ static DECLCALLBACK(int) pcarchIOPortPS2SysControlPortAWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, 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", Port, cb, u32); } /** * @callback_method_impl{FNIOMMMIOWRITE, Ignores writes to the reserved memory.} */ static DECLCALLBACK(int) pcarchReservedMemoryWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void const *pv, unsigned cb) { Log2(("pcarchReservedMemoryRead: %#RGp LB %#x %.*Rhxs\n", GCPhysAddr, cb, RT_MIN(cb, 16), pv)); NOREF(pDevIns); NOREF(pvUser); NOREF(GCPhysAddr); NOREF(pv); NOREF(cb); return VINF_SUCCESS; } /** * @callback_method_impl{FNIOMMMIOREAD, The reserved memory reads as 0xff.} */ static DECLCALLBACK(int) pcarchReservedMemoryRead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb) { Log2(("pcarchReservedMemoryRead: %#RGp LB %#x\n", GCPhysAddr, cb)); NOREF(pDevIns); NOREF(pvUser); NOREF(GCPhysAddr); memset(pv, 0xff, cb); return VINF_SUCCESS; } /** * @interface_method_impl{PDMDEVREG,pfnInitComplete, * Turn RAM pages between 0xa0000 and 0xfffff into reserved memory.} */ static DECLCALLBACK(int) pcarchInitComplete(PPDMDEVINS pDevIns) { PVM pVM = PDMDevHlpGetVM(pDevIns); int iRegion = 0; RTGCPHYS const GCPhysEnd = 0x100000; RTGCPHYS GCPhysCur = 0x0a0000; do { if (!PGMPhysIsGCPhysNormal(pVM, GCPhysCur)) GCPhysCur += X86_PAGE_SIZE; else { RTGCPHYS const GCPhysStart = GCPhysCur; do GCPhysCur += X86_PAGE_SIZE; while (GCPhysCur < GCPhysEnd && PGMPhysIsGCPhysNormal(pVM, GCPhysCur)); int rc = PDMDevHlpMMIORegister(pDevIns, GCPhysStart, GCPhysCur - GCPhysStart, NULL /*pvUser*/, IOMMMIO_FLAGS_READ_PASSTHRU | IOMMMIO_FLAGS_WRITE_PASSTHRU, pcarchReservedMemoryWrite, pcarchReservedMemoryRead, MMR3HeapAPrintf(pVM, MM_TAG_PGM_PHYS /* bad bird*/, "PC Arch Reserved #%u", iRegion)); AssertLogRelRCReturn(rc, rc); iRegion++; } } while (GCPhysCur < GCPhysEnd); return VINF_SUCCESS; } /** * @interface_method_impl{PDMDEVREG,pfnConstruct} */ static DECLCALLBACK(int) pcarchConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg) { PDMDEV_CHECK_VERSIONS_RETURN(pDevIns); PDEVPCARCH pThis = PDMINS_2_DATA(pDevIns, PDEVPCARCH); int rc; Assert(iInstance == 0); RT_NOREF(iInstance); /* * Validate configuration. */ if (!CFGMR3AreValuesValid(pCfg, "\0")) return VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES; /* * Init the data. */ pThis->pDevIns = pDevIns; /* * Register I/O Ports */ rc = PDMDevHlpIOPortRegister(pDevIns, 0xF0, 0x10, NULL, pcarchIOPortFPUWrite, pcarchIOPortFPURead, NULL, NULL, "Math Co-Processor (DOS/OS2 mode)"); if (RT_FAILURE(rc)) return rc; rc = PDMDevHlpIOPortRegister(pDevIns, 0x92, 1, NULL, pcarchIOPortPS2SysControlPortAWrite, pcarchIOPortPS2SysControlPortARead, NULL, NULL, "PS/2 system control port A (A20 and more)"); if (RT_FAILURE(rc)) return rc; return VINF_SUCCESS; } /** * The device registration structure. */ const PDMDEVREG g_DevicePcArch = { /* u32Version */ PDM_DEVREG_VERSION, /* szName */ "pcarch", /* szRCMod */ "", /* szR0Mod */ "", /* pszDescription */ "PC Architecture Device", /* fFlags */ PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT, /* fClass */ PDM_DEVREG_CLASS_ARCH, /* cMaxInstances */ 1, /* cbInstance */ sizeof(DEVPCARCH), /* pfnConstruct */ pcarchConstruct, /* pfnDestruct */ NULL, /* pfnRelocate */ NULL, /* pfnMemSetup */ NULL, /* pfnPowerOn */ NULL, /* pfnReset */ NULL, /* pfnSuspend */ NULL, /* pfnResume */ NULL, /* pfnAttach */ NULL, /* pfnDetach */ NULL, /* pfnQueryInterface. */ NULL, /* pfnInitComplete. */ pcarchInitComplete, /* pfnPowerOff */ NULL, /* pfnSoftReset */ NULL, /* u32VersionEnd */ PDM_DEVREG_VERSION };