1
0
mirror of https://github.com/pfusik/xasm.git synced 2025-01-10 17:30:27 +00:00
xasm/source/app.d
Piotr Fusik 70c907bae6 Revert "Warn about a comment starting with a comma."
Breaks Jaskier's MPT player.
https://sourceforge.net/p/asap/bugs/32/

This reverts commit 7a666799ac0f90a323ad556445ec150ceccfd47c.
2023-05-07 21:03:40 +02:00

3062 lines
60 KiB
D

// xasm 3.2.1 by Piotr Fusik <fox@scene.pl>
// http://xasm.atari.org
// Can be compiled with DMD v2.101.0.
// Poetic License:
//
// This work 'as-is' we provide.
// No warranty express or implied.
// We've done our best,
// to debug and test.
// Liability for damages denied.
//
// Permission is granted hereby,
// to copy, share, and modify.
// Use as is fit,
// free or for profit.
// These rights, on this notice, rely.
import std.algorithm;
import std.array;
import std.ascii;
import std.conv;
import std.file;
import std.math;
import std.path;
import std.stdio;
import std.string;
import std.exception : assumeUnique;
import std.range : empty, front, popFront;
version (Windows) {
import core.sys.windows.windows;
}
const string TITLE = "xasm 3.2.1";
File messageStream;
string sourceFilename = null;
bool[26] options;
string[26] optionParameters;
string[] commandLineDefinitions = null;
string objectFilename = null;
string[] makeSources = null;
immutable(ubyte)[][string] sourceFiles;
int exitCode = 0;
int totalLines;
bool pass2 = false;
bool optionFill; // opt f
bool option5200; // opt g
bool optionHeaders; // opt h
bool optionListing; // opt l
bool optionObject; // opt o
bool optionUnusedLabels; // opt u
string currentFilename;
int lineNo;
int includeLevel = 0;
string line;
int column;
bool foundEnd;
class AssemblyError : Exception {
this(string msg) {
super(msg);
}
}
class Label {
int value;
bool unused = true;
bool unknownInPass1 = false;
bool passed = false;
this(int value) {
this.value = value;
}
}
Label[string] labelTable;
Label currentLabel;
string lastGlobalLabel;
alias int function(int a, int b) OperatorFunction;
bool inOpcode = false;
struct ValOp {
int value;
OperatorFunction func;
int priority;
}
ValOp[] valOpStack;
int value;
bool unknownInPass1;
enum AddrMode {
ACCUMULATOR = 0,
IMMEDIATE = 1,
ABSOLUTE = 2,
ZEROPAGE = 3,
ABSOLUTE_X = 4,
ZEROPAGE_X = 5,
ABSOLUTE_Y = 6,
ZEROPAGE_Y = 7,
INDIRECT_X = 8,
INDIRECT_Y = 9,
INDIRECT = 10,
ABS_OR_ZP = 11, // temporarily used in readAddrMode
STANDARD_MASK = 15,
INCREMENT = 0x20,
DECREMENT = 0x30,
ZERO = 0x40
}
AddrMode addrMode;
enum OrgModifier {
NONE,
FORCE_HEADER,
FORCE_FFFF,
RELOCATE
}
int origin = -1;
int loadOrigin;
int loadingOrigin;
ushort[] blockEnds;
int blockIndex;
bool repeating; // line
int repeatCounter; // line
bool instructionBegin;
bool sequencing;
bool willSkip;
bool skipping;
ushort[] skipOffsets;
int skipOffsetsIndex = 0;
int repeatOffset; // instruction repeat
bool wereManyInstructions;
alias void function(int move) MoveFunction;
int value1;
AddrMode addrMode1;
int value2;
AddrMode addrMode2;
struct IfContext {
bool condition;
bool wasElse;
bool aConditionMatched;
}
IfContext[] ifContexts;
File listingStream;
char[32] listingLine;
int listingColumn;
string lastListedFilename = null;
File objectStream;
int objectBytes = 0;
bool getOption(char letter) {
assert(letter >= 'a' && letter <= 'z');
return options[letter - 'a'];
}
void warning(string msg, bool error = false) {
messageStream.flush();
version (Windows) {
HANDLE stderrHandle = GetStdHandle(STD_ERROR_HANDLE);
CONSOLE_SCREEN_BUFFER_INFO csbi;
GetConsoleScreenBufferInfo(stderrHandle, &csbi);
SetConsoleTextAttribute(stderrHandle, (csbi.wAttributes & ~0xf) | (error ? 12 : 14));
scope (exit) SetConsoleTextAttribute(stderrHandle, csbi.wAttributes);
}
if (line !is null)
stderr.writeln(line);
stderr.writefln("%s (%d) %s: %s", currentFilename, lineNo, error ? "ERROR" : "WARNING", msg);
exitCode = 1;
}
void illegalCharacter() {
throw new AssemblyError("Illegal character");
}
bool eol() {
return column >= line.length;
}
char readChar() {
if (eol())
throw new AssemblyError("Unexpected end of line");
return line[column++];
}
int readDigit(int base) {
if (eol()) return -1;
int r = line[column];
if (r >= '0' && r <= '9') {
r -= '0';
} else {
r &= 0xdf;
if (r >= 'A' && r <= 'Z')
r -= 'A' - 10;
else
return -1;
}
if (r < base) {
column++;
return r;
}
return -1;
}
int readNumber(int base) {
long r = readDigit(base);
if (r < 0)
illegalCharacter();
do {
int d = readDigit(base);
if (d < 0)
return cast(int) r;
r = r * base + d;
} while (r <= 0x7fffffff);
throw new AssemblyError("Number too big");
}
void readSpaces() {
switch (readChar()) {
case '\t':
case ' ':
break;
default:
throw new AssemblyError("Space expected");
}
while (!eol()) {
switch (line[column]) {
case '\t':
case ' ':
column++;
break;
default:
return;
}
}
}
string readLabel() {
int firstColumn = column;
while (!eol()) {
char c = line[column++];
if (c >= '0' && c <= '9' || c == '_' || c == '?')
continue;
c &= 0xdf;
if (c >= 'A' && c <= 'Z')
continue;
column--;
break;
}
string label = line[firstColumn .. column].toUpper;
if (label.startsWith('?')) {
if (lastGlobalLabel is null)
throw new AssemblyError("Global label must be declared first");
label = lastGlobalLabel ~ label;
}
return label >= "A" ? label : null;
}
void readComma() {
if (readChar() != ',')
throw new AssemblyError("Bad or missing function parameter");
}
string readInstruction() {
string r;
for (int i = 0; i < 3; i++) {
char c = readChar() & 0xdf;
if (c < 'A' || c > 'Z')
throw new AssemblyError("Illegal instruction");
r ~= c;
}
return r;
}
string readFunction() {
if (column + 5 >= line.length) return null;
if (line[column + 3] != '(') return null;
string r;
for (int i = 0; i < 3; i++) {
char c = line[column + i] & 0xdf;
if (c < 'A' || c > 'Z') return null;
r ~= c;
}
column += 4;
return r;
}
string readFilename() {
readSpaces();
char delimiter = readChar();
switch (delimiter) {
case '"':
case '\'':
string filename;
char c;
while ((c = readChar()) != delimiter)
filename ~= c;
return filename;
default:
illegalCharacter();
}
assert(0);
}
void readStringChar(char c) {
if (readChar() != c)
throw new AssemblyError("String error");
}
ubyte[] readString() {
if (eol()) return null;
char delimiter = readChar();
switch (delimiter) {
case '"':
case '\'':
ubyte[] r;
for (;;) {
char c = readChar();
if (c == delimiter) {
if (eol()) return r;
if (line[column] != delimiter) {
if (line[column] == '*') {
column++;
foreach (ref ubyte b; r)
b ^= 0x80;
}
return r;
}
column++;
}
r ~= cast(ubyte) c;
}
default:
column--;
return null;
}
}
void checkNoExtraCharacters() {
if (eol()) return;
switch (line[column]) {
case '\t':
case ' ':
return;
default:
throw new AssemblyError("Extra characters on line");
}
}
void checkOriginDefined() {
if (origin < 0)
throw new AssemblyError("No ORG specified");
}
int operatorPlus(int a, int b) {
return b;
}
int operatorMinus(int a, int b) {
return -b;
}
int operatorLow(int a, int b) {
return b & 0xff;
}
int operatorHigh(int a, int b) {
return (b >> 8) & 0xff;
}
int operatorLogicalNot(int a, int b) {
return !b;
}
int operatorBitwiseNot(int a, int b) {
return ~b;
}
int operatorAdd(int a, int b) {
long r = cast(long) a + b;
if (r < -0x80000000L || r > 0x7fffffffL)
throw new AssemblyError("Arithmetic overflow");
return a + b;
}
int operatorSubtract(int a, int b) {
long r = cast(long) a - b;
if (r < -0x80000000L || r > 0x7fffffffL)
throw new AssemblyError("Arithmetic overflow");
return a - b;
}
int operatorMultiply(int a, int b) {
long r = cast(long) a * b;
if (r < -0x80000000L || r > 0x7fffffffL)
throw new AssemblyError("Arithmetic overflow");
return a * b;
}
int operatorDivide(int a, int b) {
if (b == 0)
throw new AssemblyError("Divide by zero");
return a / b;
}
int operatorModulus(int a, int b) {
if (b == 0)
throw new AssemblyError("Divide by zero");
return a % b;
}
int operatorAnd(int a, int b) {
return a & b;
}
int operatorOr(int a, int b) {
return a | b;
}
int operatorXor(int a, int b) {
return a ^ b;
}
int operatorEqual(int a, int b) {
return a == b;
}
int operatorNotEqual(int a, int b) {
return a != b;
}
int operatorLess(int a, int b) {
return a < b;
}
int operatorGreater(int a, int b) {
return a > b;
}
int operatorLessEqual(int a, int b) {
return a <= b;
}
int operatorGreaterEqual(int a, int b) {
return a >= b;
}
int operatorShiftLeft(int a, int b) {
if (b < 0)
return operatorShiftRight(a, -b);
if (a != 0 && b >= 32)
throw new AssemblyError("Arithmetic overflow");
long r = cast(long) a << b;
if (r & 0xffffffff00000000L)
throw new AssemblyError("Arithmetic overflow");
return a << b;
}
int operatorShiftRight(int a, int b) {
if (b < 0)
return operatorShiftLeft(a, -b);
if (b >= 32)
b = 31;
return a >> b;
}
int operatorLogicalAnd(int a, int b) {
return a && b;
}
int operatorLogicalOr(int a, int b) {
return a || b;
}
void pushValOp(int value, OperatorFunction func, int priority) {
ValOp valOp;
valOp.value = value;
valOp.func = func;
valOp.priority = priority;
valOpStack ~= valOp;
}
void readValue() {
assert(valOpStack.length == 0);
unknownInPass1 = false;
int priority = 0;
do {
int operand;
char c = readChar();
switch (c) {
case '[':
priority += 10;
continue;
case '+':
pushValOp(0, &operatorPlus, priority + 8);
continue;
case '-':
pushValOp(0, &operatorMinus, priority + 8);
continue;
case '<':
pushValOp(0, &operatorLow, priority + 8);
continue;
case '>':
pushValOp(0, &operatorHigh, priority + 8);
continue;
case '!':
pushValOp(0, &operatorLogicalNot, priority + 4);
continue;
case '~':
pushValOp(0, &operatorBitwiseNot, priority + 8);
continue;
case '(':
throw new AssemblyError("Use square brackets instead");
case '*':
checkOriginDefined();
operand = origin;
break;
case '#':
if (!repeating)
throw new AssemblyError("'#' is allowed only in repeated lines");
operand = repeatCounter;
break;
case '\'':
case '"':
operand = readChar();
if (operand == c)
readStringChar(c);
readStringChar(c);
if (!eol() && line[column] == '*') {
column++;
operand ^= 0x80;
}
break;
case '^':
switch (readChar()) {
case '0':
operand = option5200 ? 0xc000 : 0xd000;
break;
case '1':
operand = option5200 ? 0xc010 : 0xd010;
break;
case '2':
operand = option5200 ? 0xe800 : 0xd200;
break;
case '3':
if (option5200)
throw new AssemblyError("There's no PIA chip in Atari 5200");
operand = 0xd300;
break;
case '4':
operand = 0xd400;
break;
default:
illegalCharacter();
}
int d = readDigit(16);
if (d < 0)
illegalCharacter();
operand += d;
break;
case '{':
if (inOpcode)
throw new AssemblyError("Nested opcodes not supported");
ValOp[] savedValOpStack = valOpStack;
AddrMode savedAddrMode = addrMode;
bool savedUnknownInPass1 = unknownInPass1;
bool savedInstructionBegin = instructionBegin;
valOpStack.length = 0;
inOpcode = true;
assemblyInstruction(readInstruction());
if (readChar() != '}')
throw new AssemblyError("Missing '}'");
assert(!instructionBegin);
inOpcode = false;
valOpStack = savedValOpStack;
addrMode = savedAddrMode;
unknownInPass1 = savedUnknownInPass1;
instructionBegin = savedInstructionBegin;
operand = value;
break;
case '$':
operand = readNumber(16);
break;
case '%':
operand = readNumber(2);
break;
default:
column--;
if (c >= '0' && c <= '9') {
operand = readNumber(10);
break;
}
string label = readLabel();
if (label is null)
illegalCharacter();
if (label in labelTable) {
Label l = labelTable[label];
operand = l.value;
l.unused = false;
if (pass2) {
if (l.passed) {
if (l.unknownInPass1)
unknownInPass1 = true;
} else {
if (l.unknownInPass1)
throw new AssemblyError("Illegal forward reference");
unknownInPass1 = true;
}
} else {
if (l.unknownInPass1)
unknownInPass1 = true;
}
} else {
if (pass2)
throw new AssemblyError("Undeclared label: " ~ label);
unknownInPass1 = true;
}
break;
}
while (!eol() && line[column] == ']') {
column++;
priority -= 10;
if (priority < 0)
throw new AssemblyError("Unmatched bracket");
}
if (eol()) {
if (priority != 0)
throw new AssemblyError("Unmatched bracket");
pushValOp(operand, &operatorPlus, 1);
} else {
switch (line[column++]) {
case '+':
pushValOp(operand, &operatorAdd, priority + 6);
break;
case '-':
pushValOp(operand, &operatorSubtract, priority + 6);
break;
case '*':
pushValOp(operand, &operatorMultiply, priority + 7);
break;
case '/':
pushValOp(operand, &operatorDivide, priority + 7);
break;
case '%':
pushValOp(operand, &operatorModulus, priority + 7);
break;
case '<':
switch (readChar()) {
case '<':
pushValOp(operand, &operatorShiftLeft, priority + 7);
break;
case '=':
pushValOp(operand, &operatorLessEqual, priority + 5);
break;
case '>':
pushValOp(operand, &operatorNotEqual, priority + 5);
break;
default:
column--;
pushValOp(operand, &operatorLess, priority + 5);
break;
}
break;
case '=':
switch (readChar()) {
default:
column--;
goto case '=';
case '=':
pushValOp(operand, &operatorEqual, priority + 5);
break;
}
break;
case '>':
switch (readChar()) {
case '>':
pushValOp(operand, &operatorShiftRight, priority + 7);
break;
case '=':
pushValOp(operand, &operatorGreaterEqual, priority + 5);
break;
default:
column--;
pushValOp(operand, &operatorGreater, priority + 5);
break;
}
break;
case '!':
switch (readChar()) {
case '=':
pushValOp(operand, &operatorNotEqual, priority + 5);
break;
default:
illegalCharacter();
}
break;
case '&':
switch (readChar()) {
case '&':
pushValOp(operand, &operatorLogicalAnd, priority + 3);
break;
default:
column--;
pushValOp(operand, &operatorAnd, priority + 7);
break;
}
break;
case '|':
switch (readChar()) {
case '|':
pushValOp(operand, &operatorLogicalOr, priority + 2);
break;
default:
column--;
pushValOp(operand, &operatorOr, priority + 6);
break;
}
break;
case '^':
pushValOp(operand, &operatorXor, priority + 6);
break;
default:
column--;
if (priority != 0)
throw new AssemblyError("Unmatched bracket");
pushValOp(operand, &operatorPlus, 1);
break;
}
}
for (;;) {
immutable sp = valOpStack.length - 1;
if (sp <= 0 || valOpStack[sp].priority > valOpStack[sp - 1].priority)
break;
int operand1 = valOpStack[sp - 1].value;
OperatorFunction func1 = valOpStack[sp - 1].func;
valOpStack[sp - 1] = valOpStack[sp];
valOpStack.length = sp;
if (pass2 || !unknownInPass1) { // skip operations if unknown operands
valOpStack[sp - 1].value = func1(operand1, valOpStack[sp - 1].value);
}
}
} while (valOpStack.length != 1 || valOpStack[0].priority != 1);
value = valOpStack[0].value;
valOpStack.length = 0;
}
version (unittest) int testValue(string l) {
line = l;
column = 0;
readValue();
return value;
}
unittest {
assert(testValue("123") == 123);
assert(testValue("$1234abCd") == 0x1234abcd);
assert(testValue("%101") == 5);
assert(testValue("^07") == 0xd007);
assert(testValue("^1f") == 0xd01f);
assert(testValue("^23") == 0xd203);
assert(testValue("^31") == 0xd301);
assert(testValue("^49") == 0xd409);
assert(testValue("!0") == 1);
assert(testValue("<$1234") == 0x34);
assert(testValue(">$1234567") == 0x45);
assert(testValue("1+2") == 3);
assert(testValue("1+2*3") == 7);
assert(testValue("[1+2]*3") == 9);
assert(testValue("{nop}") == 0xea);
assert(testValue("{CLC}+{sec}") == 0x50);
assert(testValue("{Jsr}") == 0x20);
assert(testValue("{bit a:}") == 0x2c);
assert(testValue("{bIt $7d}") == 0x24);
}
void mustBeKnownInPass1() {
if (unknownInPass1)
throw new AssemblyError("Label not defined before");
}
void readWord() {
readValue();
if ((!unknownInPass1 || pass2) && (value < -0xffff || value > 0xffff))
throw new AssemblyError("Value out of range");
}
void readUnsignedWord() {
readWord();
if ((!unknownInPass1 || pass2) && value < 0)
throw new AssemblyError("Value out of range");
}
void readKnownPositive() {
readValue();
mustBeKnownInPass1();
if (value <= 0)
throw new AssemblyError("Value out of range");
}
void optionalIncDec() {
if (eol()) return;
switch (line[column]) {
case '+':
column++;
addrMode += AddrMode.INCREMENT;
return;
case '-':
column++;
addrMode += AddrMode.DECREMENT;
return;
default:
return;
}
}
void readAddrMode() {
readSpaces();
char c = readChar();
switch (c) {
case '@':
addrMode = AddrMode.ACCUMULATOR;
return;
case '#':
case '<':
case '>':
addrMode = AddrMode.IMMEDIATE;
if (inOpcode && line[column] == '}')
return;
readWord();
final switch (c) {
case '#':
break;
case '<':
value &= 0xff;
break;
case '>':
value = (value >> 8) & 0xff;
break;
}
return;
case '(':
if (inOpcode) {
switch (readChar()) {
case ',':
switch (readChar()) {
case 'X':
case 'x':
break;
default:
illegalCharacter();
}
if (readChar() != ')')
throw new AssemblyError("Need parenthesis");
addrMode = AddrMode.INDIRECT_X;
return;
case ')':
if (readChar() == ',') {
switch (readChar()) {
case 'Y':
case 'y':
break;
default:
illegalCharacter();
}
addrMode = AddrMode.INDIRECT_Y;
return;
} else {
column--;
addrMode = AddrMode.INDIRECT;
return;
}
default:
column--;
break;
}
}
readUnsignedWord();
switch (readChar()) {
case ',':
switch (readChar()) {
case 'X':
case 'x':
addrMode = AddrMode.INDIRECT_X;
break;
case '0':
addrMode = cast(AddrMode) (AddrMode.INDIRECT_X + AddrMode.ZERO);
break;
default:
illegalCharacter();
}
if (readChar() != ')')
throw new AssemblyError("Need parenthesis");
return;
case ')':
if (eol()) {
addrMode = AddrMode.INDIRECT;
return;
}
if (line[column] == ',') {
column++;
switch (readChar()) {
case 'Y':
case 'y':
addrMode = AddrMode.INDIRECT_Y;
break;
case '0':
addrMode = cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.ZERO);
break;
default:
illegalCharacter();
}
optionalIncDec();
return;
}
addrMode = AddrMode.INDIRECT;
return;
default:
illegalCharacter();
}
break;
case 'A':
case 'a':
if (!eol() && line[column] == ':') {
column++;
addrMode = AddrMode.ABSOLUTE;
} else {
addrMode = AddrMode.ABS_OR_ZP;
column--;
}
break;
case 'Z':
case 'z':
if (!eol() && line[column] == ':') {
column++;
addrMode = AddrMode.ZEROPAGE;
} else {
addrMode = AddrMode.ABS_OR_ZP;
column--;
}
break;
default:
addrMode = AddrMode.ABS_OR_ZP;
column--;
break;
}
// absolute or zeropage addressing, optionally indexed
if (inOpcode && (addrMode == AddrMode.ABSOLUTE || addrMode == AddrMode.ZEROPAGE)) {
switch (readChar()) {
case '}':
column--;
return;
case ',':
switch (readChar()) {
case 'X':
case 'x':
addrMode += cast(AddrMode) (AddrMode.ABSOLUTE_X - AddrMode.ABSOLUTE);
return;
case 'Y':
case 'y':
addrMode += cast(AddrMode) (AddrMode.ABSOLUTE_Y - AddrMode.ABSOLUTE);
return;
default:
illegalCharacter();
}
return;
default:
column--;
break;
}
}
readUnsignedWord();
if (addrMode == AddrMode.ABS_OR_ZP) {
if (unknownInPass1 || value > 0xff)
addrMode = AddrMode.ABSOLUTE;
else
addrMode = AddrMode.ZEROPAGE;
}
if (eol()) return;
if (line[column] == ',') {
column++;
switch (readChar()) {
case 'X':
case 'x':
addrMode += cast(AddrMode) (AddrMode.ABSOLUTE_X - AddrMode.ABSOLUTE);
optionalIncDec();
return;
case 'Y':
case 'y':
addrMode += cast(AddrMode) (AddrMode.ABSOLUTE_Y - AddrMode.ABSOLUTE);
optionalIncDec();
return;
default:
illegalCharacter();
}
}
}
void readAbsoluteAddrMode() {
if (inOpcode && readChar() == '}') {
column--;
} else {
readAddrMode();
switch (addrMode) {
case AddrMode.ABSOLUTE:
case AddrMode.ZEROPAGE:
break;
default:
illegalAddrMode();
}
}
addrMode = AddrMode.ABSOLUTE;
}
version (unittest) AddrMode testAddrMode(string l) {
line = l;
column = 0;
readAddrMode();
return addrMode;
}
unittest {
assert(testAddrMode(" @") == AddrMode.ACCUMULATOR);
assert(testAddrMode(" #0") == AddrMode.IMMEDIATE);
assert(testAddrMode(" $abc,x-") == cast(AddrMode) (AddrMode.ABSOLUTE_X + AddrMode.DECREMENT));
assert(testAddrMode(" $ab,Y+") == cast(AddrMode) (AddrMode.ZEROPAGE_Y + AddrMode.INCREMENT));
assert(testAddrMode(" (0,x)") == AddrMode.INDIRECT_X);
assert(testAddrMode(" ($ff),Y+") == cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.INCREMENT));
assert(testAddrMode(" ($abcd)") == AddrMode.INDIRECT);
inOpcode = true;
assert(testAddrMode(" a:}") == AddrMode.ABSOLUTE);
assert(testAddrMode(" z:}") == AddrMode.ZEROPAGE);
assert(testAddrMode(" a:,x}") == AddrMode.ABSOLUTE_X);
assert(testAddrMode(" z:,y}") == AddrMode.ZEROPAGE_Y);
assert(testAddrMode(" (,X)}") == AddrMode.INDIRECT_X);
assert(testAddrMode(" (),y}") == AddrMode.INDIRECT_Y);
assert(testAddrMode(" ()}") == AddrMode.INDIRECT);
inOpcode = false;
}
bool inFalseCondition() {
foreach (IfContext ic; ifContexts) {
if (!ic.condition) return true;
}
return false;
}
string makeEscape(string s) {
return s.replace("$", "$$");
}
File openInputFile(string filename) {
if (find(makeSources, filename).empty)
makeSources ~= filename;
try {
return File(filename);
} catch (Exception e) {
throw new AssemblyError(e.msg);
}
}
File openOutputFile(string filename, string msg) {
if (filename == "-")
return stdout;
if (!getOption('q'))
messageStream.writeln(msg);
try {
return File(filename, "wb");
} catch (Exception e) {
throw new AssemblyError(e.msg);
}
}
void ensureListingFileOpen(char letter, string msg) {
if (!listingStream.isOpen) {
string filename = optionParameters[letter - 'a'];
if (filename is null)
filename = sourceFilename.setExtension("lst");
listingStream = openOutputFile(filename, msg);
listingStream.writeln(TITLE);
}
}
void listNibble(int x) {
listingLine[listingColumn++] = cast(char) (x <= 9 ? x + '0' : x + ('A' - 10));
}
void listByte(ubyte x) {
listNibble(x >> 4);
listNibble(x & 0xf);
}
void listWord(ushort x) {
listByte(cast(ubyte) (x >> 8));
listByte(cast(ubyte) x);
}
void listLine() {
if (!optionListing || !getOption('l') || line is null)
return;
assert(pass2);
if (inFalseCondition() && !getOption('c'))
return;
if (getOption('i') && includeLevel > 0)
return;
ensureListingFileOpen('l', "Writing listing file...");
if (currentFilename != lastListedFilename) {
listingStream.writeln("Source: ", currentFilename);
lastListedFilename = currentFilename;
}
int i = 4;
int x = lineNo;
while (x > 0 && i >= 0) {
listingLine[i--] = '0' + x % 10;
x /= 10;
}
while (i >= 0)
listingLine[i--] = ' ';
listingLine[5] = ' ';
while (listingColumn < 32)
listingLine[listingColumn++] = ' ';
listingStream.writefln("%.32s%s", listingLine, line);
}
void listCommentLine() {
if (currentLabel is null)
listingColumn = 6;
else {
assert(!inFalseCondition());
checkOriginDefined();
}
listLine();
}
void listLabelTable() {
if (optionParameters['t' - 'a'] !is null && listingStream.isOpen)
listingStream.close();
ensureListingFileOpen('t', "Writing label table...");
listingStream.writeln("Label table:");
foreach (string name; sort(labelTable.keys)) {
Label l = labelTable[name];
listingStream.write(l.unused ? 'n' : ' ');
listingStream.write(l.unknownInPass1 ? '2' : ' ');
int value = l.value;
char sign = ' ';
if (value < 0) {
sign = '-';
value = -value;
}
listingStream.writefln(
(l.value & 0xffff0000) != 0 ? " %s%08X %s" : " %s%04X %s",
sign, value, name);
}
}
version (unittest) ubyte[] objectBuffer;
void objectByte(ubyte b) {
version (unittest) {
objectBuffer ~= b;
} else {
assert(pass2);
if (!optionObject) return;
if (!objectStream.isOpen)
objectStream = openOutputFile(objectFilename, "Writing object file...");
try {
objectStream.write(cast(char) b);
} catch (Exception e) {
throw new AssemblyError("Error writing object file");
}
}
objectBytes++;
}
void objectWord(ushort w) {
objectByte(cast(ubyte) w);
objectByte(cast(ubyte) (w >> 8));
}
void putByte(ubyte b) {
if (inOpcode) {
if (instructionBegin) {
value = b;
instructionBegin = false;
}
return;
}
if (willSkip) {
assert(!pass2);
willSkip = false;
skipping = true;
}
if (skipping) {
assert(!pass2);
skipOffsets[$ - 1]++;
}
if (instructionBegin) {
repeatOffset = -2;
instructionBegin = false;
}
repeatOffset--;
if (optionFill && loadingOrigin >= 0 && loadingOrigin != loadOrigin) {
if (loadingOrigin > loadOrigin)
throw new AssemblyError("Can't fill from higher to lower memory location");
if (pass2) {
while (loadingOrigin < loadOrigin) {
objectByte(0xff);
loadingOrigin++;
}
}
}
version (unittest) {
objectByte(b);
}
if (pass2) {
version (unittest) {
} else {
objectByte(b);
}
if (listingColumn < 29) {
listByte(b);
listingLine[listingColumn++] = ' ';
} else {
listingLine[29] = '+';
listingColumn = 30;
}
}
if (optionHeaders) {
if (origin < 0)
throw new AssemblyError("No ORG specified");
assert(blockIndex >= 0);
if (!pass2) {
blockEnds[blockIndex] = cast(ushort) loadOrigin;
}
}
if (origin >= 0) {
origin++;
loadingOrigin = ++loadOrigin;
}
}
void putWord(ushort w) {
putByte(cast(ubyte) w);
putByte(cast(ubyte) (w >> 8));
}
void putCommand(ubyte b) {
putByte(b);
if (inOpcode) return;
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.ACCUMULATOR:
break;
case AddrMode.IMMEDIATE:
case AddrMode.ZEROPAGE:
case AddrMode.ZEROPAGE_X:
case AddrMode.ZEROPAGE_Y:
case AddrMode.INDIRECT_X:
case AddrMode.INDIRECT_Y:
if (pass2 && (value < -0xff || value > 0xff))
throw new AssemblyError("Value out of range");
putByte(cast(ubyte) value);
break;
case AddrMode.ABSOLUTE:
case AddrMode.ABSOLUTE_X:
case AddrMode.ABSOLUTE_Y:
case AddrMode.INDIRECT:
putWord(cast(ushort) value);
break;
default:
assert(0);
}
switch (addrMode) {
case cast(AddrMode) (AddrMode.ABSOLUTE_X + AddrMode.INCREMENT):
case cast(AddrMode) (AddrMode.ZEROPAGE_X + AddrMode.INCREMENT):
putByte(0xe8);
break;
case cast(AddrMode) (AddrMode.ABSOLUTE_X + AddrMode.DECREMENT):
case cast(AddrMode) (AddrMode.ZEROPAGE_X + AddrMode.DECREMENT):
putByte(0xca);
break;
case cast(AddrMode) (AddrMode.ABSOLUTE_Y + AddrMode.INCREMENT):
case cast(AddrMode) (AddrMode.ZEROPAGE_Y + AddrMode.INCREMENT):
case cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.INCREMENT):
case cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.INCREMENT + AddrMode.ZERO):
putByte(0xc8);
break;
case cast(AddrMode) (AddrMode.ABSOLUTE_Y + AddrMode.DECREMENT):
case cast(AddrMode) (AddrMode.ZEROPAGE_Y + AddrMode.DECREMENT):
case cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.DECREMENT):
case cast(AddrMode) (AddrMode.INDIRECT_Y + AddrMode.DECREMENT + AddrMode.ZERO):
putByte(0x88);
break;
default:
break;
}
}
void noOpcode() {
if (inOpcode)
throw new AssemblyError("Can't get opcode of this");
}
void directive() {
noOpcode();
if (repeating)
throw new AssemblyError("Can't repeat this directive");
if (sequencing)
throw new AssemblyError("Can't pair this directive");
}
void noRepeatSkipDirective() {
directive();
if (willSkip)
throw new AssemblyError("Can't skip over this");
repeatOffset = 0;
}
void illegalAddrMode() {
throw new AssemblyError("Illegal addressing mode");
}
void addrModeForMove(int move) {
final switch (move) {
case 0:
readAddrMode();
break;
case 1:
value = value1;
addrMode = addrMode1;
break;
case 2:
value = value2;
addrMode = addrMode2;
break;
}
}
void assemblyAccumulator(ubyte b, ubyte prefix, int move) {
addrModeForMove(move);
if (prefix != 0)
putByte(prefix);
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.ACCUMULATOR:
case AddrMode.INDIRECT:
illegalAddrMode();
break;
case AddrMode.IMMEDIATE:
if (b == 0x80) {
// STA #
illegalAddrMode();
}
putCommand(cast(ubyte) (b + 9));
break;
case AddrMode.ABSOLUTE:
putCommand(cast(ubyte) (b + 0xd));
break;
case AddrMode.ZEROPAGE:
putCommand(cast(ubyte) (b + 5));
break;
case AddrMode.ABSOLUTE_X:
putCommand(cast(ubyte) (b + 0x1d));
break;
case AddrMode.ZEROPAGE_X:
putCommand(cast(ubyte) (b + 0x15));
break;
case AddrMode.ZEROPAGE_Y:
addrMode -= 1;
goto case AddrMode.ABSOLUTE_Y;
case AddrMode.ABSOLUTE_Y:
putCommand(cast(ubyte) (b + 0x19));
break;
case AddrMode.INDIRECT_X:
if ((addrMode & AddrMode.ZERO) != 0)
putWord(0x00a2);
putCommand(cast(ubyte) (b + 1));
break;
case AddrMode.INDIRECT_Y:
if ((addrMode & AddrMode.ZERO) != 0)
putWord(0x00a0);
putCommand(cast(ubyte) (b + 0x11));
break;
default:
assert(0);
}
}
void assemblyShift(ubyte b) {
readAddrMode();
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.ACCUMULATOR:
if (b == 0xc0 || b == 0xe0) {
// INC @, DEC @
illegalAddrMode();
}
putByte(cast(ubyte) (b + 0xa));
break;
case AddrMode.ABSOLUTE:
putCommand(cast(ubyte) (b + 0xe));
break;
case AddrMode.ZEROPAGE:
putCommand(cast(ubyte) (b + 6));
break;
case AddrMode.ABSOLUTE_X:
putCommand(cast(ubyte) (b + 0x1e));
break;
case AddrMode.ZEROPAGE_X:
putCommand(cast(ubyte) (b + 0x16));
break;
default:
illegalAddrMode();
}
}
void assemblyCompareIndex(ubyte b) {
readAddrMode();
switch (addrMode) {
case AddrMode.IMMEDIATE:
putCommand(b);
break;
case AddrMode.ABSOLUTE:
putCommand(cast(ubyte) (b + 0xc));
break;
case AddrMode.ZEROPAGE:
putCommand(cast(ubyte) (b + 4));
break;
default:
illegalAddrMode();
}
}
void assemblyLda(int move) {
assemblyAccumulator(0xa0, 0, move);
}
void assemblyLdx(int move) {
addrModeForMove(move);
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.IMMEDIATE:
putCommand(0xa2);
break;
case AddrMode.ABSOLUTE:
putCommand(0xae);
break;
case AddrMode.ZEROPAGE:
putCommand(0xa6);
break;
case AddrMode.ABSOLUTE_Y:
putCommand(0xbe);
break;
case AddrMode.ZEROPAGE_Y:
putCommand(0xb6);
break;
default:
illegalAddrMode();
}
}
void assemblyLdy(int move) {
addrModeForMove(move);
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.IMMEDIATE:
putCommand(0xa0);
break;
case AddrMode.ABSOLUTE:
putCommand(0xac);
break;
case AddrMode.ZEROPAGE:
putCommand(0xa4);
break;
case AddrMode.ABSOLUTE_X:
putCommand(0xbc);
break;
case AddrMode.ZEROPAGE_X:
putCommand(0xb4);
break;
default:
illegalAddrMode();
}
}
void assemblySta(int move) {
assemblyAccumulator(0x80, 0, move);
}
void assemblyStx(int move) {
addrModeForMove(move);
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.ABSOLUTE:
putCommand(0x8e);
break;
case AddrMode.ZEROPAGE:
putCommand(0x86);
break;
case AddrMode.ABSOLUTE_Y:
addrMode += 1;
goto case AddrMode.ZEROPAGE_Y;
case AddrMode.ZEROPAGE_Y:
putCommand(0x96);
break;
default:
illegalAddrMode();
}
}
void assemblySty(int move) {
addrModeForMove(move);
switch (addrMode & AddrMode.STANDARD_MASK) {
case AddrMode.ABSOLUTE:
putCommand(0x8c);
break;
case AddrMode.ZEROPAGE:
putCommand(0x84);
break;
case AddrMode.ABSOLUTE_X:
addrMode += 1;
goto case AddrMode.ZEROPAGE_X;
case AddrMode.ZEROPAGE_X:
putCommand(0x94);
break;
default:
illegalAddrMode();
}
}
void assemblyBit() {
readAddrMode();
switch (addrMode) {
case AddrMode.ABSOLUTE:
putCommand(0x2c);
break;
case AddrMode.ZEROPAGE:
putCommand(0x24);
break;
default:
illegalAddrMode();
}
}
void putJump() {
switch (addrMode) {
case AddrMode.ZEROPAGE:
addrMode = AddrMode.ABSOLUTE;
goto case AddrMode.ABSOLUTE;
case AddrMode.ABSOLUTE:
putCommand(0x4c);
break;
case AddrMode.INDIRECT:
if (pass2 && (value & 0xff) == 0xff)
warning("Buggy indirect jump");
putCommand(0x6c);
break;
default:
illegalAddrMode();
}
}
void assemblyJmp() {
readAddrMode();
putJump();
}
void assemblyConditionalJump(ubyte b) {
noOpcode();
readAddrMode();
if ((addrMode == AddrMode.ABSOLUTE || addrMode == AddrMode.ZEROPAGE)
&& pass2 && origin >= 0 && value - origin - 2 >= -0x80 && value - origin - 2 <= 0x7f) {
warning("Plain branch instruction would be sufficient");
}
putByte(b);
putByte(3);
putJump();
}
void assemblyJsr() {
readAbsoluteAddrMode();
putCommand(0x20);
}
ubyte calculateBranch(int offset) {
if (offset < -0x80 || offset > 0x7f) {
int dist = offset < 0 ? -offset - 0x80 : offset - 0x7f;
throw new AssemblyError("Branch out of range by " ~ to!string(dist) ~ " bytes");
}
return cast(ubyte) offset;
}
void assemblyBranch(ubyte b) {
readAbsoluteAddrMode();
if (inOpcode) {
putByte(b);
return;
}
checkOriginDefined();
putByte(b);
putByte(pass2 ? calculateBranch(value - origin - 1) : 0);
}
void assemblyRepeat(ubyte b) {
noOpcode();
int offset = repeatOffset;
if (offset >= 0)
throw new AssemblyError("No instruction to repeat");
if (pass2 && wereManyInstructions)
warning("Repeating only the last instruction");
putByte(b);
putByte(calculateBranch(offset));
}
void assemblySkip(ubyte b) {
noOpcode();
if (willSkip) {
skipOffsets[$ - 1] = 2;
willSkip = false;
}
putByte(b);
if (pass2)
putByte(calculateBranch(skipOffsets[skipOffsetsIndex++]));
else {
putByte(0);
skipOffsets ~= 0;
willSkip = true;
}
}
void assemblyInw() {
noOpcode();
readAddrMode();
switch (addrMode) {
case AddrMode.ABSOLUTE:
putCommand(0xee);
putWord(0x03d0);
value++;
putCommand(0xee);
break;
case AddrMode.ZEROPAGE:
putCommand(0xe6);
putWord(0x02d0);
value++;
putCommand(0xe6);
break;
case AddrMode.ABSOLUTE_X:
putCommand(0xfe);
putWord(0x03d0);
value++;
putCommand(0xfe);
break;
case AddrMode.ZEROPAGE_X:
putCommand(0xf6);
putWord(0x02d0);
value++;
putCommand(0xf6);
break;
default:
illegalAddrMode();
}
}
void assemblyMove() {
noOpcode();
readAddrMode();
value1 = value;
addrMode1 = addrMode;
bool unknown1 = unknownInPass1;
readAddrMode();
value2 = value;
addrMode2 = addrMode;
unknownInPass1 = unknown1;
}
void assemblyMoveByte(MoveFunction load, MoveFunction store) {
assemblyMove();
load(1);
store(2);
}
void assemblyMoveWord(MoveFunction load, MoveFunction store, ubyte inc, ubyte dec) {
assemblyMove();
switch (addrMode2) {
case AddrMode.ABSOLUTE:
case AddrMode.ZEROPAGE:
case AddrMode.ABSOLUTE_X:
case AddrMode.ZEROPAGE_X:
case AddrMode.ABSOLUTE_Y:
case AddrMode.ZEROPAGE_Y:
break;
default:
illegalAddrMode();
}
switch (addrMode1) {
case AddrMode.IMMEDIATE:
int high = value1 >> 8 & 0xff;
value1 &= 0xff;
load(1);
store(2);
value2++;
if (unknownInPass1) {
value1 = high;
load(1);
} else {
if (inc != 0 && (value1 + 1 & 0xff) == high)
putByte(inc);
else if (dec != 0 && (value1 - 1 & 0xff) == high)
putByte(dec);
else if (value1 != high) {
value1 = high;
load(1);
}
}
store(2);
break;
case AddrMode.ABSOLUTE:
case AddrMode.ZEROPAGE:
case AddrMode.ABSOLUTE_X:
case AddrMode.ZEROPAGE_X:
case AddrMode.ABSOLUTE_Y:
case AddrMode.ZEROPAGE_Y:
load(1);
store(2);
value1++;
value2++;
load(1);
store(2);
break;
default:
illegalAddrMode();
}
}
void storeDtaNumber(int val, char letter) {
int limit = 0xffff;
if (letter == 'b') limit = 0xff;
if ((!unknownInPass1 || pass2) && (val < -limit || val > limit))
throw new AssemblyError("Value out of range");
final switch (letter) {
case 'a':
putWord(cast(ushort) val);
break;
case 'b':
case 'l':
putByte(cast(ubyte) val);
break;
case 'h':
putByte(cast(ubyte) (val >> 8));
break;
}
}
void assemblyDtaInteger(char letter) {
if (readFunction() == "SIN") {
readWord();
int sinCenter = value;
readComma();
readWord();
int sinAmp = value;
readComma();
readKnownPositive();
int sinPeriod = value;
int sinMin = 0;
int sinMax = sinPeriod - 1;
switch (readChar()) {
case ')':
break;
case ',':
readUnsignedWord();
mustBeKnownInPass1();
sinMin = value;
readComma();
readUnsignedWord();
mustBeKnownInPass1();
sinMax = value;
if (readChar() != ')') {
illegalCharacter();
}
break;
default:
illegalCharacter();
}
while (sinMin <= sinMax) {
int val = sinCenter + cast(int) rint(sinAmp * sin(sinMin * 2 * PI / sinPeriod));
storeDtaNumber(val, letter);
sinMin++;
}
return;
}
readWord();
storeDtaNumber(value, letter);
}
bool realSign;
int realExponent;
long realMantissa;
void putReal() {
if (realMantissa == 0) {
putWord(0);
putWord(0);
putWord(0);
return;
}
while (realMantissa < 0x1000000000L) {
realMantissa <<= 4;
realExponent--;
}
if ((realExponent & 1) != 0) {
if (realMantissa & 0xf)
throw new AssemblyError("Out of precision");
realMantissa >>= 4;
}
realExponent = (realExponent + 0x89) >> 1;
if (realExponent < 64 - 49)
throw new AssemblyError("Out of precision");
if (realExponent > 64 + 49)
throw new AssemblyError("Number too big");
putByte(cast(ubyte) (realSign ? realExponent + 0x80 : realExponent));
putByte(cast(ubyte) (realMantissa >> 32));
putByte(cast(ubyte) (realMantissa >> 24));
putByte(cast(ubyte) (realMantissa >> 16));
putByte(cast(ubyte) (realMantissa >> 8));
putByte(cast(ubyte) realMantissa);
}
bool readSign() {
switch (readChar()) {
case '+':
return false;
case '-':
return true;
default:
column--;
return false;
}
}
void readExponent() {
bool sign = readSign();
char c = readChar();
if (c < '0' || c > '9')
illegalCharacter();
int e = c - '0';
c = readChar();
if (c >= '0' && c <= '9')
e = 10 * e + c - '0';
else
column--;
realExponent += sign ? -e : e;
putReal();
}
void readFraction() {
for (;;) {
char c = readChar();
if (c >= '0' && c <= '9') {
if (c != '0' && realMantissa >= 0x1000000000L)
throw new AssemblyError("Out of precision");
realMantissa <<= 4;
realMantissa += c - '0';
realExponent--;
continue;
}
if (c == 'E' || c == 'e') {
readExponent();
return;
}
column--;
putReal();
return;
}
}
void assemblyDtaReal() {
realSign = readSign();
realExponent = 0;
realMantissa = 0;
char c = readChar();
if (c == '.') {
readFraction();
return;
}
if (c < '0' || c > '9')
illegalCharacter();
do {
if (realMantissa < 0x1000000000L) {
realMantissa <<= 4;
realMantissa += c - '0';
} else {
if (c != '0')
throw new AssemblyError("Out of precision");
realExponent++;
}
c = readChar();
} while (c >= '0' && c <= '9');
switch (c) {
case '.':
readFraction();
break;
case 'E':
case 'e':
readExponent();
break;
default:
column--;
putReal();
break;
}
}
void assemblyDtaNumbers(char letter) {
if (eol() || line[column] != '(') {
column--;
assemblyDtaInteger('b');
return;
}
column++;
for (;;) {
final switch (letter) {
case 'a':
case 'b':
case 'h':
case 'l':
assemblyDtaInteger(letter);
break;
case 'r':
assemblyDtaReal();
break;
}
switch (readChar()) {
case ')':
return;
case ',':
break;
default:
illegalCharacter();
}
}
}
void assemblyDta() {
noOpcode();
readSpaces();
for (;;) {
switch (readChar()) {
case 'A':
case 'a':
assemblyDtaNumbers('a');
break;
case 'B':
case 'b':
assemblyDtaNumbers('b');
break;
case 'C':
case 'c':
ubyte[] s = readString();
if (s is null) {
column--;
assemblyDtaInteger('b');
break;
}
foreach (ubyte b; s) {
putByte(b);
}
break;
case 'D':
case 'd':
ubyte[] s = readString();
if (s is null) {
column--;
assemblyDtaInteger('b');
break;
}
foreach (ubyte b; s) {
final switch (b & 0x60) {
case 0x00:
putByte(cast(ubyte) (b + 0x40));
break;
case 0x20:
case 0x40:
putByte(cast(ubyte) (b - 0x20));
break;
case 0x60:
putByte(b);
break;
}
}
break;
case 'H':
case 'h':
assemblyDtaNumbers('h');
break;
case 'L':
case 'l':
assemblyDtaNumbers('l');
break;
case 'R':
case 'r':
assemblyDtaNumbers('r');
break;
default:
column--;
assemblyDtaInteger('b');
break;
}
if (eol() || line[column] != ',')
break;
column++;
}
}
void assemblyEqu() {
directive();
if (currentLabel is null)
throw new AssemblyError("Label name required");
currentLabel.value = 0;
readSpaces();
readValue();
currentLabel.value = value;
currentLabel.unknownInPass1 = unknownInPass1;
if (optionListing) {
listingLine[6] = '=';
int val = value;
listingLine[7] = ' ';
if (val < 0) {
listingLine[7] = '-';
val = -val;
}
listingColumn = 8;
if ((val & 0xffff0000) != 0)
listWord(cast(ushort) (val >> 16));
else {
while (listingColumn < 12)
listingLine[listingColumn++] = ' ';
}
listWord(cast(ushort) val);
}
}
void assemblyEnd() {
directive();
assert(!foundEnd);
foundEnd = true;
}
void assemblyIftEli() {
ifContexts[$ - 1].condition = true;
if (!inFalseCondition()) {
readSpaces();
readValue();
mustBeKnownInPass1();
if (value != 0)
ifContexts[$ - 1].aConditionMatched = true;
else
listLine();
ifContexts[$ - 1].condition = value != 0;
}
}
void checkMissingIft() {
if (ifContexts.length == 0)
throw new AssemblyError("Missing IFT");
}
void assemblyIft() {
directive();
ifContexts.length++;
assemblyIftEli();
}
void assemblyEliEls() {
directive();
checkMissingIft();
if (ifContexts[$ - 1].wasElse)
throw new AssemblyError("EIF expected");
}
void assemblyEli() {
assemblyEliEls();
if (ifContexts[$ - 1].aConditionMatched) {
ifContexts[$ - 1].condition = false;
return;
}
assemblyIftEli();
}
void assemblyEls() {
assemblyEliEls();
with (ifContexts[$ - 1]) {
if (condition && aConditionMatched)
listLine();
wasElse = true;
condition = !aConditionMatched;
}
}
void assemblyEif() {
directive();
checkMissingIft();
ifContexts.length--;
}
void assemblyErt() {
directive();
readSpaces();
readValue();
if (pass2 && value != 0)
throw new AssemblyError("User-defined error");
}
bool readOption() {
switch (readChar()) {
case '-':
return false;
case '+':
return true;
default:
illegalCharacter();
}
assert(0);
}
void assemblyOpt() {
directive();
readSpaces();
while (!eol()) {
switch (line[column++]) {
case 'F':
case 'f':
optionFill = readOption();
break;
case 'G':
case 'g':
option5200 = readOption();
break;
case 'H':
case 'h':
optionHeaders = readOption();
break;
case 'L':
case 'l':
optionListing = readOption() && pass2;
break;
case 'O':
case 'o':
optionObject = readOption();
break;
case 'U':
case 'u':
optionUnusedLabels = readOption();
break;
case '?':
if (!readOption())
throw new AssemblyError("OPT ?- not supported");
break;
default:
column--;
return;
}
}
}
void originWord(ushort value, char listingChar) {
objectWord(value);
listWord(value);
listingLine[listingColumn++] = listingChar;
}
OrgModifier readOrgModifier() {
readSpaces();
if (column + 2 < line.length && line[column + 1] == ':') {
switch (line[column]) {
case 'F':
case 'f':
checkHeadersOn();
column += 2;
return OrgModifier.FORCE_FFFF;
case 'A':
case 'a':
checkHeadersOn();
column += 2;
return OrgModifier.FORCE_HEADER;
case 'R':
case 'r':
column += 2;
return OrgModifier.RELOCATE;
default:
break;
}
}
return OrgModifier.NONE;
}
void setOrigin(int addr, OrgModifier modifier) {
origin = loadOrigin = addr;
bool requestedHeader = modifier != OrgModifier.NONE;
if (requestedHeader || loadingOrigin < 0 || (addr != loadingOrigin && !optionFill)) {
blockIndex++;
if (!pass2) {
assert(blockIndex == blockEnds.length);
blockEnds ~= cast(ushort) (addr - 1);
}
if (pass2 && optionHeaders) {
if (addr - 1 == blockEnds[blockIndex]) {
if (requestedHeader)
throw new AssemblyError("Cannot generate an empty block");
return;
}
if (modifier == OrgModifier.FORCE_FFFF || objectBytes == 0) {
assert(requestedHeader || addr != loadingOrigin);
originWord(0xffff, '>');
listingLine[listingColumn++] = ' ';
}
if (requestedHeader || addr != loadingOrigin) {
originWord(cast(ushort) addr, '-');
originWord(blockEnds[blockIndex], '>');
listingLine[listingColumn++] = ' ';
loadingOrigin = -1;
}
}
}
}
void checkHeadersOn() {
if (!optionHeaders)
throw new AssemblyError("Illegal when Atari file headers disabled");
}
void assemblyOrg() {
noRepeatSkipDirective();
OrgModifier modifier = readOrgModifier();
readUnsignedWord();
mustBeKnownInPass1();
if (modifier == OrgModifier.RELOCATE) {
checkOriginDefined();
origin = value;
}
else {
setOrigin(value, modifier);
}
}
void assemblyRunIni(ushort addr) {
noRepeatSkipDirective();
checkHeadersOn();
loadingOrigin = -1; // don't fill
OrgModifier modifier = readOrgModifier();
if (modifier == OrgModifier.RELOCATE)
throw new AssemblyError("r: invalid here");
setOrigin(addr, modifier);
readUnsignedWord();
putWord(cast(ushort) (value));
loadingOrigin = -1; // don't fill
}
void assemblyIcl() {
directive();
string filename = readFilename();
checkNoExtraCharacters();
listLine();
includeLevel++;
assemblyFile(filename);
includeLevel--;
line = null;
}
void assemblyIns() {
string filename = readFilename();
int offset = 0;
int length = -1;
if (!eol() && line[column] == ',') {
column++;
readValue();
mustBeKnownInPass1();
offset = value;
if (!eol() && line[column] == ',') {
column++;
readKnownPositive();
length = value;
}
}
File stream = openInputFile(filename);
try {
stream.seek(offset, offset >= 0 ? SEEK_SET : SEEK_END);
} catch (Exception e) {
throw new AssemblyError("Error seeking file");
}
if (inOpcode)
length = 1;
while (length != 0) {
ubyte[1] b;
if (stream.rawRead(b).length != 1) {
if (length > 0)
throw new AssemblyError("File is too short");
break;
}
putByte(b[0]);
if (length > 0) length--;
}
}
void assemblyInstruction(string instruction) {
if (!inOpcode && origin < 0 && currentLabel !is null && instruction != "EQU")
throw new AssemblyError("No ORG specified");
instructionBegin = true;
switch (instruction) {
case "ADC":
assemblyAccumulator(0x60, 0, 0);
break;
case "ADD":
assemblyAccumulator(0x60, 0x18, 0);
break;
case "AND":
assemblyAccumulator(0x20, 0, 0);
break;
case "ASL":
assemblyShift(0x00);
break;
case "BCC":
assemblyBranch(0x90);
break;
case "BCS":
assemblyBranch(0xb0);
break;
case "BEQ":
assemblyBranch(0xf0);
break;
case "BIT":
assemblyBit();
break;
case "BMI":
assemblyBranch(0x30);
break;
case "BNE":
assemblyBranch(0xd0);
break;
case "BPL":
assemblyBranch(0x10);
break;
case "BRK":
putByte(0x00);
break;
case "BVC":
assemblyBranch(0x50);
break;
case "BVS":
assemblyBranch(0x70);
break;
case "CLC":
putByte(0x18);
break;
case "CLD":
putByte(0xd8);
break;
case "CLI":
putByte(0x58);
break;
case "CLV":
putByte(0xb8);
break;
case "CMP":
assemblyAccumulator(0xc0, 0, 0);
break;
case "CPX":
assemblyCompareIndex(0xe0);
break;
case "CPY":
assemblyCompareIndex(0xc0);
break;
case "DEC":
assemblyShift(0xc0);
break;
case "DEX":
putByte(0xca);
break;
case "DEY":
putByte(0x88);
break;
case "DTA":
assemblyDta();
break;
case "EIF":
assemblyEif();
break;
case "ELI":
assemblyEli();
break;
case "ELS":
assemblyEls();
break;
case "END":
assemblyEnd();
break;
case "EOR":
assemblyAccumulator(0x40, 0, 0);
break;
case "EQU":
assemblyEqu();
break;
case "ERT":
assemblyErt();
break;
case "ICL":
assemblyIcl();
break;
case "IFT":
assemblyIft();
break;
case "INC":
assemblyShift(0xe0);
break;
case "INI":
assemblyRunIni(0x2e2);
break;
case "INS":
assemblyIns();
break;
case "INX":
putByte(0xe8);
break;
case "INY":
putByte(0xc8);
break;
case "INW":
assemblyInw();
break;
case "JCC":
assemblyConditionalJump(0xb0);
break;
case "JCS":
assemblyConditionalJump(0x90);
break;
case "JEQ":
assemblyConditionalJump(0xd0);
break;
case "JMI":
assemblyConditionalJump(0x10);
break;
case "JMP":
assemblyJmp();
break;
case "JNE":
assemblyConditionalJump(0xf0);
break;
case "JPL":
assemblyConditionalJump(0x30);
break;
case "JSR":
assemblyJsr();
break;
case "JVC":
assemblyConditionalJump(0x70);
break;
case "JVS":
assemblyConditionalJump(0x50);
break;
case "LDA":
assemblyLda(0);
break;
case "LDX":
assemblyLdx(0);
break;
case "LDY":
assemblyLdy(0);
break;
case "LSR":
assemblyShift(0x40);
break;
case "MVA":
assemblyMoveByte(&assemblyLda, &assemblySta);
break;
case "MVX":
assemblyMoveByte(&assemblyLdx, &assemblyStx);
break;
case "MVY":
assemblyMoveByte(&assemblyLdy, &assemblySty);
break;
case "MWA":
assemblyMoveWord(&assemblyLda, &assemblySta, 0, 0);
break;
case "MWX":
assemblyMoveWord(&assemblyLdx, &assemblyStx, 0xe8, 0xca);
break;
case "MWY":
assemblyMoveWord(&assemblyLdy, &assemblySty, 0xc8, 0x88);
break;
case "NOP":
putByte(0xea);
break;
case "OPT":
assemblyOpt();
break;
case "ORA":
assemblyAccumulator(0x00, 0, 0);
break;
case "ORG":
assemblyOrg();
break;
case "PHA":
putByte(0x48);
break;
case "PHP":
putByte(0x08);
break;
case "PLA":
putByte(0x68);
break;
case "PLP":
putByte(0x28);
break;
case "RCC":
assemblyRepeat(0x90);
break;
case "RCS":
assemblyRepeat(0xb0);
break;
case "REQ":
assemblyRepeat(0xf0);
break;
case "RMI":
assemblyRepeat(0x30);
break;
case "RNE":
assemblyRepeat(0xd0);
break;
case "ROL":
assemblyShift(0x20);
break;
case "ROR":
assemblyShift(0x60);
break;
case "RPL":
assemblyRepeat(0x10);
break;
case "RTI":
putByte(0x40);
break;
case "RTS":
putByte(0x60);
break;
case "RUN":
assemblyRunIni(0x2e0);
break;
case "RVC":
assemblyRepeat(0x50);
break;
case "RVS":
assemblyRepeat(0x70);
break;
case "SBC":
assemblyAccumulator(0xe0, 0, 0);
break;
case "SCC":
assemblySkip(0x90);
break;
case "SCS":
assemblySkip(0xb0);
break;
case "SEC":
putByte(0x38);
break;
case "SED":
putByte(0xf8);
break;
case "SEI":
putByte(0x78);
break;
case "SEQ":
assemblySkip(0xf0);
break;
case "SMI":
assemblySkip(0x30);
break;
case "SNE":
assemblySkip(0xd0);
break;
case "SPL":
assemblySkip(0x10);
break;
case "STA":
assemblySta(0);
break;
case "STX":
assemblyStx(0);
break;
case "STY":
assemblySty(0);
break;
case "SUB":
assemblyAccumulator(0xe0, 0x38, 0);
break;
case "SVC":
assemblySkip(0x50);
break;
case "SVS":
assemblySkip(0x70);
break;
case "TAX":
putByte(0xaa);
break;
case "TAY":
putByte(0xa8);
break;
case "TSX":
putByte(0xba);
break;
case "TXA":
putByte(0x8a);
break;
case "TXS":
putByte(0x9a);
break;
case "TYA":
putByte(0x98);
break;
default:
throw new AssemblyError("Illegal instruction");
}
skipping = false;
}
version (unittest) ubyte[] testInstruction(string l) {
objectBuffer.length = 0;
line = l;
column = 0;
assemblyInstruction(readInstruction());
return objectBuffer;
}
unittest {
assert(testInstruction("nop") == representation(hexString!"ea"));
assert(testInstruction("add (5,0)") == representation(hexString!"18a2006105"));
assert(testInstruction("mwa #$abcd $1234") == representation(hexString!"a9cd8d3412a9ab8d3512"));
assert(testInstruction("mwx #-256 $80") == representation(hexString!"a2008680ca8681"));
assert(testInstruction("dta 5,d'Foo'*,a($4589)") == representation(hexString!"05a6efef8945"));
assert(testInstruction("dta r(1,12,123,1234567890,12345678900000,.5,.03,000.1664534589,1e97)")
== representation(hexString!"400100000000 401200000000 410123000000 441234567890 461234567890 3f5000000000 3f0300000000 3f1664534589 701000000000"));
}
void assemblySequence() {
assert(!inOpcode);
string instruction = readInstruction();
string[] extraInstructions;
while (!eol() && line[column] == ':') {
sequencing = true;
column++;
extraInstructions ~= readInstruction();
}
if (!extraInstructions.empty) {
int savedColumn = column;
if (willSkip)
warning("Skipping only the first instruction");
assemblyInstruction(instruction);
checkNoExtraCharacters();
wereManyInstructions = false;
foreach (nextInstruction; extraInstructions) {
column = savedColumn;
assemblyInstruction(nextInstruction);
wereManyInstructions = true;
}
} else {
sequencing = false;
assemblyInstruction(instruction);
wereManyInstructions = false;
}
}
void assemblyLine() {
lineNo++;
totalLines++;
column = 0;
listingColumn = 6;
if (origin >= 0) {
listWord(cast(ushort) origin);
listingLine[listingColumn++] = ' ';
}
string label = readLabel();
currentLabel = null;
if (label !is null) {
if (!inFalseCondition()) {
if (!label.canFind('?'))
lastGlobalLabel = label;
if (!pass2) {
if (label in labelTable)
throw new AssemblyError("Label declared twice");
currentLabel = new Label(origin);
labelTable[label] = currentLabel;
} else {
assert(label in labelTable);
currentLabel = labelTable[label];
currentLabel.passed = true;
if (currentLabel.unused && getOption('u') && optionUnusedLabels)
warning("Unused label: " ~ label);
}
}
if (eol()) {
listCommentLine();
return;
}
readSpaces();
}
commentOrRep: for (;;) {
if (eol()) {
listCommentLine();
return;
}
switch (line[column]) {
case '\t':
case ' ':
column++;
continue;
case '*':
case ';':
case '|':
listCommentLine();
return;
case ':':
if (inFalseCondition()) {
listCommentLine();
return;
}
column++;
readUnsignedWord();
mustBeKnownInPass1();
int repeatLimit = value;
if (repeatLimit == 0) {
listCommentLine();
return;
}
readSpaces();
repeating = true;
if (repeatLimit == 1)
break;
if (willSkip)
warning("Skipping only the first instruction");
int savedColumn = column;
for (repeatCounter = 0; repeatCounter < repeatLimit; repeatCounter++) {
column = savedColumn;
assemblySequence();
}
checkNoExtraCharacters();
listLine();
wereManyInstructions = true;
return;
default:
repeating = false;
break commentOrRep;
}
}
if (inFalseCondition()) {
switch (readInstruction()) {
case "END":
assemblyEnd();
break;
case "IFT":
assemblyIft();
break;
case "ELI":
assemblyEli();
break;
case "ELS":
assemblyEls();
break;
case "EIF":
assemblyEif();
break;
default:
listCommentLine();
return;
}
checkNoExtraCharacters();
listCommentLine();
return;
}
assemblySequence();
checkNoExtraCharacters();
listLine();
}
void assemblyFile(string filename) {
immutable(ubyte)[] source = sourceFiles.require(filename, {
File stream = stdin;
if (filename != "-") {
filename = filename.defaultExtension("asx");
if (getOption('p'))
filename = absolutePath(filename);
stream = openInputFile(filename);
}
return stream.byChunk(65536).joiner.array.assumeUnique;
}());
string oldFilename = currentFilename;
int oldLineNo = lineNo;
currentFilename = filename;
lineNo = 0;
foundEnd = false;
line = "";
readChar: while (!foundEnd) {
if (source.empty)
break;
ubyte b = source.front;
source.popFront;
switch (b) {
case '\r':
assemblyLine();
line = "";
if (source.empty)
break readChar;
b = source.front;
source.popFront;
if (b != '\n')
line ~= cast(char) b;
break;
case '\n':
case 0x9b:
assemblyLine();
line = "";
break;
default:
line ~= cast(char) b;
break;
}
}
if (!foundEnd)
assemblyLine();
foundEnd = false;
currentFilename = oldFilename;
lineNo = oldLineNo;
}
unittest {
sourceFiles[""] = " lda:sne:ldy:inx $1234".representation;
assemblyFile("");
pass2 = true;
objectBuffer.length = 0;
assemblyFile("");
writefln!"%(%02x%)"(objectBuffer);
assert(objectBuffer == [0xad, 0x34, 0x12, 0xd0, 0x03, 0xac, 0x34, 0x12, 0xe8]);
}
void assemblyPass() {
origin = -1;
loadOrigin = -1;
loadingOrigin = -1;
blockIndex = -1;
optionFill = false;
option5200 = false;
optionHeaders = true;
optionListing = pass2;
optionObject = true;
optionUnusedLabels = true;
willSkip = false;
skipping = false;
repeatOffset = 0;
wereManyInstructions = false;
currentFilename = "command line";
lineNo = 0;
foreach (definition; commandLineDefinitions) {
line = replaceFirst(definition, "=", " equ ");
assemblyLine();
}
line = null;
totalLines = 0;
assemblyFile(sourceFilename);
if (ifContexts.length != 0)
throw new AssemblyError("Missing EIF");
if (willSkip)
throw new AssemblyError("Can't skip over this");
}
pure bool isOption(string arg) {
if (arg.length < 2) return false;
if (arg[0] == '-') return true;
if (arg[0] != '/') return false;
if (arg.length == 2) return true;
if (arg[2] == ':') return true;
return false;
}
void setOption(char letter) {
assert(letter >= 'a' && letter <= 'z');
if (options[letter - 'a']) {
exitCode = 3;
return;
}
options[letter - 'a'] = true;
}
int main(string[] args) {
for (int i = 1; i < args.length; i++) {
string arg = args[i];
if (isOption(arg)) {
char letter = arg[1];
if (letter >= 'A' && letter <= 'Z')
letter += 'a' - 'A';
switch (letter) {
case 'c':
case 'i':
case 'm':
case 'p':
case 'q':
case 'u':
if (arg.length != 2)
exitCode = 3;
setOption(letter);
break;
case 'd':
string definition = null;
if (arg[0] == '/') {
if (arg.length >= 3 && arg[2] == ':')
definition = arg[3 .. $];
} else if (i + 1 < args.length && !isOption(args[i + 1]))
definition = args[++i];
if (definition is null || find(definition, '=').empty)
exitCode = 3;
commandLineDefinitions ~= definition;
break;
case 'l':
case 't':
case 'o':
setOption(letter);
string filename = null;
if (arg[0] == '/') {
if (arg.length >= 3 && arg[2] == ':')
filename = arg[3 .. $];
} else if (i + 1 < args.length && !isOption(args[i + 1]))
filename = args[++i];
if (filename is null && (letter == 'o' || arg.length != 2))
exitCode = 3;
optionParameters[letter - 'a'] = filename;
break;
default:
exitCode = 3;
break;
}
continue;
}
if (sourceFilename !is null)
exitCode = 3;
sourceFilename = arg;
}
version (unittest)
return 0;
else {
if (sourceFilename is null)
exitCode = 3;
objectFilename = optionParameters['o' - 'a'];
if (objectFilename is null) {
objectFilename = sourceFilename == "-"
? "-"
: sourceFilename.setExtension("obx");
}
messageStream = objectFilename == "-" ? stderr : stdout;
if (!getOption('q'))
messageStream.writeln(TITLE);
if (exitCode != 0) {
messageStream.write(
`Syntax: xasm SOURCE [OPTIONS]
-c Include false conditionals in listing
-d LABEL=VALUE Define a label
-i Don't list included files
-l [FILENAME] Generate listing
-o FILENAME Set object file name
-M Print Makefile rule
-p Print absolute paths in listing and error messages
-q Suppress info messages
-t [FILENAME] List label table
-u Warn of unused labels
`);
return exitCode;
}
try {
assemblyPass();
pass2 = true;
assemblyPass();
if (getOption('t') && labelTable.length > 0)
listLabelTable();
} catch (AssemblyError e) {
warning(e.msg, true);
exitCode = 2;
if (objectFilename != "-") {
objectStream.close();
core.stdc.stdio.remove(toStringz(objectFilename));
}
}
if (listingStream != stdout)
listingStream.close();
if (objectStream != stdout)
objectStream.close();
if (exitCode <= 1) {
if (!getOption('q')) {
messageStream.writefln("%d lines of source assembled", totalLines);
if (objectBytes > 0)
messageStream.writefln("%d bytes written to the object file", objectBytes);
}
if (getOption('m')) {
messageStream.writef("%s:", makeEscape(objectFilename));
foreach (filename; makeSources)
messageStream.writef(" %s", makeEscape(filename));
messageStream.write("\n\txasm");
for (int i = 1; i < args.length; i++) {
string arg = args[i];
if (isOption(arg)) {
char letter = arg[1];
if (letter >= 'A' && letter <= 'Z')
letter += 'a' - 'A';
switch (letter) {
case 'm':
break;
case 'o':
if (arg[0] == '/')
messageStream.writef(" /%c:$@", arg[1]);
else {
messageStream.writef(" -%c $@", arg[1]);
++i;
}
break;
default:
if (arg[0] == '-'
&& (letter == 'd' || letter == 'l' || letter == 't')
&& i + 1 < args.length && !isOption(args[i + 1])) {
messageStream.writef(" %s %s", arg, makeEscape(args[++i]));
}
else {
messageStream.writef(" %s", makeEscape(arg));
}
break;
}
continue;
}
messageStream.write(" $<");
}
messageStream.writeln();
}
}
return exitCode;
}
}