Adapt Befunge to modify code, fix char rendering

This commit is contained in:
Nilay Majorwar 2022-01-26 20:03:06 +05:30
parent 7b7475a4fb
commit 35bebf045f
3 changed files with 78 additions and 6 deletions

View File

@ -1,7 +1,13 @@
import InputStream from "./input-stream"; import InputStream from "./input-stream";
import { DocumentRange, LanguageEngine, StepExecutionResult } from "../types"; import {
DocumentEdit,
DocumentRange,
LanguageEngine,
StepExecutionResult,
} from "../types";
import { ParseError, RuntimeError } from "../worker-errors"; import { ParseError, RuntimeError } from "../worker-errors";
import { Bfg93RS, Bfg93Op, Bfg93Direction } from "./constants"; import { Bfg93RS, Bfg93Op, Bfg93Direction } from "./constants";
import { toSafePrintableChar } from "../engine-utils";
const ROWSIZE = 80; // Maximum size of a single grid row const ROWSIZE = 80; // Maximum size of a single grid row
const COLSIZE = 25; // Maximum size of a single grid column const COLSIZE = 25; // Maximum size of a single grid column
@ -50,6 +56,7 @@ export default class Befunge93LanguageEngine
private _strmode: boolean = DEFAULT_STR_MODE; private _strmode: boolean = DEFAULT_STR_MODE;
private _bounds: CodeBounds = DEFAULT_BOUNDS(); private _bounds: CodeBounds = DEFAULT_BOUNDS();
private _input: InputStream = new InputStream(""); private _input: InputStream = new InputStream("");
private _edits: DocumentEdit[] = [];
resetState() { resetState() {
this._ast = DEFAULT_AST(); this._ast = DEFAULT_AST();
@ -59,6 +66,7 @@ export default class Befunge93LanguageEngine
this._strmode = DEFAULT_STR_MODE; this._strmode = DEFAULT_STR_MODE;
this._bounds = DEFAULT_BOUNDS(); this._bounds = DEFAULT_BOUNDS();
this._input = new InputStream(""); this._input = new InputStream("");
this._edits = [];
} }
validateCode(code: string) { validateCode(code: string) {
@ -67,18 +75,22 @@ export default class Befunge93LanguageEngine
prepare(code: string, input: string) { prepare(code: string, input: string) {
this._ast = this.parseCode(code); this._ast = this.parseCode(code);
this._edits = this.getGridPaddingEdits(code);
this._input = new InputStream(input); this._input = new InputStream(input);
} }
executeStep(): StepExecutionResult<Bfg93RS> { executeStep(): StepExecutionResult<Bfg93RS> {
// Execute and update program counter // Execute and update program counter
let output: string | undefined = undefined; let output: string | undefined = undefined;
let edits: DocumentEdit[] | undefined = undefined;
let end: boolean = false; let end: boolean = false;
if (this._pc.x === -1 && this._pc.y === -1) { if (this._pc.x === -1 && this._pc.y === -1) {
this._pc = { x: 0, y: 0 }; this._pc = { x: 0, y: 0 };
edits = this._edits;
} else { } else {
const result = this.processOp(); const result = this.processOp();
output = result.output; output = result.output;
edits = result.edit && [result.edit];
end = !!result.end; end = !!result.end;
} }
@ -92,7 +104,7 @@ export default class Befunge93LanguageEngine
direction: this._dirn, direction: this._dirn,
strMode: this._strmode, strMode: this._strmode,
}; };
return { rendererState, nextStepLocation, output }; return { rendererState, nextStepLocation, output, codeEdits: edits };
} }
private parseCode(code: string) { private parseCode(code: string) {
@ -133,7 +145,7 @@ export default class Befunge93LanguageEngine
* Also updates stack and pointer states. * Also updates stack and pointer states.
* @returns String to append to output, if any * @returns String to append to output, if any
*/ */
private processOp(): { output?: string; end?: boolean } { private processOp(): { output?: string; end?: boolean; edit?: DocumentEdit } {
const char = this.getGridCell(this._pc.x, this._pc.y); const char = this.getGridCell(this._pc.x, this._pc.y);
if (this._strmode && char !== '"') { if (this._strmode && char !== '"') {
// Push character to string and return; // Push character to string and return;
@ -143,6 +155,7 @@ export default class Befunge93LanguageEngine
} }
let output: string | undefined = undefined; let output: string | undefined = undefined;
let edit: DocumentEdit | undefined = undefined;
let end: boolean = false; let end: boolean = false;
const op = this.charToOp(char); const op = this.charToOp(char);
@ -275,7 +288,7 @@ export default class Befunge93LanguageEngine
const y = this.popStack(); const y = this.popStack();
const x = this.popStack(); const x = this.popStack();
const charCode = this.popStack(); const charCode = this.popStack();
this.setGridCell(x, y, charCode); edit = this.setGridCell(x, y, charCode);
break; break;
} }
case Bfg93Op.STDIN_INT: { case Bfg93Op.STDIN_INT: {
@ -299,7 +312,7 @@ export default class Befunge93LanguageEngine
// Update grid pointer and return // Update grid pointer and return
this.updatePointer(); this.updatePointer();
return { output, end }; return { output, end, edit };
} }
/** Push a number onto the stack */ /** Push a number onto the stack */
@ -327,7 +340,7 @@ export default class Befunge93LanguageEngine
* Set cell at (x, y) of program grid to character with given ASCII value. * Set cell at (x, y) of program grid to character with given ASCII value.
* Throws if (x, y) is out of bounds * Throws if (x, y) is out of bounds
*/ */
private setGridCell(x: number, y: number, asciiVal: number): void { private setGridCell(x: number, y: number, asciiVal: number): DocumentEdit {
if (!this.isInGrid(x, y)) if (!this.isInGrid(x, y))
throw new RuntimeError("Coordinates out of bound"); throw new RuntimeError("Coordinates out of bound");
@ -340,6 +353,12 @@ export default class Befunge93LanguageEngine
// Update grid bounds // Update grid bounds
this._bounds.x[y] = Math.max(this._bounds.x[y], x); this._bounds.x[y] = Math.max(this._bounds.x[y], x);
this._bounds.y[x] = Math.max(this._bounds.y[x], y); this._bounds.y[x] = Math.max(this._bounds.y[x], y);
// Return code edit object
return {
text: toSafePrintableChar(asciiVal),
range: { line: y, charRange: { start: x, end: x + 1 } },
};
} }
/** /**
@ -383,6 +402,36 @@ export default class Befunge93LanguageEngine
} }
} }
/**
* Generate `DocumentEdit`s to apply on code to pad it up to 80x25 size.
* @param code Code content, lines separated by '\n'
* @returns Array of `DocumentEdit`s to apply on code
*/
private getGridPaddingEdits(code: string): DocumentEdit[] {
const lines = code.split("\n");
const edits: DocumentEdit[] = [];
for (let i = 0; i < COLSIZE; ++i) {
if (i < lines.length) {
if (lines[i].length === ROWSIZE) continue;
// Add padding to line upto full-length
edits.push({
range: {
line: i,
charRange: { start: lines[i].length, end: lines[i].length },
},
text: " ".repeat(ROWSIZE - lines[i].length),
});
} else {
// Add full-length empty line
edits.push({
range: { line: i, charRange: { start: 0, end: 0 } },
text: "\n" + " ".repeat(80),
});
}
}
return edits;
}
/** /**
* Cast the given character to corresponding Befunge-93 op. * Cast the given character to corresponding Befunge-93 op.
* If character is invalid op, returns null. * If character is invalid op, returns null.

20
engines/engine-utils.ts Normal file
View File

@ -0,0 +1,20 @@
/**
* For given ASCII code, returns character that is safe to insert into code.
*
* This is useful for self-modifying programs that may insert non-printable characters into
* the source code at runtime. Characters like `\n`, `\r` and `Tab` distort the grid visually
* in the code editor. This function replaces such characters with safely printable alts. Other
* control characters will be safely rendered by the code editor.
*
* @param asciiVal ASCII value to get safe character for
* @returns Character safe to print without distorting code
*/
export const toSafePrintableChar = (asciiVal: number): string => {
// "\n" -> "⤶"
if (asciiVal === 10) return "\u21b5";
// "\r" -> "␍"
else if (asciiVal === 13) return "\u240d";
// Tab -> "⇆"
else if (asciiVal === 9) return "\u21c6";
else return String.fromCharCode(asciiVal);
};

View File

@ -154,6 +154,9 @@ const CodeEditorComponent = (props: Props, ref: React.Ref<CodeEditorRef>) => {
minimap: { enabled: false }, minimap: { enabled: false },
glyphMargin: true, glyphMargin: true,
readOnly: readOnly, readOnly: readOnly,
// Self-modifying programs may add control characters to the code.
// This option ensures such characters are properly displayed.
renderControlCharacters: true,
}} }}
/> />
); );