2017-05-30 03:38:38 +02:00
|
|
|
/*
|
|
|
|
|
* 86Box A hypervisor and IBM PC system emulator that specializes in
|
|
|
|
|
* running old operating systems and software designed for IBM
|
|
|
|
|
* PC systems and compatibles from 1981 through fairly recent
|
|
|
|
|
* system designs based on the PCI bus.
|
|
|
|
|
*
|
|
|
|
|
* This file is part of the 86Box distribution.
|
|
|
|
|
*
|
|
|
|
|
* CMOS NVRAM emulation.
|
|
|
|
|
*
|
|
|
|
|
* Version: @(#)nvr.c 1.0.0 2017/05/30
|
|
|
|
|
*
|
|
|
|
|
* Author: Sarah Walker, <http://pcem-emulator.co.uk/>
|
|
|
|
|
* Miran Grca, <mgrca8@gmail.com>
|
|
|
|
|
* Mahod,
|
|
|
|
|
* Copyright 2008-2017 Sarah Walker.
|
|
|
|
|
* Copyright 2016-2017 Miran Grca.
|
|
|
|
|
* Copyright 2016-2017 Mahod.
|
|
|
|
|
*/
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
|
#include "ibm.h"
|
|
|
|
|
#include "io.h"
|
2017-05-06 04:02:03 +02:00
|
|
|
#include "mem.h"
|
2016-06-26 00:34:39 +02:00
|
|
|
#include "nvr.h"
|
|
|
|
|
#include "pic.h"
|
2017-05-06 04:02:03 +02:00
|
|
|
#include "rom.h"
|
2016-06-26 00:34:39 +02:00
|
|
|
#include "timer.h"
|
2016-07-11 01:27:23 +02:00
|
|
|
#include "rtc.h"
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
int oldromset;
|
|
|
|
|
int nvrmask=63;
|
2017-05-05 01:49:42 +02:00
|
|
|
char nvrram[128];
|
2016-06-26 00:34:39 +02:00
|
|
|
int nvraddr;
|
|
|
|
|
|
|
|
|
|
int nvr_dosave = 0;
|
|
|
|
|
|
2016-11-07 06:39:20 +01:00
|
|
|
static int nvr_onesec_time = 0, nvr_onesec_cnt = 0;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
2016-12-23 03:16:24 +01:00
|
|
|
static int rtctime;
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
void getnvrtime()
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
time_get(nvrram);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void nvr_recalc()
|
|
|
|
|
{
|
|
|
|
|
int c;
|
|
|
|
|
int newrtctime;
|
2016-08-15 23:40:02 +02:00
|
|
|
c = 1 << ((nvrram[RTC_REGA] & RTC_RS) - 1);
|
2016-06-26 00:34:39 +02:00
|
|
|
newrtctime=(int)(RTCCONST * c * (1 << TIMER_SHIFT));
|
|
|
|
|
if (rtctime>newrtctime) rtctime=newrtctime;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void nvr_rtc(void *p)
|
|
|
|
|
{
|
|
|
|
|
int c;
|
2016-08-15 23:40:02 +02:00
|
|
|
if (!(nvrram[RTC_REGA] & RTC_RS))
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
rtctime=0x7fffffff;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-08-15 23:40:02 +02:00
|
|
|
c = 1 << ((nvrram[RTC_REGA] & RTC_RS) - 1);
|
2016-06-26 00:34:39 +02:00
|
|
|
rtctime += (int)(RTCCONST * c * (1 << TIMER_SHIFT));
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_PF;
|
|
|
|
|
if (nvrram[RTC_REGB] & RTC_PIE)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_IRQF;
|
2016-06-26 00:34:39 +02:00
|
|
|
if (AMSTRAD) picint(2);
|
|
|
|
|
else picint(0x100);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-11 01:27:23 +02:00
|
|
|
int nvr_update_status = 0;
|
|
|
|
|
|
|
|
|
|
#define ALARM_DONTCARE 0xc0
|
|
|
|
|
|
|
|
|
|
int nvr_check_alarm(int nvraddr)
|
|
|
|
|
{
|
|
|
|
|
return (nvrram[nvraddr + 1] == nvrram[nvraddr] || (nvrram[nvraddr + 1] & ALARM_DONTCARE) == ALARM_DONTCARE);
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-07 06:39:20 +01:00
|
|
|
int nvr_update_end_count = 0;
|
2016-07-19 02:44:32 +02:00
|
|
|
|
|
|
|
|
void nvr_update_end(void *p)
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (!(nvrram[RTC_REGB] & RTC_SET))
|
2016-07-19 02:44:32 +02:00
|
|
|
{
|
|
|
|
|
getnvrtime();
|
|
|
|
|
/* Clear update status. */
|
|
|
|
|
nvr_update_status = 0;
|
|
|
|
|
|
2016-08-15 23:40:02 +02:00
|
|
|
if (nvr_check_alarm(RTC_SECONDS) && nvr_check_alarm(RTC_MINUTES) && nvr_check_alarm(RTC_HOURS))
|
2016-07-19 02:44:32 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_AF;
|
|
|
|
|
if (nvrram[RTC_REGB] & RTC_AIE)
|
2016-07-19 02:44:32 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_IRQF;
|
2016-07-19 02:44:32 +02:00
|
|
|
if (AMSTRAD) picint(2);
|
|
|
|
|
else picint(0x100);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* The flag and interrupt should be issued on update ended, not started. */
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_UF;
|
|
|
|
|
if (nvrram[RTC_REGB] & RTC_UIE)
|
2016-07-19 02:44:32 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGC] |= RTC_IRQF;
|
2016-07-19 02:44:32 +02:00
|
|
|
if (AMSTRAD) picint(2);
|
|
|
|
|
else picint(0x100);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nvr_update_end_count = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
void nvr_onesec(void *p)
|
|
|
|
|
{
|
|
|
|
|
nvr_onesec_cnt++;
|
2016-07-19 02:44:32 +02:00
|
|
|
if (nvr_onesec_cnt >= 100)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (!(nvrram[RTC_REGB] & RTC_SET))
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvr_update_status = RTC_UIP;
|
2016-08-15 23:33:51 +02:00
|
|
|
rtc_tick();
|
2016-07-11 01:27:23 +02:00
|
|
|
|
2016-07-19 02:44:32 +02:00
|
|
|
nvr_update_end_count = (int)((244.0 + 1984.0) * TIMER_USEC);
|
2016-07-11 01:27:23 +02:00
|
|
|
}
|
2016-08-15 23:33:51 +02:00
|
|
|
nvr_onesec_cnt = 0;
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
2016-07-19 02:44:32 +02:00
|
|
|
nvr_onesec_time += (int)(10000 * TIMER_USEC);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void writenvr(uint16_t addr, uint8_t val, void *priv)
|
|
|
|
|
{
|
|
|
|
|
int c, old;
|
|
|
|
|
if (addr&1)
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (nvraddr==RTC_REGC || nvraddr==RTC_REGD)
|
|
|
|
|
return; /* Registers C and D are read-only. There's no reason to continue. */
|
|
|
|
|
if (nvraddr > RTC_REGD && nvrram[nvraddr] != val)
|
2016-06-26 00:34:39 +02:00
|
|
|
nvr_dosave = 1;
|
2016-07-11 01:27:23 +02:00
|
|
|
|
|
|
|
|
old = nvrram[nvraddr];
|
|
|
|
|
nvrram[nvraddr]=val;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
2016-08-15 23:40:02 +02:00
|
|
|
if (nvraddr == RTC_REGA)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (val & RTC_RS)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
c = 1 << ((val & RTC_RS) - 1);
|
2016-06-26 00:34:39 +02:00
|
|
|
rtctime += (int)(RTCCONST * c * (1 << TIMER_SHIFT));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
rtctime = 0x7fffffff;
|
|
|
|
|
}
|
2016-07-11 01:27:23 +02:00
|
|
|
else
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (nvraddr == RTC_REGB)
|
2016-07-11 01:27:23 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (((old ^ val) & RTC_SET) && (val & RTC_SET))
|
2016-07-11 01:27:23 +02:00
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGA] &= ~RTC_UIP; /* This has to be done according to the datasheet. */
|
|
|
|
|
nvrram[RTC_REGB] &= ~RTC_UIE; /* This also has to happen per the specification. */
|
2016-07-11 01:27:23 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 23:40:02 +02:00
|
|
|
if ((nvraddr < RTC_REGA) || (nvraddr == RTC_CENTURY))
|
2016-07-11 01:27:23 +02:00
|
|
|
{
|
2016-07-11 02:28:22 +02:00
|
|
|
if ((nvraddr != 1) && (nvraddr != 3) && (nvraddr != 5))
|
2016-07-11 01:27:23 +02:00
|
|
|
{
|
|
|
|
|
if ((old != val) && !enable_sync)
|
|
|
|
|
{
|
|
|
|
|
time_update(nvrram, nvraddr);
|
|
|
|
|
nvr_dosave = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
else nvraddr=val&nvrmask;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t readnvr(uint16_t addr, void *priv)
|
|
|
|
|
{
|
|
|
|
|
uint8_t temp;
|
|
|
|
|
if (addr&1)
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
if (nvraddr == RTC_REGA)
|
|
|
|
|
return ((nvrram[RTC_REGA] & 0x7F) | nvr_update_status);
|
|
|
|
|
if (nvraddr == RTC_REGD)
|
|
|
|
|
nvrram[RTC_REGD] |= RTC_VRT;
|
|
|
|
|
if (nvraddr == RTC_REGC)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
if (AMSTRAD) picintc(2);
|
|
|
|
|
else picintc(0x100);
|
2016-08-15 23:40:02 +02:00
|
|
|
temp = nvrram[RTC_REGC];
|
|
|
|
|
nvrram[RTC_REGC] = 0;
|
2016-06-26 00:34:39 +02:00
|
|
|
return temp;
|
|
|
|
|
}
|
|
|
|
|
return nvrram[nvraddr];
|
|
|
|
|
}
|
|
|
|
|
return nvraddr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void loadnvr()
|
|
|
|
|
{
|
|
|
|
|
FILE *f;
|
|
|
|
|
int c;
|
|
|
|
|
nvrmask=63;
|
|
|
|
|
oldromset=romset;
|
|
|
|
|
switch (romset)
|
|
|
|
|
{
|
2017-05-06 04:02:03 +02:00
|
|
|
case ROM_PC1512: f = nvrfopen(L"pc1512.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PC1640: f = nvrfopen(L"pc1640.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PC200: f = nvrfopen(L"pc200.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PC2086: f = nvrfopen(L"pc2086.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PC3086: f = nvrfopen(L"pc3086.nvr", L"rb"); break;
|
|
|
|
|
case ROM_IBMAT: f = nvrfopen(L"at.nvr", L"rb"); break;
|
|
|
|
|
case ROM_IBMPS1_2011: f = nvrfopen(L"ibmps1_2011.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_IBMPS1_2121: f = nvrfopen(L"ibmps1_2121.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_IBMPS1_2121_ISA: f = nvrfopen(L"ibmps1_2121_isa.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_IBMPS2_M30_286: f = nvrfopen(L"ibmps2_m30_286.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_IBMPS2_M50: f = nvrfopen(L"ibmps2_m50.nvr", L"rb"); break;
|
|
|
|
|
case ROM_IBMPS2_M55SX: f = nvrfopen(L"ibmps2_m55sx.nvr", L"rb"); break;
|
|
|
|
|
case ROM_IBMPS2_M80: f = nvrfopen(L"ibmps2_m80.nvr", L"rb"); break;
|
|
|
|
|
case ROM_CMDPC30: f = nvrfopen(L"cmdpc30.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_PORTABLEII: f = nvrfopen(L"portableii.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PORTABLEIII: f = nvrfopen(L"portableiii.nvr", L"rb"); break;
|
|
|
|
|
case ROM_AMI286: f = nvrfopen(L"ami286.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_AWARD286: f = nvrfopen(L"award286.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_SUPER286TR: f = nvrfopen(L"super286tr.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_SPC4200P: f = nvrfopen(L"spc4200p.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_IBMAT386: f = nvrfopen(L"at386.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_DESKPRO_386: f = nvrfopen(L"deskpro386.nvr", L"rb"); break;
|
|
|
|
|
case ROM_PORTABLEIII386: f = nvrfopen(L"portableiii386.nvr", L"rb"); break;
|
|
|
|
|
case ROM_MEGAPC: f = nvrfopen(L"megapc.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_MEGAPCDX: f = nvrfopen(L"megapcdx.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_AMI386SX: f = nvrfopen(L"ami386.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_AMI486: f = nvrfopen(L"ami486.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_WIN486: f = nvrfopen(L"win486.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_SIS496: f = nvrfopen(L"sis496.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_430VX: f = nvrfopen(L"430vx.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_REVENGE: f = nvrfopen(L"revenge.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_ENDEAVOR: f = nvrfopen(L"endeavor.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_DTK386: f = nvrfopen(L"dtk386.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_MR386DX_OPTI495: f = nvrfopen(L"mr386dx_opti495.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_AMI386DX_OPTI495: f = nvrfopen(L"ami386dx_opti495.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_DTK486: f = nvrfopen(L"dtk486.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_R418: f = nvrfopen(L"r418.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_586MC1: f = nvrfopen(L"586mc1.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_PLATO: f = nvrfopen(L"plato.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_MB500N: f = nvrfopen(L"mb500n.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_P54TP4XE: f = nvrfopen(L"p54tp4xe.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_AP53: f = nvrfopen(L"ap53.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_P55T2S: f = nvrfopen(L"p55t2s.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_ACERM3A: f = nvrfopen(L"acerm3a.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_ACERV35N: f = nvrfopen(L"acerv35n.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_P55VA: f = nvrfopen(L"p55va.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_P55T2P4: f = nvrfopen(L"p55t2p4.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_P55TVP4: f = nvrfopen(L"p55tvp4.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_440FX: f = nvrfopen(L"440fx.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_THOR: f = nvrfopen(L"thor.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_MRTHOR: f = nvrfopen(L"mrthor.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_ZAPPA: f = nvrfopen(L"zappa.nvr", L"rb"); nvrmask = 127; break;
|
|
|
|
|
case ROM_S1668: f = nvrfopen(L"tpatx.nvr", L"rb"); nvrmask = 127; break;
|
2017-05-29 01:18:32 +02:00
|
|
|
case ROM_IBMPS1_2133: f = nvrfopen(L"ibmps1_2133.nvr", L"rb"); nvrmask = 127; break;
|
2016-06-26 00:34:39 +02:00
|
|
|
default: return;
|
|
|
|
|
}
|
|
|
|
|
if (!f)
|
|
|
|
|
{
|
|
|
|
|
memset(nvrram,0xFF,128);
|
2016-07-11 01:27:23 +02:00
|
|
|
if (!enable_sync)
|
|
|
|
|
{
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_SECONDS] = nvrram[RTC_MINUTES] = nvrram[RTC_HOURS] = 0;
|
|
|
|
|
nvrram[RTC_DOM] = nvrram[RTC_MONTH] = 1;
|
|
|
|
|
nvrram[RTC_YEAR] = BCD(80);
|
|
|
|
|
nvrram[RTC_CENTURY] = BCD(19);
|
|
|
|
|
nvrram[RTC_REGB] = RTC_2412;
|
2016-07-11 01:27:23 +02:00
|
|
|
}
|
2016-06-26 00:34:39 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
fread(nvrram,128,1,f);
|
2016-08-15 23:33:51 +02:00
|
|
|
if (enable_sync)
|
|
|
|
|
time_internal_sync(nvrram);
|
|
|
|
|
else
|
|
|
|
|
time_internal_set_nvrram(nvrram); /* Update the internal clock state based on the NVR registers. */
|
2016-06-26 00:34:39 +02:00
|
|
|
fclose(f);
|
2016-08-15 23:40:02 +02:00
|
|
|
nvrram[RTC_REGA] = 6;
|
|
|
|
|
nvrram[RTC_REGB] = RTC_2412;
|
|
|
|
|
c = 1 << ((nvrram[RTC_REGA] & RTC_RS) - 1);
|
2016-06-26 00:34:39 +02:00
|
|
|
rtctime += (int)(RTCCONST * c * (1 << TIMER_SHIFT));
|
|
|
|
|
}
|
|
|
|
|
void savenvr()
|
|
|
|
|
{
|
|
|
|
|
FILE *f;
|
|
|
|
|
switch (oldromset)
|
|
|
|
|
{
|
2017-05-06 04:02:03 +02:00
|
|
|
case ROM_PC1512: f = nvrfopen(L"pc1512.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PC1640: f = nvrfopen(L"pc1640.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PC200: f = nvrfopen(L"pc200.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PC2086: f = nvrfopen(L"pc2086.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PC3086: f = nvrfopen(L"pc3086.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMAT: f = nvrfopen(L"at.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS1_2011: f = nvrfopen(L"ibmps1_2011.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS1_2121: f = nvrfopen(L"ibmps1_2121.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS1_2121_ISA: f = nvrfopen(L"ibmps1_2121_isa.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS2_M30_286: f = nvrfopen(L"ibmps2_m30_286.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS2_M50: f = nvrfopen(L"ibmps2_m50.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS2_M55SX: f = nvrfopen(L"ibmps2_m55sx.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMPS2_M80: f = nvrfopen(L"ibmps2_m80.nvr", L"wb"); break;
|
|
|
|
|
case ROM_CMDPC30: f = nvrfopen(L"cmdpc30.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PORTABLEII: f = nvrfopen(L"portableii.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PORTABLEIII: f = nvrfopen(L"portableiii.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AMI286: f = nvrfopen(L"ami286.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AWARD286: f = nvrfopen(L"award286.nvr", L"wb"); break;
|
|
|
|
|
case ROM_SUPER286TR: f = nvrfopen(L"super286tr.nvr", L"wb"); break;
|
|
|
|
|
case ROM_SPC4200P: f = nvrfopen(L"spc4200p.nvr", L"wb"); break;
|
|
|
|
|
case ROM_IBMAT386: f = nvrfopen(L"at386.nvr", L"wb"); break;
|
|
|
|
|
case ROM_DESKPRO_386: f = nvrfopen(L"deskpro386.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PORTABLEIII386: f = nvrfopen(L"portableiii386.nvr", L"wb"); break;
|
|
|
|
|
case ROM_MEGAPC: f = nvrfopen(L"megapc.nvr", L"wb"); break;
|
|
|
|
|
case ROM_MEGAPCDX: f = nvrfopen(L"megapcdx.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AMI386SX: f = nvrfopen(L"ami386.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AMI486: f = nvrfopen(L"ami486.nvr", L"wb"); break;
|
|
|
|
|
case ROM_WIN486: f = nvrfopen(L"win486.nvr", L"wb"); break;
|
|
|
|
|
case ROM_SIS496: f = nvrfopen(L"sis496.nvr", L"wb"); break;
|
|
|
|
|
case ROM_430VX: f = nvrfopen(L"430vx.nvr", L"wb"); break;
|
|
|
|
|
case ROM_REVENGE: f = nvrfopen(L"revenge.nvr", L"wb"); break;
|
|
|
|
|
case ROM_ENDEAVOR: f = nvrfopen(L"endeavor.nvr", L"wb"); break;
|
|
|
|
|
case ROM_DTK386: f = nvrfopen(L"dtk386.nvr", L"wb"); break;
|
|
|
|
|
case ROM_MR386DX_OPTI495: f = nvrfopen(L"mr386dx_opti495.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AMI386DX_OPTI495: f = nvrfopen(L"ami386dx_opti495.nvr", L"wb"); break;
|
|
|
|
|
case ROM_DTK486: f = nvrfopen(L"dtk486.nvr", L"wb"); break;
|
|
|
|
|
case ROM_R418: f = nvrfopen(L"r418.nvr", L"wb"); break;
|
|
|
|
|
case ROM_586MC1: f = nvrfopen(L"586mc1.nvr", L"wb"); break;
|
|
|
|
|
case ROM_PLATO: f = nvrfopen(L"plato.nvr", L"wb"); break;
|
|
|
|
|
case ROM_MB500N: f = nvrfopen(L"mb500n.nvr", L"wb"); break;
|
|
|
|
|
case ROM_P54TP4XE: f = nvrfopen(L"p54tp4xe.nvr", L"wb"); break;
|
|
|
|
|
case ROM_AP53: f = nvrfopen(L"ap53.nvr", L"wb"); break;
|
|
|
|
|
case ROM_P55T2S: f = nvrfopen(L"p55t2s.nvr", L"wb"); break;
|
|
|
|
|
case ROM_ACERM3A: f = nvrfopen(L"acerm3a.nvr", L"wb"); break;
|
|
|
|
|
case ROM_ACERV35N: f = nvrfopen(L"acerv35n.nvr", L"wb"); break;
|
|
|
|
|
case ROM_P55VA: f = nvrfopen(L"p55va.nvr", L"wb"); break;
|
|
|
|
|
case ROM_P55T2P4: f = nvrfopen(L"p55t2p4.nvr", L"wb"); break;
|
|
|
|
|
case ROM_P55TVP4: f = nvrfopen(L"p55tvp4.nvr", L"wb"); break;
|
|
|
|
|
case ROM_440FX: f = nvrfopen(L"440fx.nvr", L"wb"); break;
|
|
|
|
|
case ROM_THOR: f = nvrfopen(L"thor.nvr", L"wb"); break;
|
|
|
|
|
case ROM_MRTHOR: f = nvrfopen(L"mrthor.nvr", L"wb"); break;
|
|
|
|
|
case ROM_ZAPPA: f = nvrfopen(L"zappa.nvr", L"wb"); break;
|
|
|
|
|
case ROM_S1668: f = nvrfopen(L"tpatx.nvr", L"wb"); break;
|
2017-05-29 01:18:32 +02:00
|
|
|
case ROM_IBMPS1_2133: f = nvrfopen(L"ibmps1_2133.nvr", L"wb"); break;
|
2016-06-26 00:34:39 +02:00
|
|
|
default: return;
|
|
|
|
|
}
|
|
|
|
|
fwrite(nvrram,128,1,f);
|
|
|
|
|
fclose(f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void nvr_init()
|
|
|
|
|
{
|
|
|
|
|
io_sethandler(0x0070, 0x0002, readnvr, NULL, NULL, writenvr, NULL, NULL, NULL);
|
|
|
|
|
timer_add(nvr_rtc, &rtctime, TIMER_ALWAYS_ENABLED, NULL);
|
|
|
|
|
timer_add(nvr_onesec, &nvr_onesec_time, TIMER_ALWAYS_ENABLED, NULL);
|
2016-07-19 02:44:32 +02:00
|
|
|
timer_add(nvr_update_end, &nvr_update_end_count, &nvr_update_end_count, NULL);
|
2016-08-15 23:40:02 +02:00
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|