#include "compiler.h"
#include "strres.h"
#include "scrnmng.h"
#include "cpucore.h"
#include "pccore.h"
#include "iocore.h"
#include "sound.h"
#include "fmboard.h"
#include "np2info.h"
static const OEMCHAR str_comma[] = OEMTEXT(", ");
static const OEMCHAR str_2halfMHz[] = OEMTEXT("2.5MHz");
#define str_5MHz (str_2halfMHz + 2)
static const OEMCHAR str_8MHz[] = OEMTEXT("8MHz");
static const OEMCHAR str_notexist[] = OEMTEXT("not exist");
static const OEMCHAR str_disable[] = OEMTEXT("disable");
static const OEMCHAR str_cpu[] = OEMTEXT("8086-2\00070116\00080286\00080386\00080486\0Pentium\0PentiumPro");
static const OEMCHAR str_winclr[] = OEMTEXT("256-colors\00065536-colors\0full color\0true color");
static const OEMCHAR str_winmode[] = OEMTEXT(" (window)\0 (fullscreen)");
static const OEMCHAR str_grcgchip[] = OEMTEXT("\0GRCG \0GRCG CG-Window \0EGC CG-Window ");
static const OEMCHAR str_vrammode[] = OEMTEXT("Digital\0Analog\000256colors");
static const OEMCHAR str_vrampage[] = OEMTEXT(" page-0\0 page-1\0 page-all");
static const OEMCHAR str_chpan[] = OEMTEXT("none\0Mono-R\0Mono-L\0Stereo");
static const OEMCHAR str_fmboard[] = OEMTEXT("none\0PC-9801-14\0PC-9801-26\0PC-9801-86\0PC-9801-26 + 86\0PC-9801-118\0PC-9801-86 + Chibi-oto\0Speak board\0Spark board\0AMD-98");
static const OEMCHAR str_clockfmt[] = OEMTEXT("%d.%1dMHz");
static const OEMCHAR str_memfmt[] = OEMTEXT("%3uKB");
static const OEMCHAR str_memfmt2[] = OEMTEXT("%3uKB + %uKB");
static const OEMCHAR str_memfmt3[] = OEMTEXT("%d.%1dMB");
static const OEMCHAR str_twidth[] = OEMTEXT("width-%u");
static const OEMCHAR str_dispclock[] = OEMTEXT("%u.%.2ukHz / %u.%uHz");
static const OEMCHAR str_pcm86a[] = OEMTEXT(" PCM: %dHz %dbit %s");
static const OEMCHAR str_pcm86b[] = OEMTEXT(" %d / %d / 32768");
static const OEMCHAR str_rhythm[] = OEMTEXT("BSCHTR");
// ---- common
static void info_ver(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
milstr_ncpy(str, np2version, maxlen);
(void)ex;
}
static void info_cpu(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT family;
#if defined(CPU_FAMILY)
family = min(CPU_FAMILY, 6);
#else
family = (CPU_TYPE & CPUTYPE_V30)?1:2;
#endif
milstr_ncpy(str, milstr_list(str_cpu, family), maxlen);
(void)ex;
}
static void info_clock(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT32 clk;
OEMCHAR clockstr[16];
clk = (pccore.realclock + 50000) / 100000;
OEMSPRINTF(clockstr, str_clockfmt, clk/10, clk % 10);
milstr_ncpy(str, clockstr, maxlen);
(void)ex;
}
static void info_base(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
milstr_ncpy(str,
(pccore.cpumode & CPUMODE_8MHZ)?str_8MHz:str_5MHz, maxlen);
(void)ex;
}
static void info_mem1(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT memsize;
OEMCHAR memstr[32];
memsize = np2cfg.memsw[2] & 7;
if (memsize < 6) {
memsize = (memsize + 1) * 128;
}
else {
memsize = 640;
}
if (pccore.extmem) {
OEMSPRINTF(memstr, str_memfmt2, memsize, pccore.extmem * 1024);
}
else {
OEMSPRINTF(memstr, str_memfmt, memsize);
}
milstr_ncpy(str, memstr, maxlen);
(void)ex;
}
static void info_mem2(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT memsize;
OEMCHAR memstr[16];
memsize = np2cfg.memsw[2] & 7;
if (memsize < 6) {
memsize = (memsize + 1) * 128;
}
else {
memsize = 640;
}
memsize += pccore.extmem * 1024;
OEMSPRINTF(memstr, str_memfmt, memsize);
milstr_ncpy(str, memstr, maxlen);
(void)ex;
}
static void info_mem3(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT memsize;
OEMCHAR memstr[16];
memsize = np2cfg.memsw[2] & 7;
if (memsize < 6) {
memsize = (memsize + 1) * 128;
}
else {
memsize = 640;
}
if (pccore.extmem > 1) {
OEMSPRINTF(memstr, str_memfmt3, pccore.extmem, memsize / 100);
}
else {
OEMSPRINTF(memstr, str_memfmt, memsize);
}
milstr_ncpy(str, memstr, maxlen);
(void)ex;
}
static void info_gdc(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
milstr_ncpy(str, milstr_list(str_grcgchip, grcg.chip & 3), maxlen);
milstr_ncat(str, str_2halfMHz + ((gdc.clock & 0x80)?2:0), maxlen);
(void)ex;
}
static void info_gdc2(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
OEMCHAR textstr[32];
OEMSPRINTF(textstr, str_dispclock,
gdc.hclock / 1000, (gdc.hclock / 10) % 100,
gdc.vclock / 10, gdc.vclock % 10);
milstr_ncpy(str, textstr, maxlen);
(void)ex;
}
static void info_text(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
const OEMCHAR *p;
OEMCHAR textstr[64];
if (!(gdcs.textdisp & GDCSCRN_ENABLE)) {
p = str_disable;
}
else {
OEMSPRINTF(textstr, str_twidth, ((gdc.mode1 & 0x4)?40:80));
p = textstr;
}
milstr_ncpy(str, p, maxlen);
(void)ex;
}
static void info_grph(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
const OEMCHAR *p;
UINT md;
UINT pg;
OEMCHAR work[32];
if (!(gdcs.grphdisp & GDCSCRN_ENABLE)) {
p = str_disable;
}
else {
md = (gdc.analog & (1 << GDCANALOG_16))?1:0;
pg = gdcs.access;
#if defined(SUPPORT_PC9821)
if (gdc.analog & (1 << (GDCANALOG_256))) {
md = 2;
if (gdc.analog & (1 << (GDCANALOG_256E))) {
pg = 2;
}
}
#endif
milstr_ncpy(work, milstr_list(str_vrammode, md), NELEMENTS(work));
milstr_ncat(work, milstr_list(str_vrampage, pg), NELEMENTS(work));
p = work;
}
milstr_ncpy(str, p, maxlen);
(void)ex;
}
static void info_sound(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT type;
type = 0;
switch(usesound) {
case 0x01:
type = 1;
break;
case 0x02:
type = 2;
break;
case 0x04:
type = 3;
break;
case 0x06:
type = 4;
break;
case 0x08:
type = 5;
break;
case 0x14:
type = 6;
break;
case 0x20:
type = 7;
break;
case 0x40:
type = 8;
break;
case 0x80:
type = 9;
break;
}
milstr_ncpy(str, milstr_list(str_fmboard, type), maxlen);
(void)ex;
}
static void info_extsnd(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
OEMCHAR buf[64];
info_sound(str, maxlen, ex);
if (usesound & 4) {
milstr_ncat(str, ex->cr, maxlen);
OEMSPRINTF(buf, str_pcm86a,
pcm86rate8[pcm86.fifo & 7] >> 3,
(16 - ((pcm86.dactrl >> 3) & 8)),
milstr_list(str_chpan, (pcm86.dactrl >> 4) & 3));
milstr_ncat(str, buf, maxlen);
milstr_ncat(str, ex->cr, maxlen);
OEMSPRINTF(buf, str_pcm86b, pcm86.virbuf, pcm86.fifosize);
milstr_ncat(str, buf, maxlen);
}
}
static void info_bios(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
str[0] = '\0';
if (pccore.rom & PCROM_BIOS) {
milstr_ncat(str, str_biosrom, maxlen);
}
if (soundrom.name[0]) {
if (str[0]) {
milstr_ncat(str, str_comma, maxlen);
}
milstr_ncat(str, soundrom.name, maxlen);
}
if (str[0] == '\0') {
milstr_ncat(str, str_notexist, maxlen);
}
(void)ex;
}
static void info_rhythm(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
OEMCHAR rhythmstr[8];
UINT exist;
UINT i;
exist = rhythm_getcaps();
milstr_ncpy(rhythmstr, str_rhythm, NELEMENTS(rhythmstr));
for (i=0; i<6; i++) {
if (!(exist & (1 << i))) {
rhythmstr[i] = '_';
}
}
milstr_ncpy(str, rhythmstr, maxlen);
(void)ex;
}
static void info_display(OEMCHAR *str, int maxlen, const NP2INFOEX *ex) {
UINT bpp;
bpp = scrnmng_getbpp();
milstr_ncpy(str, milstr_list(str_winclr, ((bpp >> 3) - 1) & 3), maxlen);
milstr_ncat(str, milstr_list(str_winmode, (scrnmng_isfullscreen())?1:0),
maxlen);
(void)ex;
}
// ---- make string
typedef struct {
OEMCHAR key[8];
void (*proc)(OEMCHAR *str, int maxlen, const NP2INFOEX *ex);
} INFOPROC;
static const INFOPROC infoproc[] = {
{OEMTEXT("VER"), info_ver},
{OEMTEXT("CPU"), info_cpu},
{OEMTEXT("CLOCK"), info_clock},
{OEMTEXT("BASE"), info_base},
{OEMTEXT("MEM1"), info_mem1},
{OEMTEXT("MEM2"), info_mem2},
{OEMTEXT("MEM3"), info_mem3},
{OEMTEXT("GDC"), info_gdc},
{OEMTEXT("GDC2"), info_gdc2},
{OEMTEXT("TEXT"), info_text},
{OEMTEXT("GRPH"), info_grph},
{OEMTEXT("SND"), info_sound},
{OEMTEXT("EXSND"), info_extsnd},
{OEMTEXT("BIOS"), info_bios},
{OEMTEXT("RHYTHM"), info_rhythm},
{OEMTEXT("DISP"), info_display}};
static BOOL defext(OEMCHAR *dst, const OEMCHAR *key, int maxlen,
const NP2INFOEX *ex) {
milstr_ncpy(dst, key, maxlen);
(void)ex;
return(TRUE);
}
void np2info(OEMCHAR *dst, const OEMCHAR *src, int maxlen,
const NP2INFOEX *ex) {
NP2INFOEX statex;
OEMCHAR c;
int leng;
OEMCHAR infwork[12];
const INFOPROC *inf;
const INFOPROC *infterm;
if ((dst == NULL) || (maxlen <= 0) || (src == NULL)) {
return;
}
if (ex == NULL) {
milstr_ncpy(statex.cr, str_oscr, NELEMENTS(statex.cr));
statex.ext = NULL;
}
else {
statex = *ex;
}
if (statex.ext == NULL) {
statex.ext = defext;
}
while(maxlen > 0) {
c = *src++;
if (c == '\0') {
break;
}
else if (c == '\n') {
milstr_ncpy(dst, statex.cr, maxlen);
}
else if (c != '%') {
*dst++ = c;
maxlen--;
continue;
}
else if (*src == '%') {
src++;
*dst++ = c;
maxlen--;
continue;
}
else {
leng = 0;
while(1) {
c = *src;
if (c == '\0') {
break;
}
src++;
if (c == '%') {
break;
}
if (leng < (int)(NELEMENTS(infwork) - 1)) {
infwork[leng++] = c;
}
}
infwork[leng] = '\0';
inf = infoproc;
infterm = infoproc + NELEMENTS(infoproc);
while(inf < infterm) {
if (!milstr_cmp(infwork, inf->key)) {
inf->proc(dst, maxlen, &statex);
break;
}
inf++;
}
if (inf >= infterm) {
if (!(*statex.ext)(dst, infwork, maxlen, &statex)) {
continue;
}
}
}
leng = OEMSTRLEN(dst);
dst += leng;
maxlen -= leng;
}
*dst = '\0';
}
RetroPC.NET-CVS <cvs@retropc.net>