qemu-sparc update
-----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.12 (GNU/Linux) iQEcBAABAgAGBQJU/vQnAAoJEFvCxW+uDzIfSHQIAKLVePJyOEtVA5OOUSyuA5TQ 6MNXg1X+PxD0GT8iyKhcyHqaRSibPjY/D2Hb8MyNt/lLIQj13hfL1fXXAMvnjY1+ WcCpGfbZmmcIpgjY+OWkkSVU3nSxNE6a0ZgdOvK9Cvc+dJHMpO02E5kRqig6nYX6 UuA7aIgjE2T63PPucuEjFVA4a8v7NsqO4OBmYPO/L7KEpAQk3Y/qDDGahQRXuBXl 0poyJrj99BoKHJFemf7L79SbpZHrnaQ54dndqerq5tzmBXRVjhYJa/iLm0LSjPnN kWdlMKXSGgI/rFRWZqfOk+jDIZb21xliM6y/lDpDJGdoC1aPsIWwQ0mFK6ZPVDo= =Ee+Y -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/mcayland/tags/qemu-sparc-signed' into staging qemu-sparc update # gpg: Signature made Tue Mar 10 13:39:51 2015 GMT using RSA key ID AE0F321F # gpg: Good signature from "Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>" * remotes/mcayland/tags/qemu-sparc-signed: sun4u: switch m48t59 NVRAM to MMIO access MAINTAINERS: add myself as SPARC maintainer doc: minor updates to SPARC32 and SPARC64 documentation m48t59: add m48t59 sysbus device m48t59: introduce new base-year qdev property m48t59: let init functions return a Nvram object m48t59: add a Nvram interface m48t59: register a QOM type for each nvram type we support m48t59: move ISA ports/memory regions registration to QOM constructor Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
4ba4df405c
@ -159,6 +159,7 @@ F: hw/sh4/
|
|||||||
|
|
||||||
SPARC
|
SPARC
|
||||||
M: Blue Swirl <blauwirbel@gmail.com>
|
M: Blue Swirl <blauwirbel@gmail.com>
|
||||||
|
M: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>
|
||||||
S: Maintained
|
S: Maintained
|
||||||
F: target-sparc/
|
F: target-sparc/
|
||||||
F: hw/sparc/
|
F: hw/sparc/
|
||||||
@ -518,11 +519,13 @@ SPARC Machines
|
|||||||
--------------
|
--------------
|
||||||
Sun4m
|
Sun4m
|
||||||
M: Blue Swirl <blauwirbel@gmail.com>
|
M: Blue Swirl <blauwirbel@gmail.com>
|
||||||
|
M: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>
|
||||||
S: Maintained
|
S: Maintained
|
||||||
F: hw/sparc/sun4m.c
|
F: hw/sparc/sun4m.c
|
||||||
|
|
||||||
Sun4u
|
Sun4u
|
||||||
M: Blue Swirl <blauwirbel@gmail.com>
|
M: Blue Swirl <blauwirbel@gmail.com>
|
||||||
|
M: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>
|
||||||
S: Maintained
|
S: Maintained
|
||||||
F: hw/sparc64/sun4u.c
|
F: hw/sparc64/sun4u.c
|
||||||
|
|
||||||
|
161
hw/ppc/ppc.c
161
hw/ppc/ppc.c
@ -1318,167 +1318,6 @@ void PPC_debug_write (void *opaque, uint32_t addr, uint32_t val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*****************************************************************************/
|
|
||||||
/* NVRAM helpers */
|
|
||||||
static inline uint32_t nvram_read (nvram_t *nvram, uint32_t addr)
|
|
||||||
{
|
|
||||||
return (*nvram->read_fn)(nvram->opaque, addr);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void nvram_write (nvram_t *nvram, uint32_t addr, uint32_t val)
|
|
||||||
{
|
|
||||||
(*nvram->write_fn)(nvram->opaque, addr, val);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void NVRAM_set_byte(nvram_t *nvram, uint32_t addr, uint8_t value)
|
|
||||||
{
|
|
||||||
nvram_write(nvram, addr, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
static uint8_t NVRAM_get_byte(nvram_t *nvram, uint32_t addr)
|
|
||||||
{
|
|
||||||
return nvram_read(nvram, addr);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void NVRAM_set_word(nvram_t *nvram, uint32_t addr, uint16_t value)
|
|
||||||
{
|
|
||||||
nvram_write(nvram, addr, value >> 8);
|
|
||||||
nvram_write(nvram, addr + 1, value & 0xFF);
|
|
||||||
}
|
|
||||||
|
|
||||||
static uint16_t NVRAM_get_word(nvram_t *nvram, uint32_t addr)
|
|
||||||
{
|
|
||||||
uint16_t tmp;
|
|
||||||
|
|
||||||
tmp = nvram_read(nvram, addr) << 8;
|
|
||||||
tmp |= nvram_read(nvram, addr + 1);
|
|
||||||
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void NVRAM_set_lword(nvram_t *nvram, uint32_t addr, uint32_t value)
|
|
||||||
{
|
|
||||||
nvram_write(nvram, addr, value >> 24);
|
|
||||||
nvram_write(nvram, addr + 1, (value >> 16) & 0xFF);
|
|
||||||
nvram_write(nvram, addr + 2, (value >> 8) & 0xFF);
|
|
||||||
nvram_write(nvram, addr + 3, value & 0xFF);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t NVRAM_get_lword (nvram_t *nvram, uint32_t addr)
|
|
||||||
{
|
|
||||||
uint32_t tmp;
|
|
||||||
|
|
||||||
tmp = nvram_read(nvram, addr) << 24;
|
|
||||||
tmp |= nvram_read(nvram, addr + 1) << 16;
|
|
||||||
tmp |= nvram_read(nvram, addr + 2) << 8;
|
|
||||||
tmp |= nvram_read(nvram, addr + 3);
|
|
||||||
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void NVRAM_set_string(nvram_t *nvram, uint32_t addr, const char *str,
|
|
||||||
uint32_t max)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < max && str[i] != '\0'; i++) {
|
|
||||||
nvram_write(nvram, addr + i, str[i]);
|
|
||||||
}
|
|
||||||
nvram_write(nvram, addr + i, str[i]);
|
|
||||||
nvram_write(nvram, addr + max - 1, '\0');
|
|
||||||
}
|
|
||||||
|
|
||||||
int NVRAM_get_string (nvram_t *nvram, uint8_t *dst, uint16_t addr, int max)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
memset(dst, 0, max);
|
|
||||||
for (i = 0; i < max; i++) {
|
|
||||||
dst[i] = NVRAM_get_byte(nvram, addr + i);
|
|
||||||
if (dst[i] == '\0')
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
static uint16_t NVRAM_crc_update (uint16_t prev, uint16_t value)
|
|
||||||
{
|
|
||||||
uint16_t tmp;
|
|
||||||
uint16_t pd, pd1, pd2;
|
|
||||||
|
|
||||||
tmp = prev >> 8;
|
|
||||||
pd = prev ^ value;
|
|
||||||
pd1 = pd & 0x000F;
|
|
||||||
pd2 = ((pd >> 4) & 0x000F) ^ pd1;
|
|
||||||
tmp ^= (pd1 << 3) | (pd1 << 8);
|
|
||||||
tmp ^= pd2 | (pd2 << 7) | (pd2 << 12);
|
|
||||||
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
static uint16_t NVRAM_compute_crc (nvram_t *nvram, uint32_t start, uint32_t count)
|
|
||||||
{
|
|
||||||
uint32_t i;
|
|
||||||
uint16_t crc = 0xFFFF;
|
|
||||||
int odd;
|
|
||||||
|
|
||||||
odd = count & 1;
|
|
||||||
count &= ~1;
|
|
||||||
for (i = 0; i != count; i++) {
|
|
||||||
crc = NVRAM_crc_update(crc, NVRAM_get_word(nvram, start + i));
|
|
||||||
}
|
|
||||||
if (odd) {
|
|
||||||
crc = NVRAM_crc_update(crc, NVRAM_get_byte(nvram, start + i) << 8);
|
|
||||||
}
|
|
||||||
|
|
||||||
return crc;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define CMDLINE_ADDR 0x017ff000
|
|
||||||
|
|
||||||
int PPC_NVRAM_set_params (nvram_t *nvram, uint16_t NVRAM_size,
|
|
||||||
const char *arch,
|
|
||||||
uint32_t RAM_size, int boot_device,
|
|
||||||
uint32_t kernel_image, uint32_t kernel_size,
|
|
||||||
const char *cmdline,
|
|
||||||
uint32_t initrd_image, uint32_t initrd_size,
|
|
||||||
uint32_t NVRAM_image,
|
|
||||||
int width, int height, int depth)
|
|
||||||
{
|
|
||||||
uint16_t crc;
|
|
||||||
|
|
||||||
/* Set parameters for Open Hack'Ware BIOS */
|
|
||||||
NVRAM_set_string(nvram, 0x00, "QEMU_BIOS", 16);
|
|
||||||
NVRAM_set_lword(nvram, 0x10, 0x00000002); /* structure v2 */
|
|
||||||
NVRAM_set_word(nvram, 0x14, NVRAM_size);
|
|
||||||
NVRAM_set_string(nvram, 0x20, arch, 16);
|
|
||||||
NVRAM_set_lword(nvram, 0x30, RAM_size);
|
|
||||||
NVRAM_set_byte(nvram, 0x34, boot_device);
|
|
||||||
NVRAM_set_lword(nvram, 0x38, kernel_image);
|
|
||||||
NVRAM_set_lword(nvram, 0x3C, kernel_size);
|
|
||||||
if (cmdline) {
|
|
||||||
/* XXX: put the cmdline in NVRAM too ? */
|
|
||||||
pstrcpy_targphys("cmdline", CMDLINE_ADDR, RAM_size - CMDLINE_ADDR, cmdline);
|
|
||||||
NVRAM_set_lword(nvram, 0x40, CMDLINE_ADDR);
|
|
||||||
NVRAM_set_lword(nvram, 0x44, strlen(cmdline));
|
|
||||||
} else {
|
|
||||||
NVRAM_set_lword(nvram, 0x40, 0);
|
|
||||||
NVRAM_set_lword(nvram, 0x44, 0);
|
|
||||||
}
|
|
||||||
NVRAM_set_lword(nvram, 0x48, initrd_image);
|
|
||||||
NVRAM_set_lword(nvram, 0x4C, initrd_size);
|
|
||||||
NVRAM_set_lword(nvram, 0x50, NVRAM_image);
|
|
||||||
|
|
||||||
NVRAM_set_word(nvram, 0x54, width);
|
|
||||||
NVRAM_set_word(nvram, 0x56, height);
|
|
||||||
NVRAM_set_word(nvram, 0x58, depth);
|
|
||||||
crc = NVRAM_compute_crc(nvram, 0x00, 0xF8);
|
|
||||||
NVRAM_set_word(nvram, 0xFC, crc);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* CPU device-tree ID helpers */
|
/* CPU device-tree ID helpers */
|
||||||
int ppc_get_vcpu_dt_id(PowerPCCPU *cpu)
|
int ppc_get_vcpu_dt_id(PowerPCCPU *cpu)
|
||||||
{
|
{
|
||||||
|
@ -283,7 +283,7 @@ static void ref405ep_init(MachineState *machine)
|
|||||||
#ifdef DEBUG_BOARD_INIT
|
#ifdef DEBUG_BOARD_INIT
|
||||||
printf("%s: register NVRAM\n", __func__);
|
printf("%s: register NVRAM\n", __func__);
|
||||||
#endif
|
#endif
|
||||||
m48t59_init(NULL, 0xF0000000, 0, 8192, 8);
|
m48t59_init(NULL, 0xF0000000, 0, 8192, 1968, 8);
|
||||||
/* Load kernel */
|
/* Load kernel */
|
||||||
linux_boot = (kernel_filename != NULL);
|
linux_boot = (kernel_filename != NULL);
|
||||||
if (linux_boot) {
|
if (linux_boot) {
|
||||||
|
163
hw/ppc/prep.c
163
hw/ppc/prep.c
@ -181,7 +181,7 @@ static const MemoryRegionOps PPC_XCSR_ops = {
|
|||||||
/* Fake super-io ports for PREP platform (Intel 82378ZB) */
|
/* Fake super-io ports for PREP platform (Intel 82378ZB) */
|
||||||
typedef struct sysctrl_t {
|
typedef struct sysctrl_t {
|
||||||
qemu_irq reset_irq;
|
qemu_irq reset_irq;
|
||||||
M48t59State *nvram;
|
Nvram *nvram;
|
||||||
uint8_t state;
|
uint8_t state;
|
||||||
uint8_t syscontrol;
|
uint8_t syscontrol;
|
||||||
int contiguous_map;
|
int contiguous_map;
|
||||||
@ -235,13 +235,17 @@ static void PREP_io_800_writeb (void *opaque, uint32_t addr, uint32_t val)
|
|||||||
break;
|
break;
|
||||||
case 0x0810:
|
case 0x0810:
|
||||||
/* Password protect 1 register */
|
/* Password protect 1 register */
|
||||||
if (sysctrl->nvram != NULL)
|
if (sysctrl->nvram != NULL) {
|
||||||
m48t59_toggle_lock(sysctrl->nvram, 1);
|
NvramClass *k = NVRAM_GET_CLASS(sysctrl->nvram);
|
||||||
|
(k->toggle_lock)(sysctrl->nvram, 1);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0x0812:
|
case 0x0812:
|
||||||
/* Password protect 2 register */
|
/* Password protect 2 register */
|
||||||
if (sysctrl->nvram != NULL)
|
if (sysctrl->nvram != NULL) {
|
||||||
m48t59_toggle_lock(sysctrl->nvram, 2);
|
NvramClass *k = NVRAM_GET_CLASS(sysctrl->nvram);
|
||||||
|
(k->toggle_lock)(sysctrl->nvram, 2);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0x0814:
|
case 0x0814:
|
||||||
/* L2 invalidate register */
|
/* L2 invalidate register */
|
||||||
@ -360,6 +364,144 @@ static const MemoryRegionPortio prep_portio_list[] = {
|
|||||||
|
|
||||||
static PortioList prep_port_list;
|
static PortioList prep_port_list;
|
||||||
|
|
||||||
|
/*****************************************************************************/
|
||||||
|
/* NVRAM helpers */
|
||||||
|
static inline uint32_t nvram_read(Nvram *nvram, uint32_t addr)
|
||||||
|
{
|
||||||
|
NvramClass *k = NVRAM_GET_CLASS(sysctrl->nvram);
|
||||||
|
return (k->read)(nvram, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void nvram_write(Nvram *nvram, uint32_t addr, uint32_t val)
|
||||||
|
{
|
||||||
|
NvramClass *k = NVRAM_GET_CLASS(sysctrl->nvram);
|
||||||
|
(k->write)(nvram, addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void NVRAM_set_byte(Nvram *nvram, uint32_t addr, uint8_t value)
|
||||||
|
{
|
||||||
|
nvram_write(nvram, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint8_t NVRAM_get_byte(Nvram *nvram, uint32_t addr)
|
||||||
|
{
|
||||||
|
return nvram_read(nvram, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void NVRAM_set_word(Nvram *nvram, uint32_t addr, uint16_t value)
|
||||||
|
{
|
||||||
|
nvram_write(nvram, addr, value >> 8);
|
||||||
|
nvram_write(nvram, addr + 1, value & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t NVRAM_get_word(Nvram *nvram, uint32_t addr)
|
||||||
|
{
|
||||||
|
uint16_t tmp;
|
||||||
|
|
||||||
|
tmp = nvram_read(nvram, addr) << 8;
|
||||||
|
tmp |= nvram_read(nvram, addr + 1);
|
||||||
|
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void NVRAM_set_lword(Nvram *nvram, uint32_t addr, uint32_t value)
|
||||||
|
{
|
||||||
|
nvram_write(nvram, addr, value >> 24);
|
||||||
|
nvram_write(nvram, addr + 1, (value >> 16) & 0xFF);
|
||||||
|
nvram_write(nvram, addr + 2, (value >> 8) & 0xFF);
|
||||||
|
nvram_write(nvram, addr + 3, value & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void NVRAM_set_string(Nvram *nvram, uint32_t addr, const char *str,
|
||||||
|
uint32_t max)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < max && str[i] != '\0'; i++) {
|
||||||
|
nvram_write(nvram, addr + i, str[i]);
|
||||||
|
}
|
||||||
|
nvram_write(nvram, addr + i, str[i]);
|
||||||
|
nvram_write(nvram, addr + max - 1, '\0');
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t NVRAM_crc_update (uint16_t prev, uint16_t value)
|
||||||
|
{
|
||||||
|
uint16_t tmp;
|
||||||
|
uint16_t pd, pd1, pd2;
|
||||||
|
|
||||||
|
tmp = prev >> 8;
|
||||||
|
pd = prev ^ value;
|
||||||
|
pd1 = pd & 0x000F;
|
||||||
|
pd2 = ((pd >> 4) & 0x000F) ^ pd1;
|
||||||
|
tmp ^= (pd1 << 3) | (pd1 << 8);
|
||||||
|
tmp ^= pd2 | (pd2 << 7) | (pd2 << 12);
|
||||||
|
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t NVRAM_compute_crc (Nvram *nvram, uint32_t start, uint32_t count)
|
||||||
|
{
|
||||||
|
uint32_t i;
|
||||||
|
uint16_t crc = 0xFFFF;
|
||||||
|
int odd;
|
||||||
|
|
||||||
|
odd = count & 1;
|
||||||
|
count &= ~1;
|
||||||
|
for (i = 0; i != count; i++) {
|
||||||
|
crc = NVRAM_crc_update(crc, NVRAM_get_word(nvram, start + i));
|
||||||
|
}
|
||||||
|
if (odd) {
|
||||||
|
crc = NVRAM_crc_update(crc, NVRAM_get_byte(nvram, start + i) << 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
return crc;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define CMDLINE_ADDR 0x017ff000
|
||||||
|
|
||||||
|
static int PPC_NVRAM_set_params (Nvram *nvram, uint16_t NVRAM_size,
|
||||||
|
const char *arch,
|
||||||
|
uint32_t RAM_size, int boot_device,
|
||||||
|
uint32_t kernel_image, uint32_t kernel_size,
|
||||||
|
const char *cmdline,
|
||||||
|
uint32_t initrd_image, uint32_t initrd_size,
|
||||||
|
uint32_t NVRAM_image,
|
||||||
|
int width, int height, int depth)
|
||||||
|
{
|
||||||
|
uint16_t crc;
|
||||||
|
|
||||||
|
/* Set parameters for Open Hack'Ware BIOS */
|
||||||
|
NVRAM_set_string(nvram, 0x00, "QEMU_BIOS", 16);
|
||||||
|
NVRAM_set_lword(nvram, 0x10, 0x00000002); /* structure v2 */
|
||||||
|
NVRAM_set_word(nvram, 0x14, NVRAM_size);
|
||||||
|
NVRAM_set_string(nvram, 0x20, arch, 16);
|
||||||
|
NVRAM_set_lword(nvram, 0x30, RAM_size);
|
||||||
|
NVRAM_set_byte(nvram, 0x34, boot_device);
|
||||||
|
NVRAM_set_lword(nvram, 0x38, kernel_image);
|
||||||
|
NVRAM_set_lword(nvram, 0x3C, kernel_size);
|
||||||
|
if (cmdline) {
|
||||||
|
/* XXX: put the cmdline in NVRAM too ? */
|
||||||
|
pstrcpy_targphys("cmdline", CMDLINE_ADDR, RAM_size - CMDLINE_ADDR,
|
||||||
|
cmdline);
|
||||||
|
NVRAM_set_lword(nvram, 0x40, CMDLINE_ADDR);
|
||||||
|
NVRAM_set_lword(nvram, 0x44, strlen(cmdline));
|
||||||
|
} else {
|
||||||
|
NVRAM_set_lword(nvram, 0x40, 0);
|
||||||
|
NVRAM_set_lword(nvram, 0x44, 0);
|
||||||
|
}
|
||||||
|
NVRAM_set_lword(nvram, 0x48, initrd_image);
|
||||||
|
NVRAM_set_lword(nvram, 0x4C, initrd_size);
|
||||||
|
NVRAM_set_lword(nvram, 0x50, NVRAM_image);
|
||||||
|
|
||||||
|
NVRAM_set_word(nvram, 0x54, width);
|
||||||
|
NVRAM_set_word(nvram, 0x56, height);
|
||||||
|
NVRAM_set_word(nvram, 0x58, depth);
|
||||||
|
crc = NVRAM_compute_crc(nvram, 0x00, 0xF8);
|
||||||
|
NVRAM_set_word(nvram, 0xFC, crc);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* PowerPC PREP hardware initialisation */
|
/* PowerPC PREP hardware initialisation */
|
||||||
static void ppc_prep_init(MachineState *machine)
|
static void ppc_prep_init(MachineState *machine)
|
||||||
{
|
{
|
||||||
@ -372,8 +514,7 @@ static void ppc_prep_init(MachineState *machine)
|
|||||||
MemoryRegion *sysmem = get_system_memory();
|
MemoryRegion *sysmem = get_system_memory();
|
||||||
PowerPCCPU *cpu = NULL;
|
PowerPCCPU *cpu = NULL;
|
||||||
CPUPPCState *env = NULL;
|
CPUPPCState *env = NULL;
|
||||||
nvram_t nvram;
|
Nvram *m48t59;
|
||||||
M48t59State *m48t59;
|
|
||||||
#if 0
|
#if 0
|
||||||
MemoryRegion *xcsr = g_new(MemoryRegion, 1);
|
MemoryRegion *xcsr = g_new(MemoryRegion, 1);
|
||||||
#endif
|
#endif
|
||||||
@ -543,16 +684,14 @@ static void ppc_prep_init(MachineState *machine)
|
|||||||
pci_create_simple(pci_bus, -1, "pci-ohci");
|
pci_create_simple(pci_bus, -1, "pci-ohci");
|
||||||
}
|
}
|
||||||
|
|
||||||
m48t59 = m48t59_init_isa(isa_bus, 0x0074, NVRAM_SIZE, 59);
|
m48t59 = m48t59_init_isa(isa_bus, 0x0074, NVRAM_SIZE, 2000, 59);
|
||||||
if (m48t59 == NULL)
|
if (m48t59 == NULL)
|
||||||
return;
|
return;
|
||||||
sysctrl->nvram = m48t59;
|
sysctrl->nvram = m48t59;
|
||||||
|
|
||||||
/* Initialise NVRAM */
|
/* Initialise NVRAM */
|
||||||
nvram.opaque = m48t59;
|
PPC_NVRAM_set_params(m48t59, NVRAM_SIZE, "PREP", ram_size,
|
||||||
nvram.read_fn = &m48t59_read;
|
ppc_boot_device,
|
||||||
nvram.write_fn = &m48t59_write;
|
|
||||||
PPC_NVRAM_set_params(&nvram, NVRAM_SIZE, "PREP", ram_size, ppc_boot_device,
|
|
||||||
kernel_base, kernel_size,
|
kernel_base, kernel_size,
|
||||||
kernel_cmdline,
|
kernel_cmdline,
|
||||||
initrd_base, initrd_size,
|
initrd_base, initrd_size,
|
||||||
|
@ -127,7 +127,7 @@ static void fw_cfg_boot_set(void *opaque, const char *boot_device,
|
|||||||
fw_cfg_add_i16(opaque, FW_CFG_BOOT_DEVICE, boot_device[0]);
|
fw_cfg_add_i16(opaque, FW_CFG_BOOT_DEVICE, boot_device[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void nvram_init(M48t59State *nvram, uint8_t *macaddr,
|
static void nvram_init(Nvram *nvram, uint8_t *macaddr,
|
||||||
const char *cmdline, const char *boot_devices,
|
const char *cmdline, const char *boot_devices,
|
||||||
ram_addr_t RAM_size, uint32_t kernel_size,
|
ram_addr_t RAM_size, uint32_t kernel_size,
|
||||||
int width, int height, int depth,
|
int width, int height, int depth,
|
||||||
@ -137,6 +137,7 @@ static void nvram_init(M48t59State *nvram, uint8_t *macaddr,
|
|||||||
uint32_t start, end;
|
uint32_t start, end;
|
||||||
uint8_t image[0x1ff0];
|
uint8_t image[0x1ff0];
|
||||||
struct OpenBIOS_nvpart_v1 *part_header;
|
struct OpenBIOS_nvpart_v1 *part_header;
|
||||||
|
NvramClass *k = NVRAM_GET_CLASS(nvram);
|
||||||
|
|
||||||
memset(image, '\0', sizeof(image));
|
memset(image, '\0', sizeof(image));
|
||||||
|
|
||||||
@ -170,8 +171,9 @@ static void nvram_init(M48t59State *nvram, uint8_t *macaddr,
|
|||||||
Sun_init_header((struct Sun_nvram *)&image[0x1fd8], macaddr,
|
Sun_init_header((struct Sun_nvram *)&image[0x1fd8], macaddr,
|
||||||
nvram_machine_id);
|
nvram_machine_id);
|
||||||
|
|
||||||
for (i = 0; i < sizeof(image); i++)
|
for (i = 0; i < sizeof(image); i++) {
|
||||||
m48t59_write(nvram, i, image[i]);
|
(k->write)(nvram, i, image[i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static DeviceState *slavio_intctl;
|
static DeviceState *slavio_intctl;
|
||||||
@ -1012,7 +1014,7 @@ static void sun4m_hw_init(const struct sun4m_hwdef *hwdef,
|
|||||||
|
|
||||||
lance_init(&nd_table[0], hwdef->le_base, ledma, ledma_irq);
|
lance_init(&nd_table[0], hwdef->le_base, ledma, ledma_irq);
|
||||||
|
|
||||||
nvram = m48t59_init(slavio_irq[0], hwdef->nvram_base, 0, 0x2000, 8);
|
nvram = m48t59_init(slavio_irq[0], hwdef->nvram_base, 0, 0x2000, 1968, 8);
|
||||||
|
|
||||||
slavio_timer_init_all(hwdef->counter_base, slavio_irq[19], slavio_cpu_irq, smp_cpus);
|
slavio_timer_init_all(hwdef->counter_base, slavio_irq[19], slavio_cpu_irq, smp_cpus);
|
||||||
|
|
||||||
|
@ -130,7 +130,7 @@ static void fw_cfg_boot_set(void *opaque, const char *boot_device,
|
|||||||
fw_cfg_add_i16(opaque, FW_CFG_BOOT_DEVICE, boot_device[0]);
|
fw_cfg_add_i16(opaque, FW_CFG_BOOT_DEVICE, boot_device[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int sun4u_NVRAM_set_params(M48t59State *nvram, uint16_t NVRAM_size,
|
static int sun4u_NVRAM_set_params(Nvram *nvram, uint16_t NVRAM_size,
|
||||||
const char *arch, ram_addr_t RAM_size,
|
const char *arch, ram_addr_t RAM_size,
|
||||||
const char *boot_devices,
|
const char *boot_devices,
|
||||||
uint32_t kernel_image, uint32_t kernel_size,
|
uint32_t kernel_image, uint32_t kernel_size,
|
||||||
@ -144,6 +144,7 @@ static int sun4u_NVRAM_set_params(M48t59State *nvram, uint16_t NVRAM_size,
|
|||||||
uint32_t start, end;
|
uint32_t start, end;
|
||||||
uint8_t image[0x1ff0];
|
uint8_t image[0x1ff0];
|
||||||
struct OpenBIOS_nvpart_v1 *part_header;
|
struct OpenBIOS_nvpart_v1 *part_header;
|
||||||
|
NvramClass *k = NVRAM_GET_CLASS(nvram);
|
||||||
|
|
||||||
memset(image, '\0', sizeof(image));
|
memset(image, '\0', sizeof(image));
|
||||||
|
|
||||||
@ -176,8 +177,9 @@ static int sun4u_NVRAM_set_params(M48t59State *nvram, uint16_t NVRAM_size,
|
|||||||
|
|
||||||
Sun_init_header((struct Sun_nvram *)&image[0x1fd8], macaddr, 0x80);
|
Sun_init_header((struct Sun_nvram *)&image[0x1fd8], macaddr, 0x80);
|
||||||
|
|
||||||
for (i = 0; i < sizeof(image); i++)
|
for (i = 0; i < sizeof(image); i++) {
|
||||||
m48t59_write(nvram, i, image[i]);
|
(k->write)(nvram, i, image[i]);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -610,7 +612,7 @@ pci_ebus_init1(PCIDevice *pci_dev)
|
|||||||
0, 0x1000000);
|
0, 0x1000000);
|
||||||
pci_register_bar(pci_dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &s->bar0);
|
pci_register_bar(pci_dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &s->bar0);
|
||||||
memory_region_init_alias(&s->bar1, OBJECT(s), "bar1", get_system_io(),
|
memory_region_init_alias(&s->bar1, OBJECT(s), "bar1", get_system_io(),
|
||||||
0, 0x1000);
|
0, 0x4000);
|
||||||
pci_register_bar(pci_dev, 1, PCI_BASE_ADDRESS_SPACE_IO, &s->bar1);
|
pci_register_bar(pci_dev, 1, PCI_BASE_ADDRESS_SPACE_IO, &s->bar1);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -818,11 +820,12 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
const struct hwdef *hwdef)
|
const struct hwdef *hwdef)
|
||||||
{
|
{
|
||||||
SPARCCPU *cpu;
|
SPARCCPU *cpu;
|
||||||
M48t59State *nvram;
|
Nvram *nvram;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
uint64_t initrd_addr, initrd_size, kernel_addr, kernel_size, kernel_entry;
|
uint64_t initrd_addr, initrd_size, kernel_addr, kernel_size, kernel_entry;
|
||||||
PCIBus *pci_bus, *pci_bus2, *pci_bus3;
|
PCIBus *pci_bus, *pci_bus2, *pci_bus3;
|
||||||
ISABus *isa_bus;
|
ISABus *isa_bus;
|
||||||
|
SysBusDevice *s;
|
||||||
qemu_irq *ivec_irqs, *pbm_irqs;
|
qemu_irq *ivec_irqs, *pbm_irqs;
|
||||||
DriveInfo *hd[MAX_IDE_BUS * MAX_IDE_DEVS];
|
DriveInfo *hd[MAX_IDE_BUS * MAX_IDE_DEVS];
|
||||||
DriveInfo *fd[MAX_FD];
|
DriveInfo *fd[MAX_FD];
|
||||||
@ -866,7 +869,12 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
fd[i] = drive_get(IF_FLOPPY, 0, i);
|
fd[i] = drive_get(IF_FLOPPY, 0, i);
|
||||||
}
|
}
|
||||||
fdctrl_init_isa(isa_bus, fd);
|
fdctrl_init_isa(isa_bus, fd);
|
||||||
nvram = m48t59_init_isa(isa_bus, 0x0074, NVRAM_SIZE, 59);
|
|
||||||
|
/* Map NVRAM into I/O (ebus) space */
|
||||||
|
nvram = m48t59_init(NULL, 0, 0, NVRAM_SIZE, 1968, 59);
|
||||||
|
s = SYS_BUS_DEVICE(nvram);
|
||||||
|
memory_region_add_subregion(get_system_io(), 0x2000,
|
||||||
|
sysbus_mmio_get_region(s, 0));
|
||||||
|
|
||||||
initrd_size = 0;
|
initrd_size = 0;
|
||||||
initrd_addr = 0;
|
initrd_addr = 0;
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
* QEMU M48T59 and M48T08 NVRAM emulation for PPC PREP and Sparc platforms
|
* QEMU M48T59 and M48T08 NVRAM emulation for PPC PREP and Sparc platforms
|
||||||
*
|
*
|
||||||
* Copyright (c) 2003-2005, 2007 Jocelyn Mayer
|
* Copyright (c) 2003-2005, 2007 Jocelyn Mayer
|
||||||
|
* Copyright (c) 2013 Hervé Poussineau
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
@ -37,12 +38,35 @@
|
|||||||
#define NVRAM_PRINTF(fmt, ...) do { } while (0)
|
#define NVRAM_PRINTF(fmt, ...) do { } while (0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define TYPE_M48TXX_SYS_BUS "sysbus-m48txx"
|
||||||
|
#define M48TXX_SYS_BUS_GET_CLASS(obj) \
|
||||||
|
OBJECT_GET_CLASS(M48txxSysBusDeviceClass, (obj), TYPE_M48TXX_SYS_BUS)
|
||||||
|
#define M48TXX_SYS_BUS_CLASS(klass) \
|
||||||
|
OBJECT_CLASS_CHECK(M48txxSysBusDeviceClass, (klass), TYPE_M48TXX_SYS_BUS)
|
||||||
|
#define M48TXX_SYS_BUS(obj) \
|
||||||
|
OBJECT_CHECK(M48txxSysBusState, (obj), TYPE_M48TXX_SYS_BUS)
|
||||||
|
|
||||||
|
#define TYPE_M48TXX_ISA "isa-m48txx"
|
||||||
|
#define M48TXX_ISA_GET_CLASS(obj) \
|
||||||
|
OBJECT_GET_CLASS(M48txxISADeviceClass, (obj), TYPE_M48TXX_ISA)
|
||||||
|
#define M48TXX_ISA_CLASS(klass) \
|
||||||
|
OBJECT_CLASS_CHECK(M48txxISADeviceClass, (klass), TYPE_M48TXX_ISA)
|
||||||
|
#define M48TXX_ISA(obj) \
|
||||||
|
OBJECT_CHECK(M48txxISAState, (obj), TYPE_M48TXX_ISA)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The M48T02, M48T08 and M48T59 chips are very similar. The newer '59 has
|
* The M48T02, M48T08 and M48T59 chips are very similar. The newer '59 has
|
||||||
* alarm and a watchdog timer and related control registers. In the
|
* alarm and a watchdog timer and related control registers. In the
|
||||||
* PPC platform there is also a nvram lock function.
|
* PPC platform there is also a nvram lock function.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
typedef struct M48txxInfo {
|
||||||
|
const char *isa_name;
|
||||||
|
const char *sysbus_name;
|
||||||
|
uint32_t model; /* 2 = m48t02, 8 = m48t08, 59 = m48t59 */
|
||||||
|
uint32_t size;
|
||||||
|
} M48txxInfo;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Chipset docs:
|
* Chipset docs:
|
||||||
* http://www.st.com/stonline/products/literature/ds/2410/m48t02.pdf
|
* http://www.st.com/stonline/products/literature/ds/2410/m48t02.pdf
|
||||||
@ -50,12 +74,12 @@
|
|||||||
* http://www.st.com/stonline/products/literature/od/7001/m48t59y.pdf
|
* http://www.st.com/stonline/products/literature/od/7001/m48t59y.pdf
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct M48t59State {
|
typedef struct M48t59State {
|
||||||
/* Hardware parameters */
|
/* Hardware parameters */
|
||||||
qemu_irq IRQ;
|
qemu_irq IRQ;
|
||||||
MemoryRegion iomem;
|
MemoryRegion iomem;
|
||||||
uint32_t io_base;
|
|
||||||
uint32_t size;
|
uint32_t size;
|
||||||
|
int32_t base_year;
|
||||||
/* RTC management */
|
/* RTC management */
|
||||||
time_t time_offset;
|
time_t time_offset;
|
||||||
time_t stop_time;
|
time_t stop_time;
|
||||||
@ -70,28 +94,51 @@ struct M48t59State {
|
|||||||
/* NVRAM storage */
|
/* NVRAM storage */
|
||||||
uint16_t addr;
|
uint16_t addr;
|
||||||
uint8_t lock;
|
uint8_t lock;
|
||||||
|
} M48t59State;
|
||||||
|
|
||||||
|
typedef struct M48txxISAState {
|
||||||
|
ISADevice parent_obj;
|
||||||
|
M48t59State state;
|
||||||
|
uint32_t io_base;
|
||||||
|
MemoryRegion io;
|
||||||
|
} M48txxISAState;
|
||||||
|
|
||||||
|
typedef struct M48txxISADeviceClass {
|
||||||
|
ISADeviceClass parent_class;
|
||||||
|
M48txxInfo info;
|
||||||
|
} M48txxISADeviceClass;
|
||||||
|
|
||||||
|
typedef struct M48txxSysBusState {
|
||||||
|
SysBusDevice parent_obj;
|
||||||
|
M48t59State state;
|
||||||
|
MemoryRegion io;
|
||||||
|
} M48txxSysBusState;
|
||||||
|
|
||||||
|
typedef struct M48txxSysBusDeviceClass {
|
||||||
|
SysBusDeviceClass parent_class;
|
||||||
|
M48txxInfo info;
|
||||||
|
} M48txxSysBusDeviceClass;
|
||||||
|
|
||||||
|
static M48txxInfo m48txx_info[] = {
|
||||||
|
{
|
||||||
|
.sysbus_name = "sysbus-m48t02",
|
||||||
|
.model = 2,
|
||||||
|
.size = 0x800,
|
||||||
|
},{
|
||||||
|
.sysbus_name = "sysbus-m48t08",
|
||||||
|
.model = 8,
|
||||||
|
.size = 0x2000,
|
||||||
|
},{
|
||||||
|
.sysbus_name = "sysbus-m48t59",
|
||||||
|
.model = 59,
|
||||||
|
.size = 0x2000,
|
||||||
|
},{
|
||||||
|
.isa_name = "isa-m48t59",
|
||||||
|
.model = 59,
|
||||||
|
.size = 0x2000,
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
#define TYPE_ISA_M48T59 "m48t59_isa"
|
|
||||||
#define ISA_M48T59(obj) \
|
|
||||||
OBJECT_CHECK(M48t59ISAState, (obj), TYPE_ISA_M48T59)
|
|
||||||
|
|
||||||
typedef struct M48t59ISAState {
|
|
||||||
ISADevice parent_obj;
|
|
||||||
|
|
||||||
M48t59State state;
|
|
||||||
MemoryRegion io;
|
|
||||||
} M48t59ISAState;
|
|
||||||
|
|
||||||
#define SYSBUS_M48T59(obj) \
|
|
||||||
OBJECT_CHECK(M48t59SysBusState, (obj), TYPE_SYSBUS_M48T59)
|
|
||||||
|
|
||||||
typedef struct M48t59SysBusState {
|
|
||||||
SysBusDevice parent_obj;
|
|
||||||
|
|
||||||
M48t59State state;
|
|
||||||
MemoryRegion io;
|
|
||||||
} M48t59SysBusState;
|
|
||||||
|
|
||||||
/* Fake timer functions */
|
/* Fake timer functions */
|
||||||
|
|
||||||
@ -198,9 +245,8 @@ static void set_up_watchdog(M48t59State *NVRAM, uint8_t value)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Direct access to NVRAM */
|
/* Direct access to NVRAM */
|
||||||
void m48t59_write (void *opaque, uint32_t addr, uint32_t val)
|
static void m48t59_write(M48t59State *NVRAM, uint32_t addr, uint32_t val)
|
||||||
{
|
{
|
||||||
M48t59State *NVRAM = opaque;
|
|
||||||
struct tm tm;
|
struct tm tm;
|
||||||
int tmp;
|
int tmp;
|
||||||
|
|
||||||
@ -346,11 +392,7 @@ void m48t59_write (void *opaque, uint32_t addr, uint32_t val)
|
|||||||
tmp = from_bcd(val);
|
tmp = from_bcd(val);
|
||||||
if (tmp >= 0 && tmp <= 99) {
|
if (tmp >= 0 && tmp <= 99) {
|
||||||
get_time(NVRAM, &tm);
|
get_time(NVRAM, &tm);
|
||||||
if (NVRAM->model == 8) {
|
tm.tm_year = from_bcd(val) + NVRAM->base_year - 1900;
|
||||||
tm.tm_year = from_bcd(val) + 68; // Base year is 1968
|
|
||||||
} else {
|
|
||||||
tm.tm_year = from_bcd(val);
|
|
||||||
}
|
|
||||||
set_time(NVRAM, &tm);
|
set_time(NVRAM, &tm);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -368,9 +410,8 @@ void m48t59_write (void *opaque, uint32_t addr, uint32_t val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t m48t59_read (void *opaque, uint32_t addr)
|
static uint32_t m48t59_read(M48t59State *NVRAM, uint32_t addr)
|
||||||
{
|
{
|
||||||
M48t59State *NVRAM = opaque;
|
|
||||||
struct tm tm;
|
struct tm tm;
|
||||||
uint32_t retval = 0xFF;
|
uint32_t retval = 0xFF;
|
||||||
|
|
||||||
@ -453,11 +494,7 @@ uint32_t m48t59_read (void *opaque, uint32_t addr)
|
|||||||
case 0x07FF:
|
case 0x07FF:
|
||||||
/* year */
|
/* year */
|
||||||
get_time(NVRAM, &tm);
|
get_time(NVRAM, &tm);
|
||||||
if (NVRAM->model == 8) {
|
retval = to_bcd((tm.tm_year + 1900 - NVRAM->base_year) % 100);
|
||||||
retval = to_bcd(tm.tm_year - 68); // Base year is 1968
|
|
||||||
} else {
|
|
||||||
retval = to_bcd(tm.tm_year);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
/* Check lock registers state */
|
/* Check lock registers state */
|
||||||
@ -477,10 +514,8 @@ uint32_t m48t59_read (void *opaque, uint32_t addr)
|
|||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
void m48t59_toggle_lock (void *opaque, int lock)
|
static void m48t59_toggle_lock(M48t59State *NVRAM, int lock)
|
||||||
{
|
{
|
||||||
M48t59State *NVRAM = opaque;
|
|
||||||
|
|
||||||
NVRAM->lock ^= 1 << lock;
|
NVRAM->lock ^= 1 << lock;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -616,7 +651,7 @@ static void m48t59_reset_common(M48t59State *NVRAM)
|
|||||||
|
|
||||||
static void m48t59_reset_isa(DeviceState *d)
|
static void m48t59_reset_isa(DeviceState *d)
|
||||||
{
|
{
|
||||||
M48t59ISAState *isa = ISA_M48T59(d);
|
M48txxISAState *isa = M48TXX_ISA(d);
|
||||||
M48t59State *NVRAM = &isa->state;
|
M48t59State *NVRAM = &isa->state;
|
||||||
|
|
||||||
m48t59_reset_common(NVRAM);
|
m48t59_reset_common(NVRAM);
|
||||||
@ -624,7 +659,7 @@ static void m48t59_reset_isa(DeviceState *d)
|
|||||||
|
|
||||||
static void m48t59_reset_sysbus(DeviceState *d)
|
static void m48t59_reset_sysbus(DeviceState *d)
|
||||||
{
|
{
|
||||||
M48t59SysBusState *sys = SYSBUS_M48T59(d);
|
M48txxSysBusState *sys = M48TXX_SYS_BUS(d);
|
||||||
M48t59State *NVRAM = &sys->state;
|
M48t59State *NVRAM = &sys->state;
|
||||||
|
|
||||||
m48t59_reset_common(NVRAM);
|
m48t59_reset_common(NVRAM);
|
||||||
@ -641,58 +676,63 @@ static const MemoryRegionOps m48t59_io_ops = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/* Initialisation routine */
|
/* Initialisation routine */
|
||||||
M48t59State *m48t59_init(qemu_irq IRQ, hwaddr mem_base,
|
Nvram *m48t59_init(qemu_irq IRQ, hwaddr mem_base,
|
||||||
uint32_t io_base, uint16_t size, int model)
|
uint32_t io_base, uint16_t size, int base_year,
|
||||||
|
int model)
|
||||||
{
|
{
|
||||||
DeviceState *dev;
|
DeviceState *dev;
|
||||||
SysBusDevice *s;
|
SysBusDevice *s;
|
||||||
M48t59SysBusState *d;
|
int i;
|
||||||
M48t59State *state;
|
|
||||||
|
|
||||||
dev = qdev_create(NULL, TYPE_SYSBUS_M48T59);
|
for (i = 0; i < ARRAY_SIZE(m48txx_info); i++) {
|
||||||
qdev_prop_set_uint32(dev, "model", model);
|
if (!m48txx_info[i].sysbus_name ||
|
||||||
qdev_prop_set_uint32(dev, "size", size);
|
m48txx_info[i].size != size ||
|
||||||
qdev_prop_set_uint32(dev, "io_base", io_base);
|
m48txx_info[i].model != model) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
dev = qdev_create(NULL, m48txx_info[i].sysbus_name);
|
||||||
|
qdev_prop_set_int32(dev, "base-year", base_year);
|
||||||
qdev_init_nofail(dev);
|
qdev_init_nofail(dev);
|
||||||
s = SYS_BUS_DEVICE(dev);
|
s = SYS_BUS_DEVICE(dev);
|
||||||
d = SYSBUS_M48T59(dev);
|
|
||||||
state = &d->state;
|
|
||||||
sysbus_connect_irq(s, 0, IRQ);
|
sysbus_connect_irq(s, 0, IRQ);
|
||||||
memory_region_init_io(&d->io, OBJECT(d), &m48t59_io_ops, state,
|
|
||||||
"m48t59", 4);
|
|
||||||
if (io_base != 0) {
|
if (io_base != 0) {
|
||||||
memory_region_add_subregion(get_system_io(), io_base, &d->io);
|
memory_region_add_subregion(get_system_io(), io_base,
|
||||||
|
sysbus_mmio_get_region(s, 1));
|
||||||
}
|
}
|
||||||
if (mem_base != 0) {
|
if (mem_base != 0) {
|
||||||
sysbus_mmio_map(s, 0, mem_base);
|
sysbus_mmio_map(s, 0, mem_base);
|
||||||
}
|
}
|
||||||
|
|
||||||
return state;
|
return NVRAM(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
M48t59State *m48t59_init_isa(ISABus *bus, uint32_t io_base, uint16_t size,
|
assert(false);
|
||||||
int model)
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
Nvram *m48t59_init_isa(ISABus *bus, uint32_t io_base, uint16_t size,
|
||||||
|
int base_year, int model)
|
||||||
{
|
{
|
||||||
M48t59ISAState *d;
|
|
||||||
ISADevice *isadev;
|
|
||||||
DeviceState *dev;
|
DeviceState *dev;
|
||||||
M48t59State *s;
|
int i;
|
||||||
|
|
||||||
isadev = isa_create(bus, TYPE_ISA_M48T59);
|
for (i = 0; i < ARRAY_SIZE(m48txx_info); i++) {
|
||||||
dev = DEVICE(isadev);
|
if (!m48txx_info[i].isa_name ||
|
||||||
qdev_prop_set_uint32(dev, "model", model);
|
m48txx_info[i].size != size ||
|
||||||
qdev_prop_set_uint32(dev, "size", size);
|
m48txx_info[i].model != model) {
|
||||||
qdev_prop_set_uint32(dev, "io_base", io_base);
|
continue;
|
||||||
qdev_init_nofail(dev);
|
|
||||||
d = ISA_M48T59(isadev);
|
|
||||||
s = &d->state;
|
|
||||||
|
|
||||||
memory_region_init_io(&d->io, OBJECT(d), &m48t59_io_ops, s, "m48t59", 4);
|
|
||||||
if (io_base != 0) {
|
|
||||||
isa_register_ioport(isadev, &d->io, io_base);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return s;
|
dev = DEVICE(isa_create(bus, m48txx_info[i].isa_name));
|
||||||
|
qdev_prop_set_uint32(dev, "iobase", io_base);
|
||||||
|
qdev_prop_set_int32(dev, "base-year", base_year);
|
||||||
|
qdev_init_nofail(dev);
|
||||||
|
return NVRAM(dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(false);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void m48t59_realize_common(M48t59State *s, Error **errp)
|
static void m48t59_realize_common(M48t59State *s, Error **errp)
|
||||||
@ -709,25 +749,38 @@ static void m48t59_realize_common(M48t59State *s, Error **errp)
|
|||||||
|
|
||||||
static void m48t59_isa_realize(DeviceState *dev, Error **errp)
|
static void m48t59_isa_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
|
M48txxISADeviceClass *u = M48TXX_ISA_GET_CLASS(dev);
|
||||||
ISADevice *isadev = ISA_DEVICE(dev);
|
ISADevice *isadev = ISA_DEVICE(dev);
|
||||||
M48t59ISAState *d = ISA_M48T59(dev);
|
M48txxISAState *d = M48TXX_ISA(dev);
|
||||||
M48t59State *s = &d->state;
|
M48t59State *s = &d->state;
|
||||||
|
|
||||||
|
s->model = u->info.model;
|
||||||
|
s->size = u->info.size;
|
||||||
isa_init_irq(isadev, &s->IRQ, 8);
|
isa_init_irq(isadev, &s->IRQ, 8);
|
||||||
m48t59_realize_common(s, errp);
|
m48t59_realize_common(s, errp);
|
||||||
|
memory_region_init_io(&d->io, OBJECT(dev), &m48t59_io_ops, s, "m48t59", 4);
|
||||||
|
if (d->io_base != 0) {
|
||||||
|
isa_register_ioport(isadev, &d->io, d->io_base);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int m48t59_init1(SysBusDevice *dev)
|
static int m48t59_init1(SysBusDevice *dev)
|
||||||
{
|
{
|
||||||
M48t59SysBusState *d = SYSBUS_M48T59(dev);
|
M48txxSysBusDeviceClass *u = M48TXX_SYS_BUS_GET_CLASS(dev);
|
||||||
|
M48txxSysBusState *d = M48TXX_SYS_BUS(dev);
|
||||||
|
Object *o = OBJECT(dev);
|
||||||
M48t59State *s = &d->state;
|
M48t59State *s = &d->state;
|
||||||
Error *err = NULL;
|
Error *err = NULL;
|
||||||
|
|
||||||
|
s->model = u->info.model;
|
||||||
|
s->size = u->info.size;
|
||||||
sysbus_init_irq(dev, &s->IRQ);
|
sysbus_init_irq(dev, &s->IRQ);
|
||||||
|
|
||||||
memory_region_init_io(&s->iomem, OBJECT(d), &nvram_ops, s,
|
memory_region_init_io(&s->iomem, o, &nvram_ops, s, "m48t59.nvram",
|
||||||
"m48t59.nvram", s->size);
|
s->size);
|
||||||
|
memory_region_init_io(&d->io, o, &m48t59_io_ops, s, "m48t59", 4);
|
||||||
sysbus_init_mmio(dev, &s->iomem);
|
sysbus_init_mmio(dev, &s->iomem);
|
||||||
|
sysbus_init_mmio(dev, &d->io);
|
||||||
m48t59_realize_common(s, &err);
|
m48t59_realize_common(s, &err);
|
||||||
if (err != NULL) {
|
if (err != NULL) {
|
||||||
error_free(err);
|
error_free(err);
|
||||||
@ -737,59 +790,157 @@ static int m48t59_init1(SysBusDevice *dev)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint32_t m48txx_isa_read(Nvram *obj, uint32_t addr)
|
||||||
|
{
|
||||||
|
M48txxISAState *d = M48TXX_ISA(obj);
|
||||||
|
return m48t59_read(&d->state, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void m48txx_isa_write(Nvram *obj, uint32_t addr, uint32_t val)
|
||||||
|
{
|
||||||
|
M48txxISAState *d = M48TXX_ISA(obj);
|
||||||
|
m48t59_write(&d->state, addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void m48txx_isa_toggle_lock(Nvram *obj, int lock)
|
||||||
|
{
|
||||||
|
M48txxISAState *d = M48TXX_ISA(obj);
|
||||||
|
m48t59_toggle_lock(&d->state, lock);
|
||||||
|
}
|
||||||
|
|
||||||
static Property m48t59_isa_properties[] = {
|
static Property m48t59_isa_properties[] = {
|
||||||
DEFINE_PROP_UINT32("size", M48t59ISAState, state.size, -1),
|
DEFINE_PROP_INT32("base-year", M48txxISAState, state.base_year, 0),
|
||||||
DEFINE_PROP_UINT32("model", M48t59ISAState, state.model, -1),
|
DEFINE_PROP_UINT32("iobase", M48txxISAState, io_base, 0x74),
|
||||||
DEFINE_PROP_UINT32("io_base", M48t59ISAState, state.io_base, 0),
|
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
|
||||||
static void m48t59_isa_class_init(ObjectClass *klass, void *data)
|
static void m48txx_isa_class_init(ObjectClass *klass, void *data)
|
||||||
{
|
{
|
||||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||||
|
NvramClass *nc = NVRAM_CLASS(klass);
|
||||||
|
|
||||||
dc->realize = m48t59_isa_realize;
|
dc->realize = m48t59_isa_realize;
|
||||||
dc->reset = m48t59_reset_isa;
|
dc->reset = m48t59_reset_isa;
|
||||||
dc->props = m48t59_isa_properties;
|
dc->props = m48t59_isa_properties;
|
||||||
/* Reason: needs to be wired up by m48t59_init_isa() */
|
nc->read = m48txx_isa_read;
|
||||||
dc->cannot_instantiate_with_device_add_yet = true;
|
nc->write = m48txx_isa_write;
|
||||||
|
nc->toggle_lock = m48txx_isa_toggle_lock;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo m48t59_isa_info = {
|
static void m48txx_isa_concrete_class_init(ObjectClass *klass, void *data)
|
||||||
.name = TYPE_ISA_M48T59,
|
{
|
||||||
.parent = TYPE_ISA_DEVICE,
|
M48txxISADeviceClass *u = M48TXX_ISA_CLASS(klass);
|
||||||
.instance_size = sizeof(M48t59ISAState),
|
M48txxInfo *info = data;
|
||||||
.class_init = m48t59_isa_class_init,
|
|
||||||
};
|
|
||||||
|
|
||||||
static Property m48t59_properties[] = {
|
u->info = *info;
|
||||||
DEFINE_PROP_UINT32("size", M48t59SysBusState, state.size, -1),
|
}
|
||||||
DEFINE_PROP_UINT32("model", M48t59SysBusState, state.model, -1),
|
|
||||||
DEFINE_PROP_UINT32("io_base", M48t59SysBusState, state.io_base, 0),
|
static uint32_t m48txx_sysbus_read(Nvram *obj, uint32_t addr)
|
||||||
|
{
|
||||||
|
M48txxSysBusState *d = M48TXX_SYS_BUS(obj);
|
||||||
|
return m48t59_read(&d->state, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void m48txx_sysbus_write(Nvram *obj, uint32_t addr, uint32_t val)
|
||||||
|
{
|
||||||
|
M48txxSysBusState *d = M48TXX_SYS_BUS(obj);
|
||||||
|
m48t59_write(&d->state, addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void m48txx_sysbus_toggle_lock(Nvram *obj, int lock)
|
||||||
|
{
|
||||||
|
M48txxSysBusState *d = M48TXX_SYS_BUS(obj);
|
||||||
|
m48t59_toggle_lock(&d->state, lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Property m48t59_sysbus_properties[] = {
|
||||||
|
DEFINE_PROP_INT32("base-year", M48txxSysBusState, state.base_year, 0),
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
|
||||||
static void m48t59_class_init(ObjectClass *klass, void *data)
|
static void m48txx_sysbus_class_init(ObjectClass *klass, void *data)
|
||||||
{
|
{
|
||||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||||
SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
|
SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
|
||||||
|
NvramClass *nc = NVRAM_CLASS(klass);
|
||||||
|
|
||||||
k->init = m48t59_init1;
|
k->init = m48t59_init1;
|
||||||
dc->reset = m48t59_reset_sysbus;
|
dc->reset = m48t59_reset_sysbus;
|
||||||
dc->props = m48t59_properties;
|
dc->props = m48t59_sysbus_properties;
|
||||||
|
nc->read = m48txx_sysbus_read;
|
||||||
|
nc->write = m48txx_sysbus_write;
|
||||||
|
nc->toggle_lock = m48txx_sysbus_toggle_lock;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo m48t59_info = {
|
static void m48txx_sysbus_concrete_class_init(ObjectClass *klass, void *data)
|
||||||
.name = TYPE_SYSBUS_M48T59,
|
{
|
||||||
|
M48txxSysBusDeviceClass *u = M48TXX_SYS_BUS_CLASS(klass);
|
||||||
|
M48txxInfo *info = data;
|
||||||
|
|
||||||
|
u->info = *info;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TypeInfo nvram_info = {
|
||||||
|
.name = TYPE_NVRAM,
|
||||||
|
.parent = TYPE_INTERFACE,
|
||||||
|
.class_size = sizeof(NvramClass),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const TypeInfo m48txx_sysbus_type_info = {
|
||||||
|
.name = TYPE_M48TXX_SYS_BUS,
|
||||||
.parent = TYPE_SYS_BUS_DEVICE,
|
.parent = TYPE_SYS_BUS_DEVICE,
|
||||||
.instance_size = sizeof(M48t59SysBusState),
|
.instance_size = sizeof(M48txxSysBusState),
|
||||||
.class_init = m48t59_class_init,
|
.abstract = true,
|
||||||
|
.class_init = m48txx_sysbus_class_init,
|
||||||
|
.interfaces = (InterfaceInfo[]) {
|
||||||
|
{ TYPE_NVRAM },
|
||||||
|
{ }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
static const TypeInfo m48txx_isa_type_info = {
|
||||||
|
.name = TYPE_M48TXX_ISA,
|
||||||
|
.parent = TYPE_ISA_DEVICE,
|
||||||
|
.instance_size = sizeof(M48txxISAState),
|
||||||
|
.abstract = true,
|
||||||
|
.class_init = m48txx_isa_class_init,
|
||||||
|
.interfaces = (InterfaceInfo[]) {
|
||||||
|
{ TYPE_NVRAM },
|
||||||
|
{ }
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
static void m48t59_register_types(void)
|
static void m48t59_register_types(void)
|
||||||
{
|
{
|
||||||
type_register_static(&m48t59_info);
|
TypeInfo sysbus_type_info = {
|
||||||
type_register_static(&m48t59_isa_info);
|
.parent = TYPE_M48TXX_SYS_BUS,
|
||||||
|
.class_size = sizeof(M48txxSysBusDeviceClass),
|
||||||
|
.class_init = m48txx_sysbus_concrete_class_init,
|
||||||
|
};
|
||||||
|
TypeInfo isa_type_info = {
|
||||||
|
.parent = TYPE_M48TXX_ISA,
|
||||||
|
.class_size = sizeof(M48txxISADeviceClass),
|
||||||
|
.class_init = m48txx_isa_concrete_class_init,
|
||||||
|
};
|
||||||
|
int i;
|
||||||
|
|
||||||
|
type_register_static(&nvram_info);
|
||||||
|
type_register_static(&m48txx_sysbus_type_info);
|
||||||
|
type_register_static(&m48txx_isa_type_info);
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(m48txx_info); i++) {
|
||||||
|
if (m48txx_info[i].sysbus_name) {
|
||||||
|
sysbus_type_info.name = m48txx_info[i].sysbus_name;
|
||||||
|
sysbus_type_info.class_data = &m48txx_info[i];
|
||||||
|
type_register(&sysbus_type_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m48txx_info[i].isa_name) {
|
||||||
|
isa_type_info.name = m48txx_info[i].isa_name;
|
||||||
|
isa_type_info.class_data = &m48txx_info[i];
|
||||||
|
type_register(&isa_type_info);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type_init(m48t59_register_types)
|
type_init(m48t59_register_types)
|
||||||
|
@ -1,37 +1,34 @@
|
|||||||
#ifndef NVRAM_H
|
#ifndef NVRAM_H
|
||||||
#define NVRAM_H
|
#define NVRAM_H
|
||||||
|
|
||||||
/* NVRAM helpers */
|
#include "qemu-common.h"
|
||||||
typedef uint32_t (*nvram_read_t)(void *private, uint32_t addr);
|
#include "qom/object.h"
|
||||||
typedef void (*nvram_write_t)(void *private, uint32_t addr, uint32_t val);
|
|
||||||
typedef struct nvram_t {
|
|
||||||
void *opaque;
|
|
||||||
nvram_read_t read_fn;
|
|
||||||
nvram_write_t write_fn;
|
|
||||||
} nvram_t;
|
|
||||||
|
|
||||||
uint32_t NVRAM_get_lword (nvram_t *nvram, uint32_t addr);
|
#define TYPE_NVRAM "nvram"
|
||||||
int NVRAM_get_string (nvram_t *nvram, uint8_t *dst, uint16_t addr, int max);
|
|
||||||
|
|
||||||
int PPC_NVRAM_set_params (nvram_t *nvram, uint16_t NVRAM_size,
|
#define NVRAM_CLASS(klass) \
|
||||||
const char *arch,
|
OBJECT_CLASS_CHECK(NvramClass, (klass), TYPE_NVRAM)
|
||||||
uint32_t RAM_size, int boot_device,
|
#define NVRAM_GET_CLASS(obj) \
|
||||||
uint32_t kernel_image, uint32_t kernel_size,
|
OBJECT_GET_CLASS(NvramClass, (obj), TYPE_NVRAM)
|
||||||
const char *cmdline,
|
#define NVRAM(obj) \
|
||||||
uint32_t initrd_image, uint32_t initrd_size,
|
INTERFACE_CHECK(Nvram, (obj), TYPE_NVRAM)
|
||||||
uint32_t NVRAM_image,
|
|
||||||
int width, int height, int depth);
|
|
||||||
|
|
||||||
#define TYPE_SYSBUS_M48T59 "m48t59"
|
typedef struct Nvram {
|
||||||
|
Object parent;
|
||||||
|
} Nvram;
|
||||||
|
|
||||||
typedef struct M48t59State M48t59State;
|
typedef struct NvramClass {
|
||||||
|
InterfaceClass parent;
|
||||||
|
|
||||||
void m48t59_write (void *private, uint32_t addr, uint32_t val);
|
uint32_t (*read)(Nvram *obj, uint32_t addr);
|
||||||
uint32_t m48t59_read (void *private, uint32_t addr);
|
void (*write)(Nvram *obj, uint32_t addr, uint32_t val);
|
||||||
void m48t59_toggle_lock (void *private, int lock);
|
void (*toggle_lock)(Nvram *obj, int lock);
|
||||||
M48t59State *m48t59_init_isa(ISABus *bus, uint32_t io_base, uint16_t size,
|
} NvramClass;
|
||||||
|
|
||||||
|
Nvram *m48t59_init_isa(ISABus *bus, uint32_t io_base, uint16_t size,
|
||||||
|
int base_year, int type);
|
||||||
|
Nvram *m48t59_init(qemu_irq IRQ, hwaddr mem_base,
|
||||||
|
uint32_t io_base, uint16_t size, int base_year,
|
||||||
int type);
|
int type);
|
||||||
M48t59State *m48t59_init(qemu_irq IRQ, hwaddr mem_base,
|
|
||||||
uint32_t io_base, uint16_t size, int type);
|
|
||||||
|
|
||||||
#endif /* !NVRAM_H */
|
#endif /* !NVRAM_H */
|
||||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -2052,7 +2052,7 @@ firmware implementation. The goal is to implement a 100% IEEE
|
|||||||
|
|
||||||
A sample Linux 2.6 series kernel and ram disk image are available on
|
A sample Linux 2.6 series kernel and ram disk image are available on
|
||||||
the QEMU web site. There are still issues with NetBSD and OpenBSD, but
|
the QEMU web site. There are still issues with NetBSD and OpenBSD, but
|
||||||
some kernel versions work. Please note that currently older Solaris kernels
|
most kernel versions work. Please note that currently older Solaris kernels
|
||||||
don't work probably due to interface issues between OpenBIOS and
|
don't work probably due to interface issues between OpenBIOS and
|
||||||
Solaris.
|
Solaris.
|
||||||
|
|
||||||
@ -2091,8 +2091,9 @@ Set the emulated machine type. Default is SS-5.
|
|||||||
|
|
||||||
Use the executable @file{qemu-system-sparc64} to simulate a Sun4u
|
Use the executable @file{qemu-system-sparc64} to simulate a Sun4u
|
||||||
(UltraSPARC PC-like machine), Sun4v (T1 PC-like machine), or generic
|
(UltraSPARC PC-like machine), Sun4v (T1 PC-like machine), or generic
|
||||||
Niagara (T1) machine. The emulator is not usable for anything yet, but
|
Niagara (T1) machine. The Sun4u emulator is mostly complete, being
|
||||||
it can launch some kernels.
|
able to run Linux, NetBSD and OpenBSD in headless (-nographic) mode. The
|
||||||
|
Sun4v and Niagara emulators are still a work in progress.
|
||||||
|
|
||||||
QEMU emulates the following peripherals:
|
QEMU emulates the following peripherals:
|
||||||
|
|
||||||
|
@ -1 +1 @@
|
|||||||
Subproject commit 038aa78d3c331731733378a73e778ee620a5b9da
|
Subproject commit b8dea39718916f4d7d391cd1664314d52a45b872
|
Loading…
Reference in New Issue
Block a user