2019-11-12 17:03:51 -08:00
|
|
|
#include "psuedo.h"
|
2019-11-15 05:31:42 -08:00
|
|
|
#include "eval.h"
|
2019-11-12 17:03:51 -08:00
|
|
|
|
|
|
|
#define CLASS TPsuedoOp
|
|
|
|
|
|
|
|
CLASS::CLASS()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
CLASS::~CLASS()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-11-15 05:31:42 -08:00
|
|
|
constexpr unsigned int strhash(const char *str, int h = 0)
|
|
|
|
{
|
|
|
|
return !str[h] ? 5381 : (strhash(str, h + 1) * 33) ^ str[h];
|
|
|
|
}
|
|
|
|
|
|
|
|
int CLASS::doDATA(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
|
|
|
UNUSED(opinfo);
|
|
|
|
TEvaluator eval(a);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
int outct = 0;
|
|
|
|
int wordsize = 2;
|
|
|
|
int endian = 0;
|
|
|
|
std::string oper = line.operand;
|
|
|
|
std::string op = Poco::toUpper(Poco::trim(line.opcode));
|
|
|
|
Poco::StringTokenizer tok(oper, ",", Poco::StringTokenizer::TOK_TRIM |
|
|
|
|
Poco::StringTokenizer::TOK_IGNORE_EMPTY);
|
|
|
|
|
|
|
|
|
|
|
|
const char *ptr = (const char *)op.c_str();
|
|
|
|
switch (strhash(ptr) )
|
|
|
|
{
|
|
|
|
case strhash((const char *)"DA"):
|
|
|
|
case strhash((const char *)"DW"):
|
|
|
|
wordsize = 2;
|
|
|
|
break;
|
|
|
|
case strhash((const char *)"DDB"):
|
|
|
|
wordsize = 2;
|
|
|
|
endian = 1;
|
|
|
|
break;
|
|
|
|
case strhash((const char *)"DFB"):
|
|
|
|
case strhash((const char *)"DB"):
|
|
|
|
wordsize = 1;
|
|
|
|
break;
|
|
|
|
case strhash((const char *)"ADR"):
|
|
|
|
wordsize = 3;
|
|
|
|
break;
|
|
|
|
case strhash((const char *)"ADRL"):
|
|
|
|
wordsize = 4;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
wordsize=0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto itr = tok.begin(); itr != tok.end(); ++itr)
|
|
|
|
{
|
|
|
|
//printf("%s\n",(*itr).c_str());
|
|
|
|
//evaluate each of these strings, check for errors on pass 2
|
|
|
|
|
|
|
|
std::string expr = *itr;
|
|
|
|
int64_t eval_result = 0;
|
|
|
|
uint8_t shift;
|
|
|
|
int r;
|
|
|
|
uint8_t b;
|
|
|
|
|
|
|
|
shift=0;
|
|
|
|
r = eval.evaluate(expr, eval_result, shift);
|
|
|
|
if (r < 0)
|
|
|
|
{
|
|
|
|
//printf("eval error %d |%s|\n", r,expr.c_str());
|
|
|
|
if (a.pass > 0)
|
|
|
|
{
|
|
|
|
line.setError(errBadEvaluation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (shift=='>')
|
|
|
|
{
|
|
|
|
eval_result=(eval_result) & 0xFF;
|
|
|
|
}
|
|
|
|
if (shift=='<')
|
|
|
|
{
|
|
|
|
eval_result=(eval_result>>8) & 0xFF;
|
|
|
|
}
|
|
|
|
else if ((shift=='^') || (shift=='|'))
|
|
|
|
{
|
|
|
|
eval_result=(eval_result>>16)&0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
outct += wordsize;
|
|
|
|
if (a.pass > 0)
|
|
|
|
{
|
|
|
|
if (!endian) // little endian
|
|
|
|
{
|
|
|
|
for (i = 0; i < wordsize; i++)
|
|
|
|
{
|
|
|
|
b=(eval_result >> (8 * i))&0xFF;
|
|
|
|
line.outbytes.push_back(b);
|
|
|
|
//printf("%02X\n",b);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// big endian
|
|
|
|
for (i = 0; i < wordsize; i++)
|
|
|
|
{
|
|
|
|
b=(eval_result >> ((wordsize-1-i) * 8))&0xFF;
|
|
|
|
line.outbytes.push_back(b);
|
|
|
|
//printf("%02X\n",b);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
// SGQ - remove when complete
|
|
|
|
line.datafillct = outct;
|
|
|
|
line.datafillbyte = 0xCA;
|
|
|
|
// ===============
|
|
|
|
#endif
|
|
|
|
line.outbytect=outct;
|
|
|
|
return (outct);
|
|
|
|
}
|
2019-11-13 06:54:48 -08:00
|
|
|
|
|
|
|
int CLASS::doDS(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
UNUSED(opinfo);
|
|
|
|
|
2019-11-13 06:54:48 -08:00
|
|
|
int res = 0;
|
|
|
|
int32_t v = line.expr_value;
|
2019-11-14 17:08:11 -08:00
|
|
|
if (line.eval_result != 0)
|
2019-11-13 06:54:48 -08:00
|
|
|
{
|
|
|
|
line.setError(errForwardRef);
|
|
|
|
}
|
|
|
|
else if ((v < 0) || ((a.PC.currentpc + v) >= 0x10000)) // no neg, or crossing bank bound
|
|
|
|
{
|
|
|
|
line.setError(errOverflow);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
res = v;
|
2019-11-13 15:45:39 -08:00
|
|
|
|
2019-11-15 05:31:42 -08:00
|
|
|
line.datafillbyte = line.eval_result & 0xFF;
|
|
|
|
line.datafillct = v;
|
|
|
|
#if 0
|
2019-11-14 17:08:11 -08:00
|
|
|
if (a.pass > 0)
|
2019-11-13 15:45:39 -08:00
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
for (int i = 0; i < v; i++)
|
2019-11-13 15:45:39 -08:00
|
|
|
{
|
|
|
|
line.outbytes.push_back(0x00);
|
|
|
|
}
|
2019-11-14 17:08:11 -08:00
|
|
|
line.outbytect = v;
|
2019-11-13 15:45:39 -08:00
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
#endif
|
2019-11-13 15:45:39 -08:00
|
|
|
|
2019-11-13 06:54:48 -08:00
|
|
|
}
|
|
|
|
return (res);
|
|
|
|
}
|
|
|
|
|
2019-11-12 20:32:10 -08:00
|
|
|
int CLASS::doDUM(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
UNUSED(opinfo);
|
|
|
|
|
2019-11-13 06:54:48 -08:00
|
|
|
int res = 0;
|
|
|
|
bool isdend = ((opinfo.opcode == P_DEND) ? true : false);
|
|
|
|
|
|
|
|
if (!isdend)
|
|
|
|
{
|
|
|
|
a.dumstart = 1;
|
|
|
|
a.dumstartaddr = line.expr_value;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
a.dumstart = -1;
|
|
|
|
if (a.PCstack.size() == 0)
|
|
|
|
{
|
|
|
|
line.setError(errBadDUMop);
|
|
|
|
a.dumstart = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (res);
|
2019-11-12 20:32:10 -08:00
|
|
|
}
|
|
|
|
|
2019-11-12 17:03:51 -08:00
|
|
|
int CLASS::doLST(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
UNUSED(opinfo);
|
|
|
|
|
|
|
|
std::string s;
|
2019-11-12 17:03:51 -08:00
|
|
|
if (a.pass > 0)
|
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
s = Poco::toUpper(Poco::trim(line.operand));
|
2019-11-12 17:03:51 -08:00
|
|
|
if ((s == "") || (s == "ON") || (line.expr_value > 0))
|
|
|
|
{
|
|
|
|
//printf("ON\n");
|
|
|
|
a.skiplist = true;
|
|
|
|
a.listing = true;
|
|
|
|
}
|
|
|
|
else if ((s == "OFF") || (line.expr_value == 0))
|
|
|
|
{
|
|
|
|
//printf("OFF\n");
|
|
|
|
a.skiplist = true;
|
|
|
|
a.listing = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2019-11-13 20:09:45 -08:00
|
|
|
int CLASS::doHEX(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
UNUSED(opinfo);
|
2019-11-14 17:08:11 -08:00
|
|
|
|
|
|
|
std::string os = Poco::toUpper(Poco::trim(line.operand));
|
|
|
|
|
2019-11-15 05:31:42 -08:00
|
|
|
uint32_t bytect = 0;
|
|
|
|
uint8_t b = 0;
|
|
|
|
uint8_t ct = 0;
|
2019-11-14 17:08:11 -08:00
|
|
|
for ( uint32_t i = 0; i < os.length(); ++i )
|
|
|
|
{
|
|
|
|
char c = os[i];
|
|
|
|
|
2019-11-15 05:31:42 -08:00
|
|
|
if ((c >= '0') && (c <= '9'))
|
|
|
|
{
|
|
|
|
c = c - '0';
|
|
|
|
}
|
|
|
|
else if ((c >= 'a') && (c <= 'f'))
|
|
|
|
{
|
|
|
|
c = c - 'a' + 10;
|
|
|
|
}
|
|
|
|
else if ((c >= 'A') && (c <= 'F'))
|
|
|
|
{
|
|
|
|
c = c - 'A' + 10;
|
|
|
|
}
|
|
|
|
else if (c == ',')
|
2019-11-14 17:08:11 -08:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
else
|
2019-11-14 17:08:11 -08:00
|
|
|
{
|
|
|
|
line.setError(errBadOperand);
|
2019-11-15 05:31:42 -08:00
|
|
|
return 0;
|
2019-11-14 17:08:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Got a good char, append to hex string and see if we've got a byte
|
2019-11-15 05:31:42 -08:00
|
|
|
switch (ct)
|
2019-11-14 17:08:11 -08:00
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
case 0:
|
|
|
|
b = (c << 4);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
b |= c;
|
|
|
|
break;
|
2019-11-14 17:08:11 -08:00
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
ct = (ct + 1) & 0x01;
|
|
|
|
if (!ct)
|
2019-11-14 17:08:11 -08:00
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
if (a.pass > 0)
|
2019-11-14 17:08:11 -08:00
|
|
|
{
|
2019-11-15 05:31:42 -08:00
|
|
|
line.outbytes.push_back(b);
|
2019-11-14 17:08:11 -08:00
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
b = 0;
|
|
|
|
bytect++;
|
2019-11-14 17:08:11 -08:00
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
|
2019-11-14 17:08:11 -08:00
|
|
|
}
|
2019-11-15 05:31:42 -08:00
|
|
|
line.outbytect = bytect;
|
|
|
|
//printf("bytect=%d\n",bytect);
|
|
|
|
return bytect;
|
2019-11-13 20:09:45 -08:00
|
|
|
}
|
|
|
|
|
2019-11-15 05:31:42 -08:00
|
|
|
|
2019-11-12 17:03:51 -08:00
|
|
|
int CLASS::ProcessOpcode(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
|
|
|
|
switch (opinfo.opcode)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
res = -1; // undefined p-op
|
|
|
|
line.setError(errUnimplemented);
|
|
|
|
break;
|
2019-11-13 06:54:48 -08:00
|
|
|
case P_DS:
|
|
|
|
res = doDS(a, line, opinfo);
|
|
|
|
break;
|
2019-11-13 15:45:39 -08:00
|
|
|
case P_PUT:
|
|
|
|
case P_USE:
|
|
|
|
// both of these are handled by the input file processor, just allow them to be
|
|
|
|
// processed with no errors here
|
|
|
|
break;
|
2019-11-12 20:32:10 -08:00
|
|
|
case P_DUM:
|
|
|
|
case P_DEND:
|
2019-11-13 06:54:48 -08:00
|
|
|
res = doDUM(a, line, opinfo);
|
|
|
|
break;
|
2019-11-12 17:03:51 -08:00
|
|
|
case P_ORG:
|
2019-11-14 17:08:11 -08:00
|
|
|
if (line.operand.length() > 0)
|
2019-11-13 06:54:48 -08:00
|
|
|
{
|
2019-11-14 17:08:11 -08:00
|
|
|
a.PC.orgsave = a.PC.currentpc;
|
2019-11-13 06:54:48 -08:00
|
|
|
a.PC.currentpc = line.expr_value;
|
2019-11-14 17:08:11 -08:00
|
|
|
line.startpc = line.expr_value;
|
2019-11-13 06:54:48 -08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
a.PC.currentpc = a.PC.orgsave;
|
2019-11-14 17:08:11 -08:00
|
|
|
line.startpc = a.PC.orgsave;
|
2019-11-13 06:54:48 -08:00
|
|
|
}
|
2019-11-12 17:03:51 -08:00
|
|
|
break;
|
|
|
|
case P_SAV:
|
2019-11-15 05:31:42 -08:00
|
|
|
a.savepath = a.processFilename(line.operand, Poco::Path::current(), 0);
|
2019-11-12 17:03:51 -08:00
|
|
|
break;
|
|
|
|
case P_LST:
|
|
|
|
res = doLST(a, line, opinfo);
|
|
|
|
break;
|
2019-11-14 18:25:58 -08:00
|
|
|
case P_HEX:
|
2019-11-14 17:08:11 -08:00
|
|
|
res = doHEX(a, line, opinfo);
|
|
|
|
break;
|
2019-11-15 05:31:42 -08:00
|
|
|
case P_DATA:
|
|
|
|
res = doDATA(a, line, opinfo);
|
|
|
|
break;
|
2019-11-12 17:03:51 -08:00
|
|
|
}
|
|
|
|
return (res);
|
|
|
|
}
|