Bochs/bochs/memory/memory-bochs.h
Justin Moore aac6dd5d13
Bump BIOSROMSZ to 4M to accommodate OVMF (#284)
OVMF is 4 megabytes by default, so in order to avoid customizing their
build process we'll want to accommodate a 4M BIOS in Bochs.
2024-03-15 19:26:56 +01:00

216 lines
7.7 KiB
C++

/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2001-2023 The Bochs Project
//
// I/O memory handlers API Copyright (C) 2003 by Frank Cornelis
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later version.
//
// This library 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
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
//
/////////////////////////////////////////////////////////////////////////
#ifndef BX_MEM_H
# define BX_MEM_H 1
#if BX_USE_MEM_SMF
// if static member functions on, then there is only one memory
# define BX_MEM_SMF static
# define BX_MEM_THIS BX_MEM(0)->
#else
# define BX_MEM_SMF
# define BX_MEM_THIS this->
#endif
class BX_CPU_C;
// 512K BIOS ROM @0xfff80000
const Bit32u BIOSROMSZ = (1 << 22); // 4M BIOS ROM @0xffc00000, must be a power of 2
const Bit32u EXROMSIZE = (0x20000); // ROMs 0xc0000-0xdffff (area 0xe0000-0xfffff=bios mapped)
const Bit32u BIOS_MASK = BIOSROMSZ-1;
const Bit32u EXROM_MASK = EXROMSIZE-1;
class BOCHSAPI BX_MEMORY_STUB_C : public logfunctions {
protected:
Bit64u len, allocated; // could be > 4G
Bit32u block_size; // individual block size, must be power of 2
Bit8u *actual_vector;
Bit8u *vector; // aligned correctly
Bit8u **blocks;
Bit8u *rom; // 512k BIOS rom space + 128k expansion rom space
Bit8u *bogus; // 4k for unexisting memory
Bit32u used_blocks;
#if BX_LARGE_RAMFILE
static Bit8u * const swapped_out; // NULL; // (NULL - sizeof(Bit8u));
Bit32u next_swapout_idx;
FILE *overflow_file;
BX_MEM_SMF void read_block(Bit32u block);
#endif
public:
BX_MEMORY_STUB_C();
virtual ~BX_MEMORY_STUB_C();
BX_MEM_SMF void init_memory(Bit64u guest, Bit64u host, Bit32u block_size);
BX_MEM_SMF void cleanup_memory(void);
BX_MEM_SMF Bit8u* get_vector(bx_phy_address addr);
BX_MEM_SMF Bit8u* getHostMemAddr(BX_CPU_C *cpu, bx_phy_address addr, unsigned rw);
// Note: accesses should always be contained within a single page
BX_MEM_SMF void readPhysicalPage(BX_CPU_C *cpu, bx_phy_address addr,
unsigned len, void *data);
BX_MEM_SMF void writePhysicalPage(BX_CPU_C *cpu, bx_phy_address addr,
unsigned len, void *data);
BX_MEM_SMF bool dbg_fetch_mem(BX_CPU_C *cpu, bx_phy_address addr, unsigned len, Bit8u *buf);
#if (BX_DEBUGGER || BX_GDBSTUB)
BX_MEM_SMF bool dbg_set_mem(BX_CPU_C *cpu, bx_phy_address addr, unsigned len, Bit8u *buf);
BX_MEM_SMF bool dbg_crc32(bx_phy_address addr1, bx_phy_address addr2, Bit32u *crc);
#endif
BX_MEM_SMF Bit64u get_memory_len(void);
BX_MEM_SMF void allocate_block(Bit32u index);
BX_MEM_SMF Bit8u* alloc_vector_aligned(Bit64u bytes, Bit64u alignment);
#if BX_SUPPORT_MONITOR_MWAIT
BX_MEM_SMF bool is_monitor(bx_phy_address begin_addr, unsigned len);
BX_MEM_SMF void check_monitor(bx_phy_address addr, unsigned len);
#endif
};
typedef bool (*memory_handler_t)(bx_phy_address addr, unsigned len, void *data, void *param);
// return a pointer to 4K region containing <addr> or NULL if direct access is not allowed
// same format as getHostMemAddr method
typedef Bit8u* (*memory_direct_access_handler_t)(bx_phy_address addr, unsigned rw, void *param);
struct memory_handler_struct {
struct memory_handler_struct *next;
void *param;
bx_phy_address begin;
bx_phy_address end;
Bit16u bitmap;
memory_handler_t read_handler;
memory_handler_t write_handler;
memory_direct_access_handler_t da_handler;
};
#define BIOS_MAP_LAST128K(addr) (((addr) | 0xfff00000) & BIOS_MASK)
#define BIOS_ROM_LOWER 0x01
#define BIOS_ROM_EXTENDED 0x02
#define BIOS_ROM_1MEG 0x04
enum memory_area_t {
BX_MEM_AREA_C0000 = 0,
BX_MEM_AREA_C4000,
BX_MEM_AREA_C8000,
BX_MEM_AREA_CC000,
BX_MEM_AREA_D0000,
BX_MEM_AREA_D4000,
BX_MEM_AREA_D8000,
BX_MEM_AREA_DC000,
BX_MEM_AREA_E0000,
BX_MEM_AREA_E4000,
BX_MEM_AREA_E8000,
BX_MEM_AREA_EC000,
BX_MEM_AREA_F0000
};
class BOCHSAPI BX_MEM_C : public BX_MEMORY_STUB_C {
private:
struct memory_handler_struct **memory_handlers;
bool pci_enabled;
bool bios_write_enabled;
bool smram_available;
bool smram_enable;
bool smram_restricted;
bool rom_present[65];
bool memory_type[13][2];
Bit32u bios_rom_addr;
Bit8u bios_rom_access;
Bit8u flash_type;
Bit8u flash_status;
Bit8u flash_wsm_state;
bool flash_modified;
BX_MEM_SMF Bit8u flash_read(Bit32u addr);
BX_MEM_SMF void flash_write(Bit32u addr, Bit8u data);
public:
BX_MEM_C();
virtual ~BX_MEM_C();
BX_MEM_SMF void init_memory(Bit64u guest, Bit64u host, Bit32u block_size);
BX_MEM_SMF void cleanup_memory(void);
BX_MEM_SMF void enable_smram(bool enable, bool restricted);
BX_MEM_SMF void disable_smram(void);
BX_MEM_SMF bool is_smram_accessible(void);
BX_MEM_SMF void set_bios_write(bool enabled);
BX_MEM_SMF void set_bios_rom_access(Bit8u region, bool enabled);
BX_MEM_SMF void set_memory_type(memory_area_t area, bool rw, bool dram);
BX_MEM_SMF Bit8u* getHostMemAddr(BX_CPU_C *cpu, bx_phy_address addr, unsigned rw);
// Note: accesses should always be contained within a single page
BX_MEM_SMF void readPhysicalPage(BX_CPU_C *cpu, bx_phy_address addr,
unsigned len, void *data);
BX_MEM_SMF void writePhysicalPage(BX_CPU_C *cpu, bx_phy_address addr,
unsigned len, void *data);
BX_MEM_SMF void dmaReadPhysicalPage(bx_phy_address addr, unsigned len, Bit8u *data);
BX_MEM_SMF void dmaWritePhysicalPage(bx_phy_address addr, unsigned len, Bit8u *data);
BX_MEM_SMF bool load_flash_data(const char *path);
BX_MEM_SMF bool save_flash_data(const char *path);
BX_MEM_SMF void load_ROM(const char *path, bx_phy_address romaddress, Bit8u type);
BX_MEM_SMF void load_RAM(const char *path, bx_phy_address romaddress);
BX_MEM_SMF bool dbg_fetch_mem(BX_CPU_C *cpu, bx_phy_address addr, unsigned len, Bit8u *buf);
#if (BX_DEBUGGER || BX_GDBSTUB)
BX_MEM_SMF bool dbg_set_mem(BX_CPU_C *cpu, bx_phy_address addr, unsigned len, Bit8u *buf);
#endif
BX_MEM_SMF bool registerMemoryHandlers(void *param, memory_handler_t read_handler,
memory_handler_t write_handler, memory_direct_access_handler_t da_handler,
bx_phy_address begin_addr, bx_phy_address end_addr);
BX_MEM_SMF BX_CPP_INLINE bool registerMemoryHandlers(void *param, memory_handler_t read_handler,
memory_handler_t write_handler,
bx_phy_address begin_addr, bx_phy_address end_addr)
{
return registerMemoryHandlers(param, read_handler, write_handler, NULL, begin_addr, end_addr);
}
BX_MEM_SMF bool unregisterMemoryHandlers(void *param, bx_phy_address begin_addr, bx_phy_address end_addr);
void register_state(void);
#if BX_LARGE_RAMFILE
friend void ramfile_save_handler(void *devptr, FILE *fp);
#endif
friend Bit64s memory_param_save_handler(void *devptr, bx_param_c *param);
friend void memory_param_restore_handler(void *devptr, bx_param_c *param, Bit64s val);
};
BOCHSAPI extern BX_MEM_C bx_mem;
#endif