diff --git a/engines/execution-controller.ts b/engines/execution-controller.ts index d88cf7d..a087545 100644 --- a/engines/execution-controller.ts +++ b/engines/execution-controller.ts @@ -14,6 +14,8 @@ class ExecutionController { private _engine: LanguageEngine; private _breakpoints: number[] = []; private _result: StepExecutionResult | null; + private _resolvePause: (() => void) | null = null; + private _isPaused: boolean = false; /** * Create a new ExecutionController. @@ -51,16 +53,75 @@ class ExecutionController { this._breakpoints = points; } + /** + * Pause the ongoing execution. + * - If already paused, returns immediately + * - Queues up with any existing pause calls + * @returns Promise that resolves when execution is paused + */ + async pauseExecution(): Promise { + // If already paused, return immediately + if (this._isPaused) return; + + // If there's another "pause" call waiting, chain up with older resolver. + // This kinda creates a linked list of resolvers, with latest resolver at head. + if (this._resolvePause) { + console.log("Chaining pause calls"); + return new Promise((resolve) => { + // Keep a reference to the existing resolver + const oldResolve = this._resolvePause; + // Replace resolver with new chained resolver + this._resolvePause = () => { + oldResolve && oldResolve(); + resolve(); + }; + }); + } + + // Else, create a callback to be called by the execution loop + // when it finishes current execution step. + return new Promise( + (resolve) => + (this._resolvePause = () => { + this._resolvePause = null; + this._isPaused = true; + resolve(); + }) + ); + } + + /** + * Execute the loaded program until stopped. + * @param param0.interval Interval between two execution steps + * @param param0.onResult Callback called with result on each execution step + * @returns Returns last (already used) execution result + */ async executeAll({ interval, onResult }: ExecuteAllArgs) { while (true) { this._result = this._engine.executeStep(); - onResult && onResult(this._result); - if (!this._result.nextStepLocation) break; - await this.sleep(interval || 0); + console.log("Result: ", this._result); + if (!this._result.nextStepLocation) { + // End of program + onResult && onResult(this._result); + this._resolvePause && this._resolvePause(); // In case pause happens on same cycle + break; + } else if (this._resolvePause) { + // Execution has been paused/stopped + this._result.signal = "paused"; + onResult && onResult(this._result); + this._resolvePause(); + break; + } else { + onResult && onResult(this._result); + // Sleep for specified interval + await this.sleep(interval || 0); + } } + return this._result; } + /** Asynchronously sleep for a period of time */ private async sleep(millis: number) { return new Promise((resolve) => setTimeout(resolve, millis)); } diff --git a/engines/types.ts b/engines/types.ts index 57d00c4..3b49d9e 100644 --- a/engines/types.ts +++ b/engines/types.ts @@ -35,6 +35,9 @@ export type StepExecutionResult = { * Passing `null` indicates reaching the end of program. */ nextStepLocation: DocumentRange | null; + + /** Signal if execution has been paused/stopped */ + signal?: "paused"; }; /** diff --git a/engines/worker-constants.ts b/engines/worker-constants.ts index c1ac3a6..cec0a74 100644 --- a/engines/worker-constants.ts +++ b/engines/worker-constants.ts @@ -21,10 +21,19 @@ export type WorkerRequestData = | { type: "Execute"; params: { interval?: number }; + } + | { + type: "Pause"; + params?: null; }; /** Kinds of acknowledgement responses the worker can send */ -export type WorkerAckType = "init" | "reset" | "bp-update" | "prepare"; +export type WorkerAckType = + | "init" // on initialization + | "reset" // on state reset + | "bp-update" // on updating breakpoints + | "prepare" // on preparing for execution + | "pause"; // on pausing execution /** Types of responses the worker can send */ export type WorkerResponseData = diff --git a/engines/worker.ts b/engines/worker.ts index 8042537..94826f1 100644 --- a/engines/worker.ts +++ b/engines/worker.ts @@ -71,11 +71,20 @@ const execute = (interval?: number) => { }); }; -addEventListener("message", (ev: MessageEvent) => { +/** + * Trigger pause in program execution + */ +const pauseExecution = async () => { + await _controller!.pauseExecution(); + postMessage(ackMessage("pause")); +}; + +addEventListener("message", async (ev: MessageEvent) => { if (ev.data.type === "Init") return initController(); if (ev.data.type === "Reset") return resetController(); if (ev.data.type === "Prepare") return prepare(ev.data.params); if (ev.data.type === "Execute") return execute(ev.data.params.interval); + if (ev.data.type === "Pause") return await pauseExecution(); if (ev.data.type === "UpdateBreakpoints") return updateBreakpoints(ev.data.params.points); throw new Error("Invalid worker message type"); diff --git a/ui/MainLayout.tsx b/ui/MainLayout.tsx index 161ffe5..7421d2f 100644 --- a/ui/MainLayout.tsx +++ b/ui/MainLayout.tsx @@ -16,6 +16,7 @@ type Props = { renderRenderer: () => React.ReactNode; renderInput: () => React.ReactNode; renderOutput: () => React.ReactNode; + renderExecControls: () => React.ReactNode; }; export const MainLayout = (props: Props) => { @@ -48,7 +49,9 @@ export const MainLayout = (props: Props) => { path={path} title={WindowTitles[windowId]} - toolbarControls={} + toolbarControls={ + windowId === "editor" ? props.renderExecControls() : + } > {MOSAIC_MAP[windowId]()} diff --git a/ui/Mainframe.tsx b/ui/Mainframe.tsx index 1c26c0f..c71b9a7 100644 --- a/ui/Mainframe.tsx +++ b/ui/Mainframe.tsx @@ -6,6 +6,7 @@ import { useExecController } from "../ui/use-exec-controller"; import { DocumentRange, LanguageProvider } from "../engines/types"; import BrainfuckProvider from "../engines/brainfuck"; import { OutputViewer } from "../ui/output-viewer"; +import { ExecutionControls } from "./execution-controls"; export const Mainframe = () => { const codeEditorRef = React.useRef(null); @@ -20,18 +21,18 @@ export const Mainframe = () => { DocumentRange | undefined >(); - const testDrive = React.useCallback(async () => { - console.info("=== RUNNING TEST DRIVE ==="); - - // Check that controller is ready to execute - const readyStates = ["empty", "ready", "done"]; + /** Reset and begin a new execution */ + const runProgram = async () => { + // Check if controller is free for execution + const readyStates = ["empty", "done"]; if (!readyStates.includes(execController.state)) { console.error(`Controller not ready: state is ${execController.state}`); return; } - // Prepare for execution + // Reset any existing execution state setOutput(""); + setRendererState(null); await execController.resetState(); await execController.prepare( codeEditorRef.current!.getValue(), @@ -39,21 +40,65 @@ export const Mainframe = () => { ); // Begin execution - await execController.executeAll((result) => { + await execController.execute((result) => { setRendererState(result.rendererState); setCodeHighlights(result.nextStepLocation || undefined); setOutput((o) => (o || "") + (result.output || "")); - }, 20); - }, [execController.state]); + }, 1000); + }; - React.useEffect(() => { - const handler = (ev: KeyboardEvent) => { - if (!(ev.ctrlKey && ev.code === "KeyY")) return; - testDrive(); - }; - document.addEventListener("keydown", handler); - return () => document.removeEventListener("keydown", handler); - }, [testDrive]); + /** Pause the ongoing execution */ + const pauseExecution = async () => { + // Check if controller is indeed executing code + if (execController.state !== "processing") { + console.error("Controller not processing any code"); + return; + } + await execController.pauseExecution(); + }; + + /** Resume the currently paused execution */ + const resumeExecution = async () => { + // Check if controller is indeed paused + if (execController.state !== "paused") { + console.error("Controller is not paused"); + return; + } + + // Begin execution + await execController.execute((result) => { + setRendererState(result.rendererState); + setCodeHighlights(result.nextStepLocation || undefined); + setOutput((o) => (o || "") + (result.output || "")); + }, 1000); + }; + + /** Stop the currently active execution */ + const stopExecution = async () => { + // Check if controller has execution + if (!["paused", "processing"].includes(execController.state)) { + console.error("No active execution in controller"); + return; + } + + // If currently processing, pause execution loop first + if (execController.state === "processing") + await execController.pauseExecution(); + + // Reset all execution states + await execController.resetState(); + setOutput(null); + setRendererState(null); + setCodeHighlights(undefined); + }; + + /** Translate execution controller state to debug controls state */ + const getDebugState = () => { + const currState = execController.state; + if (currState === "processing") return "running"; + else if (currState === "paused") return "paused"; + else return "off"; + }; return ( { )} renderInput={() => } renderOutput={() => } + renderExecControls={() => ( + + )} /> ); }; diff --git a/ui/execution-controls.tsx b/ui/execution-controls.tsx new file mode 100644 index 0000000..3942bc4 --- /dev/null +++ b/ui/execution-controls.tsx @@ -0,0 +1,76 @@ +import { Button, ButtonGroup, Icon } from "@blueprintjs/core"; + +const styles = { + container: { + display: "flex", + alignItems: "center", + paddingRight: 5, + }, +}; + +/** Button for starting code execution */ +const RunButton = ({ onClick }: { onClick: () => void }) => ( + +); + +/** Button group for debugging controls */ +const DebugControls = (props: { + paused: boolean; + onPause: () => void; + onResume: () => void; + onStop: () => void; +}) => { + return ( + +