qasm/psuedo.cpp

1087 lines
20 KiB
C++
Raw Normal View History

#include "psuedo.h"
2019-11-15 05:09:41 -08:00
#include "eval.h"
#define CLASS TPsuedoOp
CLASS::CLASS()
{
}
CLASS::~CLASS()
{
}
2019-11-16 09:27:24 -08:00
2019-11-18 06:07:44 -08:00
uint32_t CLASS::doShift(uint32_t value, uint8_t shift)
{
if (shift == '<')
{
2019-11-19 17:09:58 -08:00
value = (value) & 0xFFFFFF;
2019-11-18 06:07:44 -08:00
}
if (shift == '>')
{
2019-11-19 17:09:58 -08:00
value = (value >> 8) & 0xFFFFFF;
2019-11-18 06:07:44 -08:00
}
else if ((shift == '^') || (shift == '|'))
{
2019-11-19 17:09:58 -08:00
value = (value >> 16) & 0xFFFFFF;
2019-11-18 06:07:44 -08:00
}
return (value);
}
2019-11-16 09:27:24 -08:00
int CLASS::doDO(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
TEvaluator eval(a);
2019-11-22 09:33:24 -08:00
eval.allowMX = true; // allow the built in MX symbol
2019-11-16 09:27:24 -08:00
2019-11-18 06:07:44 -08:00
int64_t eval_value = 0;
2019-11-16 09:27:24 -08:00
uint8_t shift;
2019-11-16 10:48:01 -08:00
uint32_t result32;
2019-11-16 09:27:24 -08:00
int res = 0;
int err = 0;
std::string op = Poco::toUpper(line.opcode);
2019-11-16 11:14:51 -08:00
std::string oper = line.operand_expr;
result32 = 0xFFFFFFFF;
2019-11-16 09:27:24 -08:00
if (op == "IF")
{
if (oper == "")
{
err = errIllegalCharOperand;
}
goto out;
}
if (op == "DO")
{
2019-11-16 10:48:01 -08:00
a.DOstack.push(a.curDO);
2019-11-16 09:27:24 -08:00
if (oper == "")
{
err = errIllegalCharOperand;
2019-11-16 11:14:51 -08:00
a.curDO.doskip = false;
2019-11-16 09:27:24 -08:00
goto out;
}
shift = 0;
2019-11-18 06:07:44 -08:00
eval_value = 0;
int x = eval.evaluate(line.operand_expr, eval_value, shift);
2019-11-16 09:27:24 -08:00
if (x < 0)
{
2019-11-16 10:48:01 -08:00
a.curDO.doskip = false;
2019-11-16 09:27:24 -08:00
err = errBadLabel;
if (a.pass == 0)
{
err = errForwardRef;
}
2019-11-16 10:48:01 -08:00
goto out;
2019-11-16 09:27:24 -08:00
}
2019-11-18 06:07:44 -08:00
result32 = eval_value & 0xFFFFFFFF;
2019-11-16 11:14:51 -08:00
a.curDO.doskip = (result32 != 0) ? false : true;
2019-11-16 10:48:01 -08:00
2019-11-16 09:27:24 -08:00
goto out;
}
if (op == "ELSE")
{
2019-11-16 10:48:01 -08:00
if (a.DOstack.size() > 0)
{
//line.flags |= FLAG_NOLINEPRINT;
a.curDO.doskip = !a.curDO.doskip;
}
else
{
err = errUnexpectedOp;
}
2019-11-16 09:27:24 -08:00
goto out;
}
if (op == "FIN")
{
//line.flags |= FLAG_NOLINEPRINT;
if (a.DOstack.size() > 0)
{
a.curDO = a.DOstack.top();
a.DOstack.pop();
}
else
{
// kind of a silent error here, just make sure we reinitialize
2019-11-16 10:48:01 -08:00
err = errUnexpectedOp;
2019-11-16 11:14:51 -08:00
a.curDO.doskip = false;
2019-11-16 09:27:24 -08:00
}
goto out;
}
out:
2019-11-16 10:48:01 -08:00
//printf("DO eval: %08X %s\n", result32, a.curDO.doskip ? "true" : "false");
2019-11-16 09:27:24 -08:00
if (err > 0)
{
line.setError(err);
}
return (res);
}
2019-11-20 21:50:00 -08:00
int CLASS::doMAC(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
int res = 0;
int err = 0;
std::string op = Poco::toUpper(line.opcode);
if (op == "MAC")
{
2019-11-22 09:33:24 -08:00
if (a.expand_macrostack.size() > 0)
2019-11-20 21:50:00 -08:00
{
2019-11-22 09:33:24 -08:00
line.flags |= FLAG_NOLINEPRINT;
2019-11-20 21:50:00 -08:00
goto out;
}
if (line.lable.length() == 0)
2019-11-20 21:50:00 -08:00
{
err = errBadLabel;
2019-11-20 21:50:00 -08:00
goto out;
}
a.macrostack.push(a.currentmacro);
a.currentmacro.clear();
a.currentmacro.name = line.lable;
a.currentmacro.lcname = Poco::toLower(line.lable);
a.currentmacro.start = line.lineno;
a.currentmacro.running = true;
2019-11-22 09:33:24 -08:00
if (!a.casesen)
{
a.currentmacro.name = Poco::toUpper(a.currentmacro.name);
}
2019-11-20 21:50:00 -08:00
if (a.pass == 0)
{
}
else
{
// don't need to do anything on pass > 0
}
//printf("macro stack size=%zu\n",a.macrostack.size());
2019-11-20 21:50:00 -08:00
}
2019-11-21 18:14:30 -08:00
else if (op == ">>>")
2019-11-21 08:47:03 -08:00
{
// don't do anything here, let the macro call handler stuff do ths (asm.cpp)
}
2019-11-20 21:50:00 -08:00
else // it is EOM or <<<
{
while (a.macrostack.size() > 0)
2019-11-20 21:50:00 -08:00
{
a.currentmacro.end = line.lineno - 1;
a.currentmacro.len = 0;
if (a.currentmacro.end >= a.currentmacro.start)
{
a.currentmacro.len = a.currentmacro.end - a.currentmacro.start;
//printf("macro len=%d\n",a.currentmacro.len);
}
a.currentmacro.running = false;
2019-11-20 21:50:00 -08:00
std::pair<std::string, TMacro> p(a.currentmacro.name, a.currentmacro);
//printf("macro insert %s\n",a.currentmacro.name.c_str());
2019-11-20 21:50:00 -08:00
a.macros.insert(p);
a.currentmacro = a.macrostack.top();
a.macrostack.pop();
}
#if 0
2019-11-20 21:50:00 -08:00
else
{
err = errUnexpectedOp;
goto out;
}
#endif
2019-11-20 21:50:00 -08:00
}
out:
if (err)
{
line.setError(err);
}
return (res);
}
2019-11-15 18:30:12 -08:00
int CLASS::doLUP(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
2019-11-15 22:27:43 -08:00
TEvaluator eval(a);
2019-11-18 04:50:02 -08:00
int64_t eval_value = 0;
2019-11-15 22:27:43 -08:00
uint8_t shift;
2019-11-15 18:30:12 -08:00
int lidx, len;
int res = 0;
int err = 0;
std::string op = Poco::toUpper(line.opcode);
if (op == "LUP")
{
2019-11-16 09:27:24 -08:00
line.flags |= FLAG_NOLINEPRINT;
2019-11-15 18:30:12 -08:00
len = line.lineno - 1; // MerlinLine line numbers are +1 from actual array idx
if (len >= 0)
{
2019-11-15 22:27:43 -08:00
shift = 0;
2019-11-18 04:50:02 -08:00
eval_value = 0;
int x = eval.evaluate(line.operand_expr, eval_value, shift);
2019-11-15 22:27:43 -08:00
2019-11-15 18:30:12 -08:00
a.LUPstack.push(a.curLUP);
2019-11-21 18:14:30 -08:00
if (a.expand_macrostack.size() > 0)
{
a.curLUP.lupoffset = a.expand_macro.currentline;
2019-11-21 08:47:03 -08:00
}
else
2019-11-16 09:27:24 -08:00
{
2019-11-21 18:14:30 -08:00
a.curLUP.lupoffset = len;
2019-11-16 09:27:24 -08:00
}
2019-11-18 04:50:02 -08:00
a.curLUP.lupct = eval_value & 0xFFFF; // evaluate here
2019-11-15 18:30:12 -08:00
a.curLUP.luprunning++;
2019-11-16 09:27:24 -08:00
2019-11-18 04:50:02 -08:00
if ((x < 0) || (eval_value <= 0) || (eval_value > 0x8000))
2019-11-16 09:27:24 -08:00
{
// merlin just ignores LUP if the value is out of range
a.curLUP.lupct = 0;
a.curLUP.lupskip = true;
}
2019-11-15 18:30:12 -08:00
}
else
{
err = errUnexpectedOp;
}
}
if (op == "--^")
{
2019-11-16 09:27:24 -08:00
line.flags |= FLAG_NOLINEPRINT;
2019-11-15 18:30:12 -08:00
if (a.curLUP.luprunning > 0)
{
2019-11-21 08:47:03 -08:00
2019-11-15 18:30:12 -08:00
lidx = line.lineno - 1;
len = lidx - a.curLUP.lupoffset - 1;
if (a.curLUP.lupct > 0)
{
a.curLUP.lupct--;
if (a.curLUP.lupct != 0)
{
2019-11-21 18:14:30 -08:00
if (a.expand_macrostack.size() > 0)
{
a.expand_macro.currentline = a.curLUP.lupoffset;
}
else
2019-11-21 08:47:03 -08:00
{
2019-11-21 18:14:30 -08:00
a.lineno = a.curLUP.lupoffset;
2019-11-21 08:47:03 -08:00
}
2019-11-15 18:30:12 -08:00
goto out;
}
}
2019-11-15 22:27:43 -08:00
// kind of a silent error here, just make sure we reinitialize
2019-11-16 09:27:24 -08:00
a.curLUP.luprunning = 0;
a.curLUP.lupct = 0;
a.curLUP.lupskip = false;
2019-11-15 18:30:12 -08:00
//printf("start=%d end=%d len=%d\n", a.curLUP.lupoffset, lidx, len);
if (a.LUPstack.size() > 0)
{
a.curLUP = a.LUPstack.top();
a.LUPstack.pop();
}
else
{
err = errUnexpectedOp;
}
}
else
{
2019-11-16 09:27:24 -08:00
a.curLUP.lupskip = false;
2019-11-15 22:27:43 -08:00
// SGQ - found a '--^' without a LUP, should we just ignore?
//err = errUnexpectedOp;
2019-11-15 18:30:12 -08:00
}
}
out:
if (err > 0)
{
line.setError(err);
}
return (res);
}
2019-11-14 23:35:04 -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)
{
2019-11-15 03:40:35 -08:00
UNUSED(opinfo);
2019-11-15 05:09:41 -08:00
TEvaluator eval(a);
int i;
2019-11-14 23:35:04 -08:00
int outct = 0;
int wordsize = 2;
int endian = 0;
2019-11-17 13:32:32 -08:00
std::string oper = line.operand;
2019-11-14 23:35:04 -08:00
std::string op = Poco::toUpper(Poco::trim(line.opcode));
2019-11-17 13:32:32 -08:00
//printf("DFB TOK1 : |%s|\n", oper.c_str());
2019-11-18 04:50:02 -08:00
2019-11-18 06:07:44 -08:00
line.eval_result = 0; // since this is an data p-op, clear the global 'bad operand' flag
2019-11-18 04:50:02 -08:00
2019-11-14 23:35:04 -08:00
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;
2019-11-15 05:09:41 -08:00
default:
2019-11-15 16:15:58 -08:00
wordsize = 0;
2019-11-15 05:09:41 -08:00
break;
2019-11-14 23:35:04 -08:00
}
for (auto itr = tok.begin(); itr != tok.end(); ++itr)
{
//evaluate each of these strings, check for errors on pass 2
2019-11-15 05:09:41 -08:00
std::string expr = *itr;
2019-11-17 13:32:32 -08:00
//printf("DFB TOK : |%s|\n", expr.c_str());
2019-11-18 04:50:02 -08:00
int64_t eval_value = 0;
2019-11-15 05:09:41 -08:00
uint8_t shift;
int r;
uint8_t b;
2019-11-17 13:32:32 -08:00
if (expr.length() > 0)
2019-11-15 05:09:41 -08:00
{
2019-11-17 13:32:32 -08:00
if (expr[0] == '#')
2019-11-15 05:09:41 -08:00
{
2019-11-17 13:32:32 -08:00
expr[0] = ' ';
expr = Poco::trim(expr);
}
shift = 0;
2019-11-18 04:50:02 -08:00
eval_value = 0;
2019-11-17 13:32:32 -08:00
//printf("DFB EVAL: |%s|\n", expr.c_str());
2019-11-18 04:50:02 -08:00
r = eval.evaluate(expr, eval_value, shift);
2019-11-17 13:32:32 -08:00
if (r < 0)
{
2019-11-17 22:16:00 -08:00
//printf("error %d\n",r);
2019-11-17 13:32:32 -08:00
if (a.pass > 0)
{
line.setError(errBadEvaluation);
}
}
2019-11-18 12:45:52 -08:00
eval_value = (uint64_t)doShift((uint32_t)eval_value, shift);
2019-11-15 05:09:41 -08:00
}
2019-11-14 23:35:04 -08:00
outct += wordsize;
if (a.pass > 0)
{
if (!endian) // little endian
{
2019-11-15 05:09:41 -08:00
for (i = 0; i < wordsize; i++)
{
2019-11-18 04:50:02 -08:00
b = (eval_value >> (8 * i)) & 0xFF;
2019-11-15 05:09:41 -08:00
line.outbytes.push_back(b);
//printf("%02X\n",b);
}
2019-11-14 23:35:04 -08:00
}
else
{
// big endian
2019-11-15 05:09:41 -08:00
for (i = 0; i < wordsize; i++)
{
2019-11-18 04:50:02 -08:00
b = (eval_value >> ((wordsize - 1 - i) * 8)) & 0xFF;
2019-11-15 05:09:41 -08:00
line.outbytes.push_back(b);
//printf("%02X\n",b);
}
2019-11-14 23:35:04 -08:00
}
}
}
2019-11-15 16:15:58 -08:00
line.outbytect = outct;
2019-11-14 23:35:04 -08:00
return (outct);
}
2019-11-20 21:50:00 -08:00
int CLASS::doDS(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
2019-11-15 03:40:35 -08:00
UNUSED(opinfo);
int res = 0;
2019-11-18 06:07:44 -08:00
TEvaluator eval(a);
int64_t eval_value = 0;
uint8_t shift;
line.eval_result = 0; // since this is an data p-op, clear the global 'bad operand' flag
2019-11-20 21:50:00 -08:00
line.flags |= FLAG_FORCEADDRPRINT;
2019-11-18 06:07:44 -08:00
std::string s;
Poco::StringTokenizer tok(line.operand, ",", Poco::StringTokenizer::TOK_TRIM |
Poco::StringTokenizer::TOK_IGNORE_EMPTY);
int32_t datact = 0;
uint8_t fill = 0x0;
bool pagefill = false;
int32_t v = 0;
int ct = 0;
for (auto itr = tok.begin(); itr != tok.end(); ++itr)
{
2019-11-18 06:07:44 -08:00
s = *itr;
if (ct == 0)
{
if (s == "\\")
{
pagefill = true;
}
else
{
shift = 0;
eval_value = 0;
int x = eval.evaluate(s, eval_value, shift);
if (x < 0)
{
line.setError(errBadOperand);
goto out;
}
2019-11-18 12:45:52 -08:00
eval_value = (uint64_t)doShift((uint32_t)eval_value, shift);
2019-11-18 06:07:44 -08:00
datact = eval_value & 0xFFFF;
if (datact < 0)
{
line.setError(errBadOperand);
goto out;
}
}
}
else if (ct == 1)
{
shift = 0;
eval_value = 0;
int x = eval.evaluate(s, eval_value, shift);
if (x < 0)
{
line.setError(errBadOperand);
goto out;
}
2019-11-18 12:45:52 -08:00
eval_value = (uint64_t)doShift((uint32_t)eval_value, shift);
2019-11-18 06:07:44 -08:00
fill = eval_value & 0xFF;
}
else if (ct > 1)
{
line.setError(errBadOperand);
}
ct++;
}
2019-11-18 06:07:44 -08:00
line.datafillbyte = fill;
v = datact;
if (pagefill)
{
2019-11-20 21:50:00 -08:00
v = line.startpc & 0xFF;
v = 0x100 - v;
}
2019-11-18 06:07:44 -08:00
line.datafillct = (uint16_t)v & 0xFFFF;
2019-11-20 21:50:00 -08:00
res = line.datafillct;
2019-11-13 15:45:39 -08:00
2019-11-18 06:07:44 -08:00
out:
//printf("res=%d %04X\n",res,res);
return (res);
}
2019-11-12 20:32:10 -08:00
int CLASS::doDUM(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
2019-11-15 03:40:35 -08:00
UNUSED(opinfo);
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
}
int CLASS::doLST(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
2019-11-15 03:40:35 -08:00
UNUSED(opinfo);
std::string s;
if (a.pass > 0)
{
2019-11-16 10:48:01 -08:00
s = Poco::toUpper(Poco::trim(line.operand_expr));
2019-11-17 13:32:32 -08:00
if (s == "")
2019-11-16 22:48:24 -08:00
{
2019-11-17 13:32:32 -08:00
a.listing = true;
a.skiplist = true;
2019-11-16 22:48:24 -08:00
}
else if (s == "RTN")
2019-11-16 11:14:51 -08:00
{
if (a.LSTstack.size())
{
a.listing = a.LSTstack.top();
a.LSTstack.pop();
}
}
else if ((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-16 11:14:51 -08:00
int CLASS::doTR(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
std::string s;
if (a.pass > 0)
{
s = Poco::toUpper(Poco::trim(line.operand_expr));
if (s == "ADR")
{
a.truncdata |= 0x03;
}
else if ((s == "ON") || (line.expr_value > 0))
{
a.truncdata |= 0x01;;
}
else if ((s == "OFF") || (line.expr_value == 0))
{
a.truncdata = 0x00;
}
}
return (0);
}
2019-11-17 18:16:39 -08:00
char hexVal( char c )
{
2019-11-17 22:16:00 -08:00
char v = -1;
if ((c >= '0') && (c <= '9'))
{
v = c - '0';
}
else if ((c >= 'a') && (c <= 'f'))
{
v = c - 'a' + 10;
}
else if ((c >= 'A') && (c <= 'F'))
{
v = c - 'A' + 10;
}
return v;
2019-11-17 18:16:39 -08:00
}
2019-11-14 17:04:35 -08:00
int CLASS::doHEX(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
2019-11-15 03:40:35 -08:00
UNUSED(opinfo);
2019-11-15 05:09:41 -08:00
2019-11-17 22:16:00 -08:00
std::string os = Poco::trim(line.operand);
2019-11-14 17:04:35 -08:00
2019-11-18 06:07:44 -08:00
line.eval_result = 0; // since this is an data p-op, clear the global 'bad operand' flag
2019-11-18 04:50:02 -08:00
2019-11-14 23:35:04 -08:00
uint32_t bytect = 0;
uint8_t b = 0;
uint8_t ct = 0;
2019-11-16 09:27:24 -08:00
2019-11-14 17:04:35 -08:00
for ( uint32_t i = 0; i < os.length(); ++i )
{
char c = os[i];
2019-11-17 18:16:39 -08:00
if (c == ',')
2019-11-14 17:04:35 -08:00
{
continue;
}
2019-11-17 22:16:00 -08:00
char hv = hexVal(c);
if ( hv < 0 )
2019-11-14 17:04:35 -08:00
{
2019-11-16 09:27:24 -08:00
line.setError(errIllegalCharOperand);
bytect = 0;
goto out;
2019-11-14 17:04:35 -08:00
}
// Got a good char, append to hex string and see if we've got a byte
2019-11-14 23:35:04 -08:00
switch (ct)
2019-11-14 17:04:35 -08:00
{
2019-11-14 21:21:03 -08:00
case 0:
2019-11-17 18:16:39 -08:00
b = (hv << 4);
2019-11-14 21:21:03 -08:00
break;
case 1:
2019-11-17 18:16:39 -08:00
b |= hv;
2019-11-14 21:21:03 -08:00
break;
2019-11-14 17:04:35 -08:00
}
2019-11-14 23:35:04 -08:00
ct = (ct + 1) & 0x01;
2019-11-14 21:21:03 -08:00
if (!ct)
2019-11-14 17:04:35 -08:00
{
2019-11-14 23:35:04 -08:00
if (a.pass > 0)
2019-11-14 17:04:35 -08:00
{
2019-11-14 21:21:03 -08:00
line.outbytes.push_back(b);
2019-11-14 17:04:35 -08:00
}
2019-11-14 23:35:04 -08:00
b = 0;
2019-11-14 21:21:03 -08:00
bytect++;
2019-11-14 17:04:35 -08:00
}
2019-11-15 16:15:58 -08:00
}
2019-11-17 18:16:39 -08:00
2019-11-15 16:15:58 -08:00
if (ct & 0x01) // we got an odd number of nibbles
{
2019-11-16 09:27:24 -08:00
line.setError(errBadOperand);
2019-11-15 16:15:58 -08:00
bytect = 0;
2019-11-14 17:04:35 -08:00
}
2019-11-16 09:27:24 -08:00
out:
2019-11-14 23:35:04 -08:00
line.outbytect = bytect;
2019-11-14 21:21:03 -08:00
return bytect;
2019-11-14 17:04:35 -08:00
}
static int usr_hash(std::string os) {
int hash = 0;
os.resize(4, ' ');
os = Poco::toUpper(os);
hash = (os[0] & 0x1f) << 11;
hash |= (os[1] & 0x1f) << 6;
hash |= (os[2] & 0x1f) << 1;
if (os[3] == 'L') hash |= 0x01;
return hash;
}
/*
handler for USR. generates a 2-byte opcode hash
USR $00 - equivalent to DW $0 (qasm)
USR 'ADC ' - hashes the 4-byte string (qasm)
USR ADC - hashes the 1-4 byte symbol (merlin)
*/
int CLASS::doUSR(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
line.eval_result = 0; // since this is an data p-op, clear the global 'bad operand' flag
std::string os = Poco::trim(line.operand);
uint32_t bytect = 2;
uint32_t hash = 0;
char c = os.empty() ? 0 : os.front();
if (c == '$' && os.length() > 1) {
for (uint32_t i = 1; i < os.length(); ++i)
{
char hv = hexVal(os[i]);
if (hv < 0) {
line.setError(errIllegalCharOperand);
bytect = 0;
goto out;
}
hash <<= 4;
hash |= hv;
}
} else if ((c == '\'' || c == '\"') && os.front() == os.back() && os.length() == 6) {
hash = usr_hash(os.substr(1, 4));
} else if (os.length() > 0 && os.length() <= 4) {
hash = usr_hash(os);
} else {
printf("line.setError(errBadOperand);\n");
line.setError(errBadOperand);
bytect = 0;
goto out;
}
if (a.pass > 0) {
line.outbytes.push_back(hash & 0xff);
line.outbytes.push_back(hash >> 8);
}
out:
line.outbytect = bytect;
return bytect;
}
2019-11-17 22:16:00 -08:00
// the handler for STR,STRL,REV,FLS,INV,DCI,ASC
2019-11-17 18:16:39 -08:00
int CLASS::doASC(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
UNUSED(opinfo);
2019-11-17 22:16:00 -08:00
std::string os = line.operand;
std::string op = Poco::toUpper(line.opcode);
2019-11-17 18:16:39 -08:00
2019-11-18 12:45:52 -08:00
uint8_t firstdelim = 0;
2019-11-17 18:16:39 -08:00
uint32_t bytect = 0;
uint8_t b = 0;
2019-11-17 22:16:00 -08:00
uint8_t b1;
2019-11-17 18:16:39 -08:00
uint8_t ct = 0;
2019-11-18 12:45:52 -08:00
uint8_t delimiter = 0;
2019-11-17 22:16:00 -08:00
uint32_t ss = 0;
2019-11-18 12:45:52 -08:00
uint32_t lastdelimidx = 0;
2019-11-17 22:16:00 -08:00
std::vector<uint8_t> bytes;
2019-11-17 18:16:39 -08:00
2019-11-18 06:07:44 -08:00
line.eval_result = 0; // since this is an ASCII p-op, clear the global 'bad operand' flag
2019-11-17 18:16:39 -08:00
for ( uint32_t i = 0; i < os.length(); ++i )
{
2019-11-18 12:45:52 -08:00
uint8_t c = os[i];
2019-11-17 22:16:00 -08:00
// are we inside a delimited string?
if ( delimiter )
{
if ( c == delimiter )
{
bytect += (i - ss);
if ( a.pass > 0 )
{
for ( ; ss < i; ++ss )
{
c = os[ss];
if ( delimiter >= '\'' )
{
c &= 0x7F;
}
else
{
c |= 0x80;
}
2019-11-18 12:45:52 -08:00
2019-11-17 22:16:00 -08:00
bytes.push_back(c);
2019-11-18 12:45:52 -08:00
lastdelimidx = (uint32_t)(bytes.size() - 1);
2019-11-17 22:16:00 -08:00
}
}
delimiter = 0;
ss = 0;
continue;
}
}
else
{
// No, check for seperator characters
if ( c == ',' || c == ' ' )
{
continue;
}
// Is this a hex char?
char hv = hexVal(c);
if ( hv < 0 )
{
// if not a hex value, then consider the character to be the string delimiter
delimiter = c;
2019-11-18 12:45:52 -08:00
if( ! firstdelim )
2019-11-17 22:16:00 -08:00
{
firstdelim = c;
}
else if (delimiter != firstdelim)
{
line.setError(errIllegalCharOperand);
}
ss = i + 1;
continue;
}
// Got a hex char, append to hex string and see if we've got a byte
switch (ct)
{
case 0:
b = (hv << 4);
break;
case 1:
b |= hv;
break;
}
ct = (ct + 1) & 0x01;
if (!ct)
{
if (a.pass > 0)
{
bytes.push_back(b);
}
b = 0;
bytect++;
}
}
2019-11-17 18:16:39 -08:00
}
2019-11-17 22:16:00 -08:00
if ( delimiter || (ct & 0x01) ) // error w/unterminated string or we got an odd number of nibbles in hex value
2019-11-17 18:16:39 -08:00
{
line.setError(errBadOperand);
bytect = 0;
}
2019-11-17 22:16:00 -08:00
else // now figure out what psuedo op we are and transfer the data to outbytect
{
uint32_t i;
bool reverse = false;
bool dci = false;
uint8_t andval = 0xFF;
uint8_t orval = 0x00;
uint8_t addlen = 0;
2019-11-18 12:45:52 -08:00
uint32_t truebytect = (uint32_t)bytes.size();
2019-11-17 22:16:00 -08:00
const char *ptr = (const char *)op.c_str();
//printf("bytect=%d bytes.size()=%zu\n",bytect,bytes.size());
switch (strhash(ptr) )
{
case strhash((const char *)"STRL"):
addlen = 2;
break;
case strhash((const char *)"STR"):
addlen = 1;
break;
case strhash((const char *)"REV"):
reverse = true;
break;
case strhash((const char *)"FLS"):
andval = (uint8_t)~0xC0;
orval = (uint8_t)0x40;
break;
case strhash((const char *)"INV"):
andval = (uint8_t)~0xC0;
orval = 0x00;
break;
case strhash((const char *)"DCI"):
dci = true;
break;
case strhash((const char *)"ASC"):
break;
default:
line.setError(errBadOpcode);
bytect = 0;
addlen = 0;
break;
}
if (a.pass > 0)
{
for (i = 0; i < addlen; i++) // if a string, push length
{
line.outbytes.push_back((truebytect >> (i * 8)) & 0xFF);
}
for (i = 0; i < truebytect; i++)
{
if (reverse)
{
b = bytes[bytect - i - 1];
}
else
{
b = bytes[i];
}
2019-11-18 12:45:52 -08:00
b1 = b & 0x7F;
2019-11-18 06:07:44 -08:00
if ((andval != 0xFF) || (orval != 0x00))
{
b = b1;
}
2019-11-18 12:45:52 -08:00
2019-11-17 22:16:00 -08:00
if ((b1 < 0x60))
{
b &= andval; // strip whatever bits needed to flash or invert
b |= orval;
}
2019-11-18 12:45:52 -08:00
if (dci && (i == lastdelimidx))
2019-11-17 22:16:00 -08:00
{
2019-11-18 12:45:52 -08:00
//lr - Merlin only toggles the high bit of string chars, not hex values
// 8D,'Hello',8D,'there',8D becomes 8D 48 65 6C 6C 6F 8D 74 68 65 72 E5
//
// The DCI instruction is documented to work like this on page 108
// (regardless of how this effects the desired lda, (bpl/bmi) functionality)
//
// I am now checking the delimiter character to determine hi/lo toggle (reversed)
// and am tracking the index to the last delimited character put into 'bytes'.
// This produces the same results as Merlin 16+ in my testing.
if ( firstdelim >= '\'' )
2019-11-17 22:16:00 -08:00
{
b |= 0x80;
}
else
{
b &= 0x7F;
}
}
line.outbytes.push_back(b);
}
}
bytect = bytect + addlen;
}
//printf("XXX bytect=%d bytes.size()=%zu\n",bytect,bytes.size());
2019-11-17 18:16:39 -08:00
line.outbytect = bytect;
return bytect;
}
2019-11-14 17:04:35 -08:00
int CLASS::ProcessOpcode(T65816Asm &a, MerlinLine &line, TSymbol &opinfo)
{
int res = 0;
2019-11-22 09:33:24 -08:00
std::string s;
switch (opinfo.opcode)
{
default:
res = -1; // undefined p-op
line.setError(errUnimplemented);
break;
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:
res = doDUM(a, line, opinfo);
2019-11-15 22:27:43 -08:00
line.flags |= FLAG_FORCEADDRPRINT;
2019-11-15 18:51:30 -08:00
break;
case P_ORG:
2019-11-16 10:48:01 -08:00
if (line.operand_expr.length() > 0)
{
2019-11-14 17:04:35 -08:00
a.PC.orgsave = a.PC.currentpc;
a.PC.currentpc = line.expr_value;
2019-11-14 17:04:35 -08:00
line.startpc = line.expr_value;
}
else
{
a.PC.currentpc = a.PC.orgsave;
2019-11-14 17:04:35 -08:00
line.startpc = a.PC.orgsave;
}
2019-11-19 13:41:27 -08:00
#if 0
// Merlin32 seems to have a bug where ORG seems like it can only be 16 bits
2019-11-20 21:50:00 -08:00
if ((line.syntax & SYNTAX_MERLIN32) == SYNTAX_MERLIN32)
2019-11-19 13:41:27 -08:00
{
// so clear the bank word in all variables
a.PC.orgsave &= 0xFFFF;
2019-11-20 21:50:00 -08:00
a.PC.currentpc &= 0xFFFF;
line.startpc &= 0xFFFF;
2019-11-19 13:41:27 -08:00
}
#endif
2019-11-15 22:27:43 -08:00
line.flags |= FLAG_FORCEADDRPRINT;
break;
case P_SAV:
2019-11-14 17:04:35 -08:00
a.savepath = a.processFilename(line.operand, Poco::Path::current(), 0);
break;
2019-11-22 09:33:24 -08:00
case P_CAS:
s = Poco::toUpper(line.operand);
if (s == "SE")
{
a.casesen = true;
}
if (s=="IN")
{
a.casesen=false;
}
res = 0;
break;
2019-11-20 21:50:00 -08:00
case P_MAC:
res = doMAC(a, line, opinfo);
break;
case P_ERR:
2019-11-20 21:50:00 -08:00
if (a.pass > 0)
{
2019-11-20 21:50:00 -08:00
if ((line.expr_value != 0) || (line.eval_result < 0))
{
line.setError(errErrOpcode);
//a.passcomplete=true; // terminate assembly
}
}
2019-11-20 21:50:00 -08:00
res = 0;
break;
case P_LST:
res = doLST(a, line, opinfo);
break;
2019-11-14 17:04:35 -08:00
case P_HEX:
res = doHEX(a, line, opinfo);
break;
2019-11-14 23:35:04 -08:00
case P_DATA:
res = doDATA(a, line, opinfo);
break;
2019-11-15 18:30:12 -08:00
case P_LUP:
res = doLUP(a, line, opinfo);
break;
2019-11-16 09:27:24 -08:00
case P_DO:
res = doDO(a, line, opinfo);
break;
2019-11-16 11:14:51 -08:00
case P_TR:
2019-11-17 13:32:32 -08:00
res = doTR(a, line, opinfo);
2019-11-16 11:14:51 -08:00
break;
2019-11-17 22:16:00 -08:00
case P_ASC:
res = doASC(a, line, opinfo);
break;
case P_USR:
res = doUSR(a, line, opinfo);
break;
}
return (res);
}