|
@@ -6,7 +6,7 @@ const fs = require("node:fs");
|
|
* The tape of integers
|
|
* The tape of integers
|
|
* @type {number[]}
|
|
* @type {number[]}
|
|
*/
|
|
*/
|
|
-const tape = [];
|
|
|
|
|
|
+let tape = [];
|
|
/**
|
|
/**
|
|
* The current address of the tape being modified
|
|
* The current address of the tape being modified
|
|
*
|
|
*
|
|
@@ -44,8 +44,8 @@ const ASM_DICTIONARY = {
|
|
"out": insertOutput,
|
|
"out": insertOutput,
|
|
"jnz": insertTestJump.bind(null, OP_CODE_DICTIONARY.JUMP_IF_TRUE),
|
|
"jnz": insertTestJump.bind(null, OP_CODE_DICTIONARY.JUMP_IF_TRUE),
|
|
"jz": insertTestJump.bind(null, OP_CODE_DICTIONARY.JUMP_IF_FALSE),
|
|
"jz": insertTestJump.bind(null, OP_CODE_DICTIONARY.JUMP_IF_FALSE),
|
|
- "slt": instructionNotYetImplemented.bind(null, "SLT"),
|
|
|
|
- "seq": instructionNotYetImplemented.bind(null, "SEQ"),
|
|
|
|
|
|
+ "slt": insertLessThanCheck,
|
|
|
|
+ "seq": insertEqualityCheck,
|
|
"incb": instructionNotYetImplemented.bind(null, "INCB"),
|
|
"incb": instructionNotYetImplemented.bind(null, "INCB"),
|
|
"hlt": insertHalt,
|
|
"hlt": insertHalt,
|
|
};
|
|
};
|
|
@@ -65,6 +65,9 @@ function transpileAssembly(assembly) {
|
|
// Call the function for this instruction, and pass the rest of the line as parameters
|
|
// Call the function for this instruction, and pass the rest of the line as parameters
|
|
ASM_DICTIONARY[instruction](...line);
|
|
ASM_DICTIONARY[instruction](...line);
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ // Clean the values on the tape
|
|
|
|
+ tape = tape.map((inst) => strToDecimal(inst));
|
|
}
|
|
}
|
|
|
|
|
|
/* ASSEMBLY HANDLERS */
|
|
/* ASSEMBLY HANDLERS */
|
|
@@ -76,7 +79,7 @@ function transpileAssembly(assembly) {
|
|
* @param {number} value
|
|
* @param {number} value
|
|
*/
|
|
*/
|
|
function loadToAddress(address, value) {
|
|
function loadToAddress(address, value) {
|
|
- tape[strToDecimal(address)] = strToDecimal(value);
|
|
|
|
|
|
+ tape[strToDecimal(address)] = value;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
@@ -95,10 +98,10 @@ function insertArithmetic(opCode, operand1, operand2, outputDestination) {
|
|
// Add the parameter mode for the two operands
|
|
// Add the parameter mode for the two operands
|
|
opCode = getParameterMode(operand2) + getParameterMode(operand1) + opCode;
|
|
opCode = getParameterMode(operand2) + getParameterMode(operand1) + opCode;
|
|
|
|
|
|
- tape[tapePointer++] = strToDecimal(opCode);
|
|
|
|
- tape[tapePointer++] = strToDecimal(operand1);
|
|
|
|
- tape[tapePointer++] = strToDecimal(operand2);
|
|
|
|
- tape[tapePointer++] = strToDecimal(outputDestination);
|
|
|
|
|
|
+ tape[tapePointer++] = opCode;
|
|
|
|
+ tape[tapePointer++] = operand1;
|
|
|
|
+ tape[tapePointer++] = operand2;
|
|
|
|
+ tape[tapePointer++] = outputDestination;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
/*
|
|
@@ -113,8 +116,8 @@ function insertInput() {
|
|
* @param {string} outputParameter The parameter of the value to output. May be an immediate value or an address
|
|
* @param {string} outputParameter The parameter of the value to output. May be an immediate value or an address
|
|
*/
|
|
*/
|
|
function insertOutput(outputParameter) {
|
|
function insertOutput(outputParameter) {
|
|
- tape[tapePointer++] = strToDecimal(getParameterMode(outputParameter) + OP_CODE_DICTIONARY.OUTPUT);
|
|
|
|
- tape[tapePointer++] = strToDecimal(outputParameter);
|
|
|
|
|
|
+ tape[tapePointer++] = getParameterMode(outputParameter) + OP_CODE_DICTIONARY.OUTPUT;
|
|
|
|
+ tape[tapePointer++] = outputParameter;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
@@ -125,9 +128,45 @@ function insertOutput(outputParameter) {
|
|
* @param {string} jumpDestination The address to jump to if the test passes
|
|
* @param {string} jumpDestination The address to jump to if the test passes
|
|
*/
|
|
*/
|
|
function insertTestJump(opcode, testingParameter, jumpDestination) {
|
|
function insertTestJump(opcode, testingParameter, jumpDestination) {
|
|
- tape[tapePointer++] = strToDecimal(getParameterMode(jumpDestination) + getParameterMode(testingParameter) + opcode);
|
|
|
|
- tape[tapePointer++] = strToDecimal(testingParameter);
|
|
|
|
- tape[tapePointer++] = strToDecimal(jumpDestination);
|
|
|
|
|
|
+ tape[tapePointer++] = getParameterMode(jumpDestination) + getParameterMode(testingParameter) + opcode;
|
|
|
|
+ tape[tapePointer++] = testingParameter;
|
|
|
|
+ tape[tapePointer++] = jumpDestination;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Insert a less than check onto the tape
|
|
|
|
+ *
|
|
|
|
+ * @param {string} operand1 The first parameter to test. Can be either an immediate value or an address
|
|
|
|
+ * @param {string} operand2 The parameter to test against. Can be either an immediate value or an address
|
|
|
|
+ * @param {string} outputDestination The address to write the output to
|
|
|
|
+ */
|
|
|
|
+function insertLessThanCheck(operand1, operand2, outputDestination) {
|
|
|
|
+ if (!isAddress(outputDestination)) {
|
|
|
|
+ throw new WrongParameterModeError();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ tape[tapePointer++] = getParameterMode(operand2) + getParameterMode(operand1) + OP_CODE_DICTIONARY.LESS_THAN;
|
|
|
|
+ tape[tapePointer++] = operand1;
|
|
|
|
+ tape[tapePointer++] = operand2;
|
|
|
|
+ tape[tapePointer++] = outputDestination;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Insert an equality check onto the tape
|
|
|
|
+ *
|
|
|
|
+ * @param {string} operand1 The first parameter to test. Can be either an immediate value or an address
|
|
|
|
+ * @param {string} operand2 The parameter to test equality against. Can be either an immediate value or an address
|
|
|
|
+ * @param {string} outputDestination The address to write the output to
|
|
|
|
+ */
|
|
|
|
+function insertEqualityCheck(operand1, operand2, outputDestination) {
|
|
|
|
+ if (!isAddress(outputDestination)) {
|
|
|
|
+ throw new WrongParameterModeError();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ tape[tapePointer++] = getParameterMode(operand2) + getParameterMode(operand1) + OP_CODE_DICTIONARY.EQUALS;
|
|
|
|
+ tape[tapePointer++] = operand1;
|
|
|
|
+ tape[tapePointer++] = operand2;
|
|
|
|
+ tape[tapePointer++] = outputDestination;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
@@ -179,6 +218,10 @@ function getParameterMode(parameter) {
|
|
* @returns {number}
|
|
* @returns {number}
|
|
*/
|
|
*/
|
|
function strToDecimal(decimalValue) {
|
|
function strToDecimal(decimalValue) {
|
|
|
|
+ if (typeof decimalValue == "number") {
|
|
|
|
+ return decimalValue;
|
|
|
|
+ }
|
|
|
|
+
|
|
return parseInt(decimalValue.replace(/^0d/, ""), 10);
|
|
return parseInt(decimalValue.replace(/^0d/, ""), 10);
|
|
}
|
|
}
|
|
|
|
|