mirror of
https://github.com/shagr4th/droid48
synced 2025-01-01 06:20:14 +01:00
1556 lines
30 KiB
C
1556 lines
30 KiB
C
/*
|
|
* This file is part of x48, an emulator of the HP-48sx Calculator.
|
|
* Copyright (C) 1994 Eddie C. Dost (ecd@dressler.de)
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
/* $Log: rpl.c,v $
|
|
* Revision 1.3 1995/01/11 18:20:01 ecd
|
|
* major update to support HP48 G/GX
|
|
*
|
|
* Revision 1.2 1994/12/07 20:20:50 ecd
|
|
* lots of more functions
|
|
*
|
|
* Revision 1.2 1994/12/07 20:20:50 ecd
|
|
* lots of more functions
|
|
*
|
|
* Revision 1.1 1994/12/07 10:15:47 ecd
|
|
* Initial revision
|
|
*
|
|
*
|
|
* $Id: rpl.c,v 1.3 1995/01/11 18:20:01 ecd Exp ecd $
|
|
*/
|
|
|
|
#include "global.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
|
|
#include "hp48.h"
|
|
#include "hp48_emu.h"
|
|
#include "rpl.h"
|
|
#include "debugger.h"
|
|
#include "append.h"
|
|
#include "disasm.h"
|
|
#include "romio.h"
|
|
#define DEFINE_TRANS_TABLE 1
|
|
#include "hp48char.h"
|
|
#undef DEFINE_TRANS_TABLE
|
|
|
|
struct objfunc {
|
|
char *name;
|
|
short length;
|
|
word_20 prolog;
|
|
char *(*func) __ProtoType__((word_20 *addr, char *string));
|
|
} objects[] = {
|
|
{ "System Binary", 0, DOBINT, dec_bin_int },
|
|
{ "Real", 0, DOREAL, dec_real },
|
|
{ "Long Real", 0, DOEREL, dec_long_real },
|
|
{ "Complex", 0, DOCMP, dec_complex },
|
|
{ "Long Complex", 0, DOECMP, dec_long_complex },
|
|
{ "Character", 0, DOCHAR, dec_char },
|
|
{ "Array", 0, DOARRY, dec_array },
|
|
{ "Linked Array", 0, DOLNKARRY, dec_lnk_array },
|
|
{ "String", 2, DOCSTR, dec_string },
|
|
{ "Hex String", 1, DOHSTR, dec_hex_string },
|
|
{ "List", 0, DOLIST, dec_list },
|
|
{ "Directory", 0, DORRP, skip_ob },
|
|
{ "Symbolic", 0, DOSYMB, dec_symb },
|
|
{ "Unit", 0, DOEXT, dec_unit },
|
|
{ "Tagged", 0, DOTAG, skip_ob },
|
|
{ "Graphic", 0, DOGROB, skip_ob },
|
|
{ "Library", 0, DOLIB, dec_library },
|
|
{ "Backup", 0, DOBAK, skip_ob },
|
|
{ "Library Data", 0, DOEXT0, dec_library_data },
|
|
{ "ACPTR", 0, DOACPTR, dec_acptr },
|
|
{ "External 2", 0, DOEXT2, skip_ob },
|
|
{ "External 3", 0, DOEXT3, skip_ob },
|
|
{ "External 4", 0, DOEXT4, skip_ob },
|
|
{ "Program", 0, DOCOL, dec_prog },
|
|
{ "Code", 1, DOCODE, dec_code },
|
|
{ "Global Ident", 0, DOIDNT, dec_global_ident },
|
|
{ "Local Ident", 0, DOLAM, dec_local_ident },
|
|
{ "XLib Name", 0, DOROMP, dec_xlib_name },
|
|
{ "*", 0, UM_MUL, dec_unit_op },
|
|
{ "/", 0, UM_DIV, dec_unit_op },
|
|
{ "^", 0, UM_POW, dec_unit_op },
|
|
{ " ", 0, UM_PRE, dec_unit_op },
|
|
{ "_", 0, UM_END, dec_unit_op },
|
|
{ 0, 0, 0 }
|
|
};
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
skip_ob(word_20 *addr, char *string)
|
|
#else
|
|
skip_ob(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 size, type;
|
|
char *p = string;
|
|
struct objfunc *op;
|
|
|
|
type = read_nibbles(*addr - 5, 5);
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == type)
|
|
break;
|
|
}
|
|
|
|
if (op->prolog)
|
|
{
|
|
sprintf(p, "%s", op->name);
|
|
p += strlen(p);
|
|
}
|
|
|
|
size = read_nibbles(*addr, 5);
|
|
*addr += size;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
long
|
|
#ifdef __FunctionProto__
|
|
hxs2real(long hxs)
|
|
#else
|
|
hxs2real(hxs)
|
|
long hxs;
|
|
#endif
|
|
{
|
|
int n = 0, c = 1;
|
|
|
|
while (hxs)
|
|
{
|
|
n += (hxs & 0xf) * c;
|
|
c *= 10;
|
|
hxs >>= 4;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_bin_int(word_20 *addr, char *string)
|
|
#else
|
|
dec_bin_int(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
char *p = string;
|
|
word_20 n = 0;
|
|
|
|
n = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
sprintf(p, "<%lXh>", (long)n);
|
|
p += strlen(p);
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
real_number(word_20 *addr, char *string, int ml, int xl)
|
|
#else
|
|
real_number(addr, string, ml, xl)
|
|
word_20 *addr;
|
|
char *string;
|
|
int ml;
|
|
int xl;
|
|
#endif
|
|
{
|
|
hp_real r;
|
|
long re, xs;
|
|
int i;
|
|
char fmt[20];
|
|
char m[16];
|
|
char *p = string;
|
|
|
|
/*
|
|
* Read the number
|
|
*/
|
|
r.x = read_nibbles(*addr, xl);
|
|
*addr += xl;
|
|
r.ml = read_nibbles(*addr, ml - 8);
|
|
*addr += ml - 8;
|
|
r.mh = read_nibbles(*addr, 8);
|
|
*addr += 8;
|
|
r.m = read_nibbles(*addr, 1);
|
|
(*addr)++;
|
|
r.s = read_nibbles(*addr, 1);
|
|
(*addr)++;
|
|
|
|
/*
|
|
* Figure out the exponent
|
|
*/
|
|
xs = 5;
|
|
while (--xl) xs *= 10;
|
|
re = hxs2real(r.x);
|
|
if (re >= xs)
|
|
re = re - 2 * xs;
|
|
|
|
|
|
if ((re >= 0) && (re < ml + 1))
|
|
{
|
|
if (r.s >= 5)
|
|
*p++ = '-';
|
|
|
|
sprintf(fmt, "%%.1X%%.8lX%%.%dlX", ml - 8);
|
|
sprintf(m, fmt, r.m, r.mh, r.ml);
|
|
|
|
for (i = 0; i <= re; i++)
|
|
*p++ = m[i];
|
|
*p++ = '.';
|
|
for ( ; i < ml + 1; i++)
|
|
*p++ = m[i];
|
|
p--;
|
|
while(*p == '0') p--;
|
|
if (*p == '.') p--;
|
|
*++p = '\0';
|
|
|
|
return p;
|
|
}
|
|
|
|
if ((re < 0) && (re >= -ml - 1))
|
|
{
|
|
sprintf(fmt, "%%.1X%%.8lX%%.%dlX", ml - 8);
|
|
sprintf(m, fmt, r.m, r.mh, r.ml);
|
|
|
|
for (i = ml; m[i] == '0'; i--) ;
|
|
|
|
if (-re <= ml - i + 1)
|
|
{
|
|
if (r.s >= 5)
|
|
*p++ = '-';
|
|
|
|
*p++ = '.';
|
|
|
|
for (i = 1; i < -re; i++)
|
|
*p++ = '0';
|
|
|
|
for (i = 0; i < ml + 1; i++)
|
|
*p++ = m[i];
|
|
p--;
|
|
while(*p == '0') p--;
|
|
*++p = '\0';
|
|
|
|
return p;
|
|
}
|
|
}
|
|
|
|
sprintf(fmt, "%%s%%X.%%.8lX%%.%dlX", ml - 8);
|
|
sprintf(p, fmt, (r.s >= 5) ? "-" : "", r.m, r.mh, r.ml);
|
|
|
|
p += strlen(p) - 1;
|
|
|
|
while(*p == '0') p--;
|
|
*++p = '\0';
|
|
|
|
if (re)
|
|
{
|
|
sprintf(p, "E%ld", re);
|
|
p += strlen(p);
|
|
*p = '\0';
|
|
}
|
|
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_real(word_20 *addr, char *string)
|
|
#else
|
|
dec_real(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
return real_number(addr, string, 11, 3);
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_long_real(word_20 *addr, char *string)
|
|
#else
|
|
dec_long_real(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
return real_number(addr, string, 14, 5);
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_complex(word_20 *addr, char *string)
|
|
#else
|
|
dec_complex(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
char *p = string;
|
|
|
|
*p++ = '(';
|
|
p = real_number(addr, p, 11, 3);
|
|
*p++ = ',';
|
|
p = real_number(addr, p, 11, 3);
|
|
*p++ = ')';
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_long_complex(word_20 *addr, char *string)
|
|
#else
|
|
dec_long_complex(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
char *p = string;
|
|
|
|
*p++ = '(';
|
|
p = real_number(addr, p, 14, 5);
|
|
*p++ = ',';
|
|
p = real_number(addr, p, 14, 5);
|
|
*p++ = ')';
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_string(word_20 *addr, char *string)
|
|
#else
|
|
dec_string(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 len;
|
|
unsigned char c;
|
|
char *p = string;
|
|
int i, n;
|
|
|
|
len = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
len -= 5;
|
|
len /= 2;
|
|
|
|
n = len;
|
|
if (len > 1000)
|
|
n = 1000;
|
|
|
|
*p++ = '\"';
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
c = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
}
|
|
|
|
if (n != len)
|
|
{
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
}
|
|
|
|
*p++ = '\"';
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_hex_string(word_20 *addr, char *string)
|
|
#else
|
|
dec_hex_string(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int len, lead, i, n;
|
|
static char hex[] = "0123456789ABCDEF";
|
|
char *p = string;
|
|
|
|
len = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
len -= 5;
|
|
|
|
if (len <= 16)
|
|
{
|
|
*p++ = '#';
|
|
*p++ = ' ';
|
|
lead = 1;
|
|
for (i = len - 1; i >= 0; i--)
|
|
{
|
|
*p = hex[read_nibble(*addr + i)];
|
|
if (lead)
|
|
if ((i != 0) && (*p == '0'))
|
|
p--;
|
|
else
|
|
lead = 0;
|
|
p++;
|
|
}
|
|
|
|
*p++ = 'h';
|
|
}
|
|
else
|
|
{
|
|
*p++ = 'C';
|
|
*p++ = '#';
|
|
*p++ = ' ';
|
|
|
|
sprintf(p, "%d", len);
|
|
p += strlen(p);
|
|
|
|
*p++ = ' ';
|
|
|
|
n = len;
|
|
if (len > 1000)
|
|
n = 1000;
|
|
|
|
for (i = 0; i < n; i++)
|
|
*p++ = hex[read_nibble(*addr + i)];
|
|
|
|
if (n != len)
|
|
{
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
}
|
|
}
|
|
|
|
*addr += len;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_list(word_20 *addr, char *string)
|
|
#else
|
|
dec_list(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 semi;
|
|
char *p = string;
|
|
|
|
*p++ = '{';
|
|
*p++ = ' ';
|
|
semi = read_nibbles(*addr, 5);
|
|
while (semi != SEMI)
|
|
{
|
|
p = dec_rpl_obj(addr, p);
|
|
semi = read_nibbles(*addr, 5);
|
|
if (semi != SEMI)
|
|
{
|
|
*p++ = ' ';
|
|
*p = '\0';
|
|
}
|
|
}
|
|
*p++ = ' ';
|
|
*p++ = '}';
|
|
*p = '\0';
|
|
|
|
*addr += 5;
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_symb(word_20 *addr, char *string)
|
|
#else
|
|
dec_symb(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 semi;
|
|
char *p = string;
|
|
|
|
semi = read_nibbles(*addr, 5);
|
|
*p++ = '\'';
|
|
while (semi != SEMI)
|
|
{
|
|
p = dec_rpl_obj(addr, p);
|
|
semi = read_nibbles(*addr, 5);
|
|
if (semi != SEMI)
|
|
{
|
|
*p++ = ' ';
|
|
*p = '\0';
|
|
}
|
|
}
|
|
*addr += 5;
|
|
|
|
*p++ = '\'';
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_unit(word_20 *addr, char *string)
|
|
#else
|
|
dec_unit(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 semi;
|
|
char *p = string;
|
|
|
|
semi = read_nibbles(*addr, 5);
|
|
while (semi != SEMI)
|
|
{
|
|
p = dec_rpl_obj(addr, p);
|
|
semi = read_nibbles(*addr, 5);
|
|
if (semi != SEMI)
|
|
{
|
|
*p++ = ' ';
|
|
*p = '\0';
|
|
}
|
|
}
|
|
*addr += 5;
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_unit_op(word_20 *addr, char *string)
|
|
#else
|
|
dec_unit_op(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 op;
|
|
char *p = string;
|
|
|
|
op = read_nibbles(*addr - 5, 5);
|
|
switch (op)
|
|
{
|
|
case UM_MUL:
|
|
*p++ = '*';
|
|
break;
|
|
case UM_DIV:
|
|
*p++ = '/';
|
|
break;
|
|
case UM_POW:
|
|
*p++ = '^';
|
|
break;
|
|
case UM_END:
|
|
*p++ = '_';
|
|
break;
|
|
case UM_PRE:
|
|
p--;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_library(word_20 *addr, char *string)
|
|
#else
|
|
dec_library(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 libsize, libidsize;
|
|
/*
|
|
word_20 hashoff, mesgoff, linkoff, cfgoff;
|
|
word_20 mesgloc, cfgloc;
|
|
*/
|
|
int i, libnum;
|
|
unsigned char c;
|
|
char *p = string;
|
|
|
|
libsize = read_nibbles(*addr, 5);
|
|
libidsize = read_nibbles(*addr + 5, 2);
|
|
libnum = read_nibbles(*addr + 2 * libidsize + 9, 3);
|
|
|
|
sprintf(p, "Library %d: ", libnum);
|
|
p += strlen(p);
|
|
|
|
for (i = 0; i < libidsize; i++)
|
|
{
|
|
c = read_nibbles(*addr + 2 * i + 7, 2);
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
}
|
|
|
|
*addr += libsize;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_library_data(word_20 *addr, char *string)
|
|
#else
|
|
dec_library_data(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 size;
|
|
char *p = string;
|
|
|
|
size = read_nibbles(*addr, 5);
|
|
|
|
sprintf(p, "Library Data");
|
|
p += strlen(p);
|
|
|
|
*addr += size;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_acptr(word_20 *addr, char *string)
|
|
#else
|
|
dec_acptr(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 size;
|
|
char *p = string;
|
|
int i;
|
|
static char hex[] = "0123456789ABCDEF";
|
|
|
|
if (opt_gx)
|
|
{
|
|
size = 10;
|
|
sprintf(p, "ACPTR ");
|
|
p += strlen(p);
|
|
for (i = 0; i < 5; i++)
|
|
*p++ = hex[read_nibble(*addr + i)];
|
|
*p++ = ' ';
|
|
for (i = 5; i < 10; i++)
|
|
*p++ = hex[read_nibble(*addr + i)];
|
|
}
|
|
else
|
|
{
|
|
size = read_nibbles(*addr, 5);
|
|
sprintf(p, "Ext 1");
|
|
p += strlen(p);
|
|
}
|
|
|
|
*addr += size;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_prog(word_20 *addr, char *string)
|
|
#else
|
|
dec_prog(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 semi;
|
|
char *p = string;
|
|
|
|
semi = read_nibbles(*addr, 5);
|
|
while (semi != SEMI)
|
|
{
|
|
p = dec_rpl_obj(addr, p);
|
|
semi = read_nibbles(*addr, 5);
|
|
if (semi != SEMI)
|
|
{
|
|
*p++ = ' ';
|
|
*p = '\0';
|
|
}
|
|
}
|
|
*addr += 5;
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_code(word_20 *addr, char *string)
|
|
#else
|
|
dec_code(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
char *p = string;
|
|
word_20 n, len;
|
|
|
|
len = read_nibbles(*addr, 5);
|
|
sprintf(p, "Code");
|
|
p += strlen(p);
|
|
|
|
n = 0;
|
|
while (n < len)
|
|
{
|
|
/*
|
|
*addr = disassemble(*addr, p);
|
|
*/
|
|
n += len;
|
|
}
|
|
|
|
*addr += len;
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_local_ident(word_20 *addr, char *string)
|
|
#else
|
|
dec_local_ident(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int len, i, n;
|
|
char *p = string;
|
|
unsigned char c;
|
|
|
|
len = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
|
|
n = len;
|
|
if (len > 1000)
|
|
n = 1000;
|
|
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
c = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
}
|
|
|
|
if (n != len)
|
|
{
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
}
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_global_ident(word_20 *addr, char *string)
|
|
#else
|
|
dec_global_ident(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int len, i, n;
|
|
char *p = string;
|
|
unsigned char c;
|
|
|
|
len = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
|
|
n = len;
|
|
if (len > 1000)
|
|
n = 1000;
|
|
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
c = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
}
|
|
|
|
if (n != len)
|
|
{
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
*p++ = '.';
|
|
}
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
xlib_name(int lib, int command, char *string)
|
|
#else
|
|
xlib_name(lib, command, string)
|
|
int lib;
|
|
int command;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int n, len;
|
|
int i, lib_n = 0;
|
|
unsigned char c;
|
|
word_20 romptab, acptr;
|
|
word_20 offset, hash_end;
|
|
word_20 lib_addr, name_addr;
|
|
word_20 type, ram_base, ram_mask;
|
|
short present = 0;
|
|
char *p = string;
|
|
|
|
/*
|
|
* Configure RAM to address 0x70000
|
|
*/
|
|
ram_base = saturn.mem_cntl[1].config[0];
|
|
ram_mask = saturn.mem_cntl[1].config[1];
|
|
if (opt_gx)
|
|
{
|
|
saturn.mem_cntl[1].config[0] = 0x80000;
|
|
saturn.mem_cntl[1].config[1] = 0xc0000;
|
|
romptab = ROMPTAB_GX;
|
|
}
|
|
else
|
|
{
|
|
saturn.mem_cntl[1].config[0] = 0x70000;
|
|
saturn.mem_cntl[1].config[1] = 0xf0000;
|
|
romptab = ROMPTAB_SX;
|
|
}
|
|
|
|
/*
|
|
* look up number of installed libs in romptab
|
|
*/
|
|
n = read_nibbles(romptab, 3);
|
|
romptab += 3;
|
|
|
|
if (n > 0)
|
|
{
|
|
/*
|
|
* look up lib number in romptab
|
|
*/
|
|
while (n--)
|
|
{
|
|
lib_n = read_nibbles(romptab, 3);
|
|
romptab += 3;
|
|
if (lib_n == lib)
|
|
break;
|
|
romptab += 5;
|
|
if (opt_gx)
|
|
romptab += 8;
|
|
}
|
|
if (lib_n == lib)
|
|
{
|
|
/*
|
|
* look at hash table pointer
|
|
*/
|
|
lib_addr = read_nibbles(romptab, 5);
|
|
if (opt_gx)
|
|
{
|
|
romptab += 5;
|
|
acptr = read_nibbles(romptab, 5);
|
|
if (acptr != 0x00000)
|
|
{
|
|
saturn.mem_cntl[1].config[0] = ram_base;
|
|
saturn.mem_cntl[1].config[1] = ram_mask;
|
|
sprintf(p, "XLIB %d %d", lib, command);
|
|
p += strlen(p);
|
|
return p;
|
|
}
|
|
}
|
|
lib_addr += 3;
|
|
offset = read_nibbles(lib_addr, 5);
|
|
if (offset > 0)
|
|
{
|
|
/*
|
|
* look at the hash table
|
|
*/
|
|
lib_addr += offset;
|
|
|
|
/*
|
|
* check if library is in ROM
|
|
*/
|
|
if (!opt_gx)
|
|
if (lib_addr < 0x70000)
|
|
saturn.mem_cntl[1].config[0] = 0xf0000;
|
|
|
|
/*
|
|
* check pointer type
|
|
*/
|
|
type = read_nibbles(lib_addr, 5);
|
|
if (type == DOBINT)
|
|
{
|
|
/*
|
|
* follow pointer to real address
|
|
*/
|
|
lib_addr += 5;
|
|
lib_addr = read_nibbles(lib_addr, 5);
|
|
}
|
|
else if (type == DOACPTR)
|
|
{
|
|
/*
|
|
* follow pointer to real address
|
|
*/
|
|
lib_addr += 5;
|
|
acptr = lib_addr + 5;
|
|
lib_addr = read_nibbles(lib_addr, 5);
|
|
acptr = read_nibbles(acptr, 5);
|
|
if (acptr != 0x00000)
|
|
{
|
|
saturn.mem_cntl[1].config[0] = ram_base;
|
|
saturn.mem_cntl[1].config[1] = ram_mask;
|
|
sprintf(p, "XLIB %d %d", lib, command);
|
|
p += strlen(p);
|
|
return p;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* get length of hash table
|
|
*/
|
|
lib_addr += 5;
|
|
hash_end = read_nibbles(lib_addr, 5);
|
|
hash_end += lib_addr;
|
|
|
|
/*
|
|
* go into real name table
|
|
*/
|
|
lib_addr += 85;
|
|
offset = read_nibbles(lib_addr, 5);
|
|
lib_addr += offset;
|
|
|
|
/*
|
|
* look at library name number 'command'
|
|
*/
|
|
offset = 5 * command;
|
|
lib_addr += offset;
|
|
if (lib_addr < hash_end)
|
|
{
|
|
offset = read_nibbles(lib_addr, 5);
|
|
if (offset > 0)
|
|
{
|
|
name_addr = lib_addr - offset;
|
|
len = read_nibbles(name_addr, 2);
|
|
name_addr += 2;
|
|
present = 1;
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
c = read_nibbles(name_addr, 2);
|
|
name_addr += 2;
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
}
|
|
*p = '\0';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Reconfigure RAM
|
|
*/
|
|
saturn.mem_cntl[1].config[0] = ram_base;
|
|
saturn.mem_cntl[1].config[1] = ram_mask;
|
|
|
|
if (!present)
|
|
{
|
|
sprintf(p, "XLIB %d %d", lib, command);
|
|
p += strlen(p);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_xlib_name(word_20 *addr, char *string)
|
|
#else
|
|
dec_xlib_name(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int lib, command;
|
|
|
|
lib = read_nibbles(*addr, 3);
|
|
*addr += 3;
|
|
command = read_nibbles(*addr, 3);
|
|
*addr += 3;
|
|
|
|
return xlib_name(lib, command, string);
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
any_array(word_20 *addr, char *string, short lnk_flag)
|
|
#else
|
|
any_array(addr, string, lnk_flag)
|
|
word_20 *addr;
|
|
char *string;
|
|
short lnk_flag;
|
|
#endif
|
|
{
|
|
word_20 len, type, dim;
|
|
word_20 *dim_lens, *dims;
|
|
word_20 array_addr, elem_addr;
|
|
long elems;
|
|
int d, i;
|
|
char *p = string;
|
|
struct objfunc *op;
|
|
|
|
array_addr = *addr;
|
|
len = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
type = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
dim = read_nibbles(*addr, 5);
|
|
*addr += 5;
|
|
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == type)
|
|
break;
|
|
}
|
|
|
|
dim_lens = (word_20 *)malloc(dim * sizeof(word_20));
|
|
dims = (word_20 *)malloc(dim * sizeof(word_20));
|
|
elems = 1;
|
|
for (i = 0; i < dim; i++)
|
|
{
|
|
dim_lens[i] = read_nibbles(*addr, 5);
|
|
dims[i] = dim_lens[i];
|
|
elems *= dim_lens[i];
|
|
*addr += 5;
|
|
}
|
|
|
|
if (op->prolog == 0)
|
|
{
|
|
sprintf(p, "of Type %.5lX, Dim %ld, Size ", type, (long)dim);
|
|
p += strlen(p);
|
|
for (i = 0; i < dim; i++)
|
|
{
|
|
sprintf(p, "%ld", (long)dim_lens[i]);
|
|
p += strlen(p);
|
|
if (i < dim - 1)
|
|
{
|
|
sprintf(p, " x ");
|
|
p += strlen(p);
|
|
}
|
|
}
|
|
*p = '\0';
|
|
*addr = array_addr + len;
|
|
free(dim_lens);
|
|
free(dims);
|
|
return p;
|
|
}
|
|
|
|
d = -1;
|
|
while (elems--)
|
|
{
|
|
if (d < dim - 1)
|
|
{
|
|
for ( ; d < dim - 1; d++)
|
|
{
|
|
*p++ = '[';
|
|
}
|
|
d = dim - 1;
|
|
}
|
|
if (lnk_flag)
|
|
{
|
|
elem_addr = read_nibbles(*addr, 5);
|
|
elem_addr += *addr;
|
|
*addr += 5;
|
|
p = (*op->func)(&elem_addr, p);
|
|
}
|
|
else
|
|
p = (*op->func)(addr, p);
|
|
*p = '\0';
|
|
dims[d]--;
|
|
if (dims[d])
|
|
*p++ = ' ';
|
|
while (dims[d] == 0)
|
|
{
|
|
dims[d] = dim_lens[d];
|
|
d--;
|
|
dims[d]--;
|
|
*p++ = ']';
|
|
}
|
|
}
|
|
|
|
free(dim_lens);
|
|
free(dims);
|
|
*addr = array_addr + len;
|
|
|
|
*p = '\0';
|
|
return p;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_array(word_20 *addr, char *string)
|
|
#else
|
|
dec_array(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
return any_array(addr, string, 0);
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_lnk_array(word_20 *addr, char *string)
|
|
#else
|
|
dec_lnk_array(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
return any_array(addr, string, 1);
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_char(word_20 *addr, char *string)
|
|
#else
|
|
dec_char(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
char *p = string;
|
|
unsigned char c;
|
|
|
|
c = read_nibbles(*addr, 2);
|
|
*addr += 2;
|
|
|
|
*p++ = '\'';
|
|
if (hp48_trans_tbl[c].trans)
|
|
{
|
|
sprintf(p, hp48_trans_tbl[c].trans);
|
|
p += strlen(p);
|
|
}
|
|
else
|
|
*p++ = c;
|
|
*p++ = '\'';
|
|
|
|
*p = 0;
|
|
return p;
|
|
}
|
|
|
|
short
|
|
#ifdef __FunctionProto__
|
|
check_xlib(word_20 addr, char *string)
|
|
#else
|
|
check_xlib(addr, string)
|
|
word_20 addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
int n, lib, command;
|
|
word_20 romptab;
|
|
word_20 offset, link_end;
|
|
word_20 acptr;
|
|
word_20 lib_addr;
|
|
word_20 type, ram_base, ram_mask;
|
|
char *p = string;
|
|
|
|
/*
|
|
* Configure RAM to address 0x70000
|
|
*/
|
|
ram_base = saturn.mem_cntl[1].config[0];
|
|
ram_mask = saturn.mem_cntl[1].config[1];
|
|
if (opt_gx)
|
|
{
|
|
saturn.mem_cntl[1].config[0] = 0x80000;
|
|
saturn.mem_cntl[1].config[1] = 0xc0000;
|
|
romptab = ROMPTAB_GX;
|
|
}
|
|
else
|
|
{
|
|
saturn.mem_cntl[1].config[0] = 0x70000;
|
|
saturn.mem_cntl[1].config[1] = 0xf0000;
|
|
romptab = ROMPTAB_SX;
|
|
}
|
|
|
|
/*
|
|
* look up number of installed libs in romptab
|
|
*/
|
|
n = read_nibbles(romptab, 3);
|
|
romptab += 3;
|
|
|
|
/*
|
|
LOGE( "Number of Libraries = %d\n", n);
|
|
fflush(stderr);
|
|
*/
|
|
|
|
if (n > 0)
|
|
{
|
|
/*
|
|
* look up lib number in romptab
|
|
*/
|
|
while (n--)
|
|
{
|
|
lib = read_nibbles(romptab, 3);
|
|
romptab += 3;
|
|
/*
|
|
LOGE( "Library num = %d\n", lib);
|
|
fflush(stderr);
|
|
*/
|
|
/*
|
|
* look at link table pointer
|
|
*/
|
|
lib_addr = read_nibbles(romptab, 5);
|
|
/*
|
|
LOGE( "Library addr = %.5lx\n", lib_addr);
|
|
fflush(stderr);
|
|
*/
|
|
romptab += 5;
|
|
|
|
if (opt_gx)
|
|
{
|
|
acptr = read_nibbles(romptab, 5);
|
|
romptab += 8;
|
|
if (acptr != 0x00000)
|
|
continue;
|
|
}
|
|
|
|
lib_addr += 13;
|
|
offset = read_nibbles(lib_addr, 5);
|
|
if (offset > 0)
|
|
{
|
|
/*
|
|
* look at the link table
|
|
*/
|
|
lib_addr += offset;
|
|
/*
|
|
LOGE( "Link table addr = %.5lx\n", lib_addr);
|
|
fflush(stderr);
|
|
*/
|
|
/*
|
|
* check if library is in ROM
|
|
*/
|
|
if (!opt_gx)
|
|
if (lib_addr < 0x70000)
|
|
saturn.mem_cntl[1].config[0] = 0xf0000;
|
|
|
|
/*
|
|
* check pointer type
|
|
*/
|
|
type = read_nibbles(lib_addr, 5);
|
|
if (type == DOBINT)
|
|
{
|
|
/*
|
|
* follow pointer to real address
|
|
*/
|
|
lib_addr += 5;
|
|
lib_addr = read_nibbles(lib_addr, 5);
|
|
}
|
|
/*
|
|
LOGE( "Link table addr (2) = %.5lx\n", lib_addr);
|
|
fflush(stderr);
|
|
*/
|
|
/*
|
|
* get length of link table
|
|
*/
|
|
lib_addr += 5;
|
|
link_end = read_nibbles(lib_addr, 5);
|
|
link_end += lib_addr;
|
|
/*
|
|
LOGE( "Link table end = %.5lx\n", link_end);
|
|
fflush(stderr);
|
|
*/
|
|
/*
|
|
* look at library commands
|
|
*/
|
|
lib_addr += 5;
|
|
command = 0;
|
|
while (lib_addr < link_end)
|
|
{
|
|
offset = read_nibbles(lib_addr, 5);
|
|
if (offset > 0)
|
|
{
|
|
if (addr == ((lib_addr + offset) & 0xfffff))
|
|
{
|
|
p = xlib_name(lib, command, p);
|
|
saturn.mem_cntl[1].config[0] = ram_base;
|
|
saturn.mem_cntl[1].config[1] = ram_mask;
|
|
return 1;
|
|
}
|
|
}
|
|
lib_addr += 5;
|
|
command++;
|
|
}
|
|
if (opt_gx)
|
|
saturn.mem_cntl[1].config[0] = 0x80000;
|
|
else
|
|
saturn.mem_cntl[1].config[0] = 0x70000;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Reconfigure RAM
|
|
*/
|
|
saturn.mem_cntl[1].config[0] = ram_base;
|
|
saturn.mem_cntl[1].config[1] = ram_mask;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
dec_rpl_obj(word_20 *addr, char *string)
|
|
#else
|
|
dec_rpl_obj(addr, string)
|
|
word_20 *addr;
|
|
char *string;
|
|
#endif
|
|
{
|
|
word_20 prolog = 0;
|
|
word_20 prolog_2;
|
|
char *p = string;
|
|
char tmp_str[80];
|
|
struct objfunc *op;
|
|
|
|
prolog = read_nibbles(*addr, 5);
|
|
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == prolog)
|
|
break;
|
|
}
|
|
|
|
if (op->prolog == 0)
|
|
{
|
|
if (check_xlib(prolog, tmp_str))
|
|
{
|
|
p = append_str(p, tmp_str);
|
|
}
|
|
else
|
|
{
|
|
prolog_2 = read_nibbles(prolog, 5);
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == prolog_2)
|
|
break;
|
|
}
|
|
if (op->prolog)
|
|
p = dec_rpl_obj(&prolog, p);
|
|
else
|
|
p = append_str(p, "External");
|
|
}
|
|
*addr += 5;
|
|
return p;
|
|
}
|
|
|
|
*addr += 5;
|
|
p = (*op->func)(addr, p);
|
|
|
|
return p;
|
|
}
|
|
|
|
void
|
|
#ifdef __FunctionProto__
|
|
decode_rpl_obj_2(word_20 addr, char *typ, char *dat)
|
|
#else
|
|
decode_rpl_obj_2(addr, typ, dat)
|
|
word_20 addr;
|
|
char *typ;
|
|
char *dat;
|
|
#endif
|
|
{
|
|
word_20 prolog = 0;
|
|
int len;
|
|
char tmp_str[80];
|
|
struct objfunc *op;
|
|
|
|
typ[0] = '\0';
|
|
dat[0] = '\0';
|
|
|
|
prolog = read_nibbles(addr, 5);
|
|
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == prolog)
|
|
break;
|
|
}
|
|
|
|
if (op->prolog == 0)
|
|
{
|
|
if (addr == SEMI)
|
|
{
|
|
append_str(typ, "Primitive Code");
|
|
append_str(dat, "SEMI");
|
|
}
|
|
else if (addr + 5 == prolog)
|
|
{
|
|
append_str(typ, "Primitive Code");
|
|
sprintf(dat, "at %.5lX", prolog);
|
|
}
|
|
else
|
|
{
|
|
append_str(typ, "PTR");
|
|
sprintf(dat, "%.5lX", prolog);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (op->prolog == DOCOL)
|
|
{
|
|
if (check_xlib(addr, tmp_str))
|
|
{
|
|
append_str(typ, "XLib Call");
|
|
append_str(dat, tmp_str);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (op->length)
|
|
{
|
|
len = (read_nibbles(addr + 5, 5) - 5) / op->length;
|
|
sprintf(typ, "%s %d", op->name, len);
|
|
}
|
|
else
|
|
{
|
|
append_str(typ, op->name);
|
|
}
|
|
|
|
addr += 5;
|
|
(*op->func)(&addr, dat);
|
|
|
|
return;
|
|
}
|
|
|
|
char *
|
|
#ifdef __FunctionProto__
|
|
decode_rpl_obj(word_20 addr, char *buf)
|
|
#else
|
|
decode_rpl_obj(addr, buf)
|
|
word_20 addr;
|
|
char *buf;
|
|
#endif
|
|
{
|
|
word_20 prolog = 0;
|
|
int len;
|
|
char *p = buf;
|
|
char tmp_str[80];
|
|
struct objfunc *op;
|
|
|
|
prolog = read_nibbles(addr, 5);
|
|
|
|
for (op = objects; op->prolog != 0; op++)
|
|
{
|
|
if (op->prolog == prolog)
|
|
break;
|
|
}
|
|
|
|
if (op->prolog == 0)
|
|
{
|
|
if (addr == SEMI)
|
|
{
|
|
p = append_str(buf, "Primitive Code");
|
|
p = append_tab_16(buf);
|
|
p = append_str(p, "SEMI");
|
|
}
|
|
else if (addr + 5 == prolog)
|
|
{
|
|
p = append_str(buf, "Primitive Code");
|
|
p = append_tab_16(buf);
|
|
sprintf(p, "at %.5lX", prolog);
|
|
p += strlen(p);
|
|
*p = '\0';
|
|
}
|
|
else
|
|
{
|
|
p = append_str(buf, "PTR");
|
|
p = append_tab_16(buf);
|
|
sprintf(p, "%.5lX", prolog);
|
|
p += strlen(p);
|
|
*p = '\0';
|
|
}
|
|
return p;
|
|
}
|
|
|
|
if (op->prolog == DOCOL)
|
|
{
|
|
if (check_xlib(addr, tmp_str))
|
|
{
|
|
p = append_str(buf, "XLib Call");
|
|
p = append_tab_16(buf);
|
|
p = append_str(p, tmp_str);
|
|
return p;
|
|
}
|
|
}
|
|
|
|
p = append_str(buf, op->name);
|
|
|
|
if (op->length)
|
|
{
|
|
len = (read_nibbles(addr + 5, 5) - 5) / op->length;
|
|
sprintf(p, " %d", len);
|
|
p += strlen(p);
|
|
}
|
|
|
|
p = append_tab_16(buf);
|
|
addr += 5;
|
|
p = (*op->func)(&addr, p);
|
|
|
|
return p;
|
|
}
|
|
|